### 13.2.3 REST Client
上一章讲到如何在SpringBoot中提供RESTfull服务,本章将介绍系统之间,如何发起REST请求。
SpringBoot提供了RestTemplate来辅助发起一个REST请求,默认通过JDK自带的HttpURLConnection来作为底层HTTP消息的发送,使用JackSon 来序列化服务器响应的JSON数据
#### 13.2.3.1 RestTemplate
RestTemplate是核心类,提供了所有访问Rest服务的接口,尽管实际上可以使用HTTP Client类或者java.net.URL 来完成,但RestTemplate提供了Restful风格的的API。
RestTemplate有6个主要得方法对应于Restful的6个主要得Http Method。
| HTTP Method | Java API |
| ----------- | ----------------------------- |
| DELETE | delete |
| GET | getForObject,getForEntity |
| HEAD | headForHeaders |
| OPTIONS | optionsForAllow |
| POST | postForObject,postForLocation |
| PUT | put |
| 其他 | exchange(通用) |
SpringBoot 提供了RestTemplateBuilder来创建一个RestTemplat。应用可以通过如下代码来创建一个RestTemplate实例
~~~java
@Autowired
RestTemplateBuilder restTemplateBuilder;
public void foo(){
RestTemplate client = restTemplateBuilder.build();
}
~~~
下面列举一下使用RestTemplate来查询订单和创建订单
~~~java
/*一个测试类*/
@Controller
@RequestMapping("/test")
public class RestClientTestCrontroller {
@Value(value = "${api.order}")
String base ;
@Autowired
RestTemplateBuilder restTemplateBuilder;
@GetMapping("/get/{orderId}")
public @ResponseBody Order testGetOrder(@PathVariable String orderId) throws Exception{
RestTemplate client = restTemplateBuilder.build();
String uri = base+"/order/{orderId}";
//核心代码
Order order = client.getForObject(uri, Order.class,orderId);
return order;
}
}
~~~
base是在配置文件application.properties中配置的订单API地址
~~~properties
api.order=http://127.0.0.1:8080/api/v1
~~~
代码首先构造RestTemplate,然后调用getForObject,此方法接受三个参数,第一个是URI 模板,第二个参数是期望返回的对象,后面是URI模板对应的参数列表。参数列表既可以是数组,也可以是个Map,如上有可以写成
~~~java
Map map = new HashMap();
map.put("orderId",orderId);
Order order = client.getForObject(uri, Order.class,map);
~~~
如果还想获取响应的HTTP头相关信息,可以调用client.getForEntity,此方法返回ResponseEntity,包含了头信息
~~~java
ResponseEntity<Order> responseEntity = client.getForEntity(uri, Order.class, orderId);
Order order = responseEntity.getBody();
HttpHeaders headers = responseEntity.getHeaders();
~~~
添加订单可以使用postForObject方法,此方法接受三个参数,第一个是URI,第二个是Post参数,可以是HttpEntity,或者是某个POJO对象,POJO对象在这种情况下回自动转成HTTPEntity,第三个参数是期望返回的类型,这个例子期望返回类型是String
~~~java
@GetMapping("/addorder")
public @ResponseBody String testAddOrder() throws Exception{
RestTemplate client = restTemplateBuilder.build();
String uri = base+"/order";
Order order = new Order();
order.setName("test");
String ret = client.postForObject(uri, order, String.class);
//{success:true,message:"添加成功"}
return ret;
}
~~~
或者使用HttpEntity
~~~java
HttpEntity<Order> body = new HttpEntity<Order>(order);
String ret = client.postForObject(uri, body, String.class);
~~~
使用HttpEnity的好处是可以提供额外的HTTP头信息。
如果期望返回的类型是一个列表,如List<Order>,不能简单调用xxxForObject,因为存在泛型的类型擦除,RestTemplate在反序列化的时候并不知道实际反序列化的类型,因此可以使用ParameterizedTypeReference来包含泛型类型,代码如下
```java
RestTemplate client = restTemplateBuilder.build();
//根据条件查询一组订单
String uri = base+"/orders?offset={offset}";
Integer offset = 1;
//无参数
HttpEntity body = null;
ParameterizedTypeReference<List<Order>> typeRef = new ParameterizedTypeReference<List<Order>>() {};
ResponseEntity<List<Order>> rs = client.exchange(uri, HttpMethod.GET, body, typeRef, offset);
List<Order> order = rs.getBody();
```
注意到typeRef实际定义是用{}结束,这里实际上创建了一个ParameterizedTypeReference子类,依据在类定义中的泛型信息是保留的原则,typeRef保留了期望返回的泛型List<Order>
exchange 是一个基础的Rest调用接口,除了需要指明HTTP Method外,调用跟其他方法都类似。
> 除了使用ParameterizedTypeReference来保留泛型信息外,也可以通过getForObject方法先映射成String,然后通过ObjectMapper来转为指定类型,可以参考第三章Jackson来了解。
#### 13.2.3.2 定制 RestTemplate
创建一个配置类实现RestTemplateCustomizer接口的customize方法,此方法会
~~~java
@Configuration
public class RestConf implements RestTemplateCustomizer {
public void customize(RestTemplate restTemplate) {
SimpleClientHttpRequestFactory jdkHttp = (SimpleClientHttpRequestFactory)restTemplate.getRequestFactory();
jdkHttp.setConnectTimeout(1000);
}
}
~~~
如上代码,customize方法会定制RestTemplate,上面的代码设置链接超时时间为1000毫秒。Spring Boot因为默认使用了JDK的URLConnection作为底层的HTTP工具,如果想使用了OkHttp。需要添加如下依赖
~~~xml
<dependency>
<groupId>com.squareup.okhttp3</groupId>
<artifactId>okhttp</artifactId>
<version>4.0.1</version>
</dependency>
~~~
那上面的代码应该是
~~~java
public void customize(RestTemplate restTemplate) {
OkHttp3ClientHttpRequestFactory okHttp = (OkHttp3ClientHttpRequestFactory)restTemplate.getRequestFactory();
okHttp.setReadTimeout(5000);
okHttp.setWriteTimeout(3000);
}
~~~
- 再版说明
- 13 REST
- 13.1 REST
- 13.1.1 REST 风格的架构
- 13.1.2 使用api作为上下文
- 13.1.3 增加一个版本标示
- 13.1.4 标识资源
- 13.1.5 确定HTTP METHOD
- 13.1.6 确定HTTP Status
- 13.1.7 REST VS WebService
- 13.2 SpringBoot Rest集成
- 13.2.1 集成REST
- 13.2.2 @RestController
- 13.2.3 REST Client
- 13.3 Swager UI
- 13.3.2 集成Swager UI
- 13.3.3 OpenAPI 规范
- 13.3.4 接口描述
- 13.3.5 查询参数描述
- 13.3.6 URI中的参数
- 13.3.7 HTTP 头参数
- 13.3.8 表单参数
- 13.3.9 整个请求体作为参数
- 14 MongoDB
- 14.1 安装MongoDB
- 14.2 使用shell
- 14.2 .1 指定数据库
- 14.2.2 插入文档
- 14.2.3 查询文档
- 14.2.4 更新操作
- 14.2.5 删除操作
- 14.3 Spring Boot集成
- 14.4 增删改查
- 14.4.1 增加API
- 14.4.2 根据主键查询API
- 14.4.3 查询API
- 14.4.4 修改API
- 14.4.5 删除API
- 14.4.6 使用MongoDatabase
- 14.4.7 打印日志
- 14.5 使用Spring Data
- 15 Redis
- 15.1 安装Redis
- 15.2 使用redis-cli
- 15.2 .1 安全设置
- 15.2.2 基本操作
- 15.2.3 Keys
- 15.2.3 Redis List
- 15.2.4 Redis Hash
- 15.2.5 Set
- 15.2.6 Pub&Sub
- 15.3 Spring Boot 集成
- 15.4 使用StringRedisTemplate
- 15.4.1 opsFor
- 15.4.2 绑定key的操作
- 15.4.3 RedisConnection
- 15.4.4 Pub&Sub
- 15.4.5 Pipeline
- 15.5 序列化策略
- 15.5.1 默认序列化策略
- 15.5.2 自定义序列化策略
- 16 Elastic Search
- 16.1 Elastic Search 介绍
- 16.1.1Elastic 安装
- 16.2.1 创建Index
- 16.2.2 添加文档
- 16.2.3 根据主键查询
- 16.2.4 根据主键更新
- 16.2.5 根据主键删除
- 16.2.6 搜索文档
- 16.2.7 联合多个索引搜索
- 16.2.8 Mapping
- 16.2.9 路由
- 16.3 使用RestTemplate 访问ES
- 16.3.1 创建Book
- 16.3.2 使用RestTemplate 获取搜索结果
- 16.4 Spring Data Elastic
- 16.4.1 安装Spring Data
- 16.4.2 编写entity
- 15.4.3 编写Dao
- 16.4.4 编写Controller
- 17 Cache
- 17.1 关于Cache
- 17.1.1 Cache 组件和概念
- 17.1.2 Cache的单机应用
- 17.1.3 使用专有的Cache服务器
- 17.1.4 使用一二级缓存服务器
- 17.2 Spring Boot Cache
- 17.2.1 集成Spring cache
- 17.3 注释驱动缓存
- 17.3.1 @Cacheable
- 17.3.2 KEY生成器
- 17.3.2 @CachePut
- 17.3.3 @CacheEvict
- 17.3.4 @Caching
- 17.3.5 @CacheConfig
- 17.4 使用Redis Cache
- 17.4.1集成Redis 缓存
- 17.4.2 禁止缓存
- 17.4.3 定制缓存
- 17.4. 5 Redis缓存原理
- 17.5 实现Redis 俩级缓存
- 17.5.1 实现 TwoLevelCacheManager
- 17.5.2 创建RedisAndLocalCache
- 17.5.3 缓存同步说明
- 17.5.4 将代码组合在一起
- 18 Spring Session
- 18.1 水平扩展实现
- 18.2 nginx 安装和配置
- 18.2.1 安装nginx
- 18.2.2 配置nginx
- 18.3 Spring Session
- 18.3.1 Spring Session介绍
- 18.3.2 使用Redis
- 18.3.3 Nginx+Redis
- 19 SpringBoot 和 Zookeeper
- 19.1 Zookeeper
- 19.1.1 zookeeper 数据结构
- 19.1.2 Zookeeper 安装
- 19.1.3 zookeeper 基本命令
- 19.1.4 领导选取演示
- 19.1.5 分布式锁演示
- 19.1.6 服务注册演示
- 19.2 Spring boot 集成zookeeper
- 19.2.1 集成 Curator
- 19.2.2 Curator api
- 19.3 分布式锁实现
- 19.4 服务注册
- 19.4.1 服务注册
- 19.4.2 获取服务
- 19.5 领导选取
- 19.6 屏障
- 20 监控SpringBoot应用
- 20.1 安装Acutator
- 20.2 HTTP 跟踪
- 20.3 日志查看
- 20.4 线程栈信息
- 20.5 内存信息
- 20.6 查看URL映射
- 20.7 查看Spring管理Bean
- 20.8 其他监控
- 20.9 编写自己的监控信息
- 20.9.1 编写HealthIndicator
- 20.9.2 自定义监控。