RestTemplate là mộ trong những thành phần cốt lõi của Spring framework cho phép thực thi các HTTP request và nhận các các response tương ứng. Bản thân RestTemplate là một high-level API cho phép loại bỏ các mã code nhàm chám để cấu hình một java.net.HttpURLConnection sử dụng để gửi và nhận các request/response mà Java SDK cung cấp.
Để hiểu rõ hơn, nếu sử dụng HttpURLConnection để gửi một request thì chúng ta phải thực thi các công việc sau lặp đi lặp lại nhiều lần:
Khởi tạo URL object
Cấu hình HTTP request.
Thực thi HTTP request.
Nhận phản hồi response.
Chuyển đổi HTTP response sang Java Object.
Xử lý exception.
Khi sử dụng RestTemplate, tất cả những công việc trên được chạy ngầm và chúng ta không cần quan tâm đến chúng.
Tuy nhiên, bắt đầu từ phiên bản Spring 5, non-blocking và reactive WebClient được giới thiệu để thay thế RestTemplate. WebClient hỗ trợ synchronous và asynchronous HTTP request và cả các kịch bản streaming. Vì thế RestTemplate sẽ được đánh dấu deprecated trong các phiên bản sau và sẽ không được cập nhật thêm các tính năng mới.
Maven dependency
Để sử dụng RestTemplate trong Spring Boot chúng ta cần thêm các dependency sau:
RestTemplate có thể sử dụng để request đến một Rest API bất kỳ, tuy nhiên để đảm bảo ví dụ có thể hoạt động trong khuôn khổ bài viết này, mình sẽ sử dụng RestTemplate request đến chính server hiện tại.
package com.deft.resttemplate.entity;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Employee {
private long id;
private String firstName;
private String lastName;
private long yearlyIncome;
}
package com.deft.resttemplate.entity;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Employee {
private long id;
private String firstName;
private String lastName;
private long yearlyIncome;
}
package com.deft.resttemplate.entity;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Employee {
private long id;
private String firstName;
private String lastName;
private long yearlyIncome;
}
Trong đó sử dụng mình sử dụng Lombok để giảm bớt các các bước triển khai constructor, getter, setter v.v
Tiếp theo định nghĩa một MockRestAPI định nghĩa các API mà sau đó chúng ta sẽ dùng RestTemplate để request đến.
package com.deft.resttemplate.mockapi;
import com.deft.resttemplate.entity.Employee;
import org.springframework.http.HttpStatus;
import org.springframework.http.MediaType;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.*;
import java.util.HashMap;
import java.util.Map;
import java.util.Optional;
@RestController
@RequestMapping("/mock")
public class MockRestAPI {
private static final Map<Long, Employee> employees;
Tương tự, có 3 cách để thực thị một HTTP Post trong RestTemplate:
postForObject
postForLocation
postForEntity
Trong đó postForObject và postForObject lần lượt nhận về các response là các Object type được chỉ định khi gọi, còn postForLocation chỉ nhận về một URI thay vì Object.
curl -X POST http://localhost:8080/post-for-object
{"id":7,"firstName":"Deft","lastName":"blog","yearlyIncome":2020}
// -------------------------
curl -X POST http://localhost:8080/post-for-location
"/rest/employees/8"
// -------------------------
curl -X POST http://localhost:8080/post-for-entity
{"id":9,"firstName":"Deft","lastName":"blog","yearlyIncome":2020}
curl -X POST http://localhost:8080/post-for-object
{"id":7,"firstName":"Deft","lastName":"blog","yearlyIncome":2020}
// -------------------------
curl -X POST http://localhost:8080/post-for-location
"/rest/employees/8"
// -------------------------
curl -X POST http://localhost:8080/post-for-entity
{"id":9,"firstName":"Deft","lastName":"blog","yearlyIncome":2020}
PUT
HTTP PUT method được dùng để cập nhật một entity cụ thể trong, đều này đồng nghĩa bạn phải gửi một Employee hoàn chỉnh bao gồm các thông tin ban đầu và các thông tin đã được thay đổi.
Để thực thi một PUT HTTP request trong RestTemplate chúng ta có thể sử dụng put() method được hỗ trợ sẵn.
@PutMapping("/put/{id}")
public voidput(@PathVariable Long id){
Employee employee = newEmployee();
employee.setId(id);
employee.setFirstName("Deft");
employee.setLastName("blog");
employee.setYearlyIncome(2021);
restTemplate.put("/" + id, employee);
}
@PutMapping("/put/{id}")
public void put(@PathVariable Long id) {
Employee employee = new Employee();
employee.setId(id);
employee.setFirstName("Deft");
employee.setLastName("blog");
employee.setYearlyIncome(2021);
restTemplate.put("/" + id, employee);
}
@PutMapping("/put/{id}")
public void put(@PathVariable Long id) {
Employee employee = new Employee();
employee.setId(id);
employee.setFirstName("Deft");
employee.setLastName("blog");
employee.setYearlyIncome(2021);
restTemplate.put("/" + id, employee);
}
curl -X PUT http://localhost:8080/put/1
// --------------
curl http://localhost:8080/get-for-object
{"id":1,"firstName":"Deft","lastName":"blog","yearlyIncome":2021
curl -X PUT http://localhost:8080/put/1
// --------------
curl http://localhost:8080/get-for-object
{"id":1,"firstName":"Deft","lastName":"blog","yearlyIncome":2021
Tuy nhiên, nếu chúng ta muốn thực thi PUT method và nhận về response chứa các thông tin được thay đổi và mã trạng thái thực thi thì có thể sử dụng exchange()
@PutMapping("/put-ex/{id}")
public Employee putExchange(@PathVariable Long id){
Employee employee = newEmployee();
employee.setId(id);
employee.setFirstName("Deft");
employee.setLastName("blog");
employee.setYearlyIncome(2021);
return restTemplate.exchange("/" + id,
HttpMethod.PUT,
new HttpEntity<>(employee),
Employee.class,
Long.toString(id)).getBody();
}
@PutMapping("/put-ex/{id}")
public Employee putExchange(@PathVariable Long id) {
Employee employee = new Employee();
employee.setId(id);
employee.setFirstName("Deft");
employee.setLastName("blog");
employee.setYearlyIncome(2021);
return restTemplate.exchange( "/" + id,
HttpMethod.PUT,
new HttpEntity<>(employee),
Employee.class,
Long.toString(id)).getBody();
}
@PutMapping("/put-ex/{id}")
public Employee putExchange(@PathVariable Long id) {
Employee employee = new Employee();
employee.setId(id);
employee.setFirstName("Deft");
employee.setLastName("blog");
employee.setYearlyIncome(2021);
return restTemplate.exchange( "/" + id,
HttpMethod.PUT,
new HttpEntity<>(employee),
Employee.class,
Long.toString(id)).getBody();
}
curl -X PUT http://localhost:8080/put-ex/2
// ---------------
{"id":2,"firstName":"Deft","lastName":"blog","yearlyIncome":2021}
curl -X PUT http://localhost:8080/put-ex/2
// ---------------
{"id":2,"firstName":"Deft","lastName":"blog","yearlyIncome":2021}
DELETE
Trong trường hợp muốn xóa một entity thông qua ID thì chúng ta có thể sử dụng delete() hoặc exchange() tương tự như PUT.
@DeleteMapping("/delete/{id}")
public voiddelete(@PathVariable String id){
restTemplate.delete("/" + id);
}
@DeleteMapping("/delete-ex/{id}")
public ResponseEntity<Void>deleteEx(@PathVariable Long id){
return restTemplate.exchange("/" + id,
HttpMethod.DELETE,
null,
Void.class,
id);
}
@DeleteMapping("/delete/{id}")
public void delete(@PathVariable String id) {
restTemplate.delete("/" + id);
}
@DeleteMapping("/delete-ex/{id}")
public ResponseEntity<Void> deleteEx(@PathVariable Long id) {
return restTemplate.exchange( "/" + id,
HttpMethod.DELETE,
null,
Void.class,
id);
}
@DeleteMapping("/delete/{id}")
public void delete(@PathVariable String id) {
restTemplate.delete("/" + id);
}
@DeleteMapping("/delete-ex/{id}")
public ResponseEntity<Void> deleteEx(@PathVariable Long id) {
return restTemplate.exchange( "/" + id,
HttpMethod.DELETE,
null,
Void.class,
id);
}
Kết bài
RestTemplate trong các dự án spring cũ có thể là lựa chọn tốt. Tuy nhiên nếu bạn bắt đầu một dự án spring từ khi đọc bài viết này, thì nên cân nhắc sử dụng WebClient với nhiều tính năng mới và tối ưu hóa hiệu năng hơn.
Cuối cùng mã nguồn được mình công khai trên gitlab để các bạn tiện theo dõi và kiểm thử: resttemplate
Necessary cookies help make a website usable by enabling basic functions like page navigation and access to secure areas of the website. The website cannot function properly without these cookies.
We do not use cookies of this type.
Marketing cookies are used to track visitors across websites. The intention is to display ads that are relevant and engaging for the individual user and thereby more valuable for publishers and third party advertisers.
We do not use cookies of this type.
Analytics cookies help website owners to understand how visitors interact with websites by collecting and reporting information anonymously.
We do not use cookies of this type.
Preference cookies enable a website to remember information that changes the way the website behaves or looks, like your preferred language or the region that you are in.
We do not use cookies of this type.
Unclassified cookies are cookies that we are in the process of classifying, together with the providers of individual cookies.
We do not use cookies of this type.
Cookies are small text files that can be used by websites to make a user's experience more efficient. The law states that we can store cookies on your device if they are strictly necessary for the operation of this site. For all other types of cookies we need your permission. This site uses different types of cookies. Some cookies are placed by third party services that appear on our pages.