Spring Cloud Alibaba - RestTemplate
Spring Cloud Alibaba - RestTemplate
Controller导入依赖和相关属性
@SuppressWarnings("all")
@RestController
public class TemplateController {
@Autowired
private RestTemplate restTemplate;
@Autowired(required = false)
private LoadBalancerClient loadBalancerClient;
final String serviceName = "http://nacos-discovery-provider"; //可以通过配置文件远程获取
}
Get请求,无参数,返回String
//GET请求,无参数,返回String
@RequestMapping("/getstring")
public String get(){
// ResponseEntity中包含关于这个请求返回的许多信息,不光是结果
ResponseEntity<String> responseEntity = restTemplate.getForEntity(serviceName+"/get",String.class);
//String.class 指定返回类型为String
//====================================================
//开始获取 ResponseEntity中的详细信息
int statusCodeValue = responseEntity.getStatusCodeValue(); //状态码的值
HttpStatus httpStatus = responseEntity.getStatusCode(); //状态码
HttpHeaders httpHeaders = responseEntity.getHeaders(); //头部信息
String body = responseEntity.getBody(); //返回内容体
//====================================================
System.out.println(statusCodeValue);
System.out.println(httpStatus);
System.out.println(httpHeaders);
System.out.println(body);
//返回其中的内容体 , 等同于返回上面的 String body
return restTemplate.getForEntity(serviceName+"/get",String.class).getBody();
}
因为此处没有使用LoadBalancerClient
做负载均衡,所以在启动类中需要添加注解@LoadBalanced
结果为:
其中statusCodeValue = 200 httpStatus = 200 OK httpHeaders = [Content-Type:"text/plain;charset=UTF-8", Date:"Fri, 23 Jul 2021 09:02:34 GMT", Keep-Alive:"timeout=60", Connection:"keep-alive", Content-Length:"22"] =[Content-Type:"text/plain;charset=UTF-8", Date:"Fri, 23 Jul 2021 09:02:34 GMT", Keep-Alive:"timeout=60", Connection:"keep-alive", Content-Length:"22"]
body = this is a get response
restTemplate.getForEntity(); 方法获取的是状态码、头部信息、结果体
Get请求,有参数,返回一个实体对象
首先在消费者和服务提供者的工程项目中都必须要含有同样的实体类
@Component
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
private String name;
private String age;
private String msg;
}
在服务提供者中存在对应的调用接口:
@RestController
public class RestTemplateController {
@RequestMapping("/getUser")
public User getUser(@RequestParam("name")String name,
@RequestParam("age")String age,
@RequestParam("msg")String msg){
return new User(name,age,msg);
}
}
在消费者中调用:
@RequestMapping("/getEntity")
public String getUser(){
String[] data = {"yangchen","18","from array"};
//调用服务,获取对象的statusCodeValue,httpStatus,httpHeaders,body
ResponseEntity<User> responseEntity =
restTemplate.getForEntity(serviceName+"/getUser?name={0}&age={1}&msg={2}",User.class,data);
//通过将数组中的参数放入请求链接中请求结果。通过{0} = data[0] 这样的方式一一对应
//getForObject仅返回该请求的内容体
User user_array = restTemplate.getForObject(serviceName+"/getUser?name={0}&age={1}&msg={2}",User.class,data);
//输出信息
System.out.println("user_array = "+user_array);
//============ 第二种方法 ================================
Map<String,Object> param = new ConcurrentHashMap<String, Object>();
param.put("name","yangchen");
param.put("age","21");
param.put("msg","from map");
//调用服务,获取对象的statusCodeValue,httpStatus,httpHeaders,body
ResponseEntity<User> responseEntity1 =
restTemplate.getForEntity(serviceName+"/getUser?name={name}&age={age}&msg={msg}",User.class,param);
//通过将数组中的参数放入请求链接中请求结果。通过{name} = map中的{name} 这个key的value 这样的方式一一对应
//getForObject仅返回该请求的内容体
User user_map = restTemplate.getForObject(serviceName+"/getUser?name={name}&age={age}&msg={msg}",User.class,param);
//输出信息
System.out.println("user_map = "+user_map);
return "success";
}
因为有两种方法,暂且返回String类型,在控制台将两种方法获得的User对象输出:
tips:restTemplate.getForObject( )方法仅返回该请求的内容体
Post请求,有参数,返回实体对象
消费者中编写接口:
//Post请求,有参数,返回实体对象
@RequestMapping("/postEntity")
public User postUser(){
//Post请求的数据必须存放在这种类型的Map中
MultiValueMap<String,Object> data = new LinkedMultiValueMap<String, Object>();
data.add("name","yangchen");
data.add("age","21");
data.add("msg","from post map");
//调用换成服务,注意这里用的是postForEntity,其中的传参位置有所变化!!!
ResponseEntity<User> responseEntity =
restTemplate.postForEntity(serviceName+"/getUser",data,User.class);
//也可以通过postForObject直接返回请求的body
User user_postForObject = restTemplate.postForObject(serviceName+"/getUser",data,User.class);
System.out.println("user = "+user_postForObject);
return restTemplate.postForEntity(serviceName+"/getUser",data,User.class).getBody();
}
结果为:
![image-20210723183459508](E:\software\JAVA\springcloud-alibaba\document\动力节点spring cloud alibaba\image-20210723183459508.png)
tips:注意参数存放只能存放在MultiValueMap接口实现类为LinkedMultiValueMap的map中,其次调用restTemplate.postForObject( )或者是restTemplate.postForEntity( )方法中的传参顺序与get不同,并且传入的url中不需要带有参数
从此以后,restTemplate.xxxForObject( ) 和 restTemplate.xxxForEntity( )两种方法只选择一种方法进行演示
Post请求,传输实体对象,返回实体对象
服务端提供相应的服务,通过@RequestBody注解来接收消费者端传进来的对象:
@RequestMapping("/responseUser")
public User responseUser(@RequestBody User user){
return user;
}
消费者端提供接口:
//Post请求,传输实体对象,返回实体对象
@RequestMapping("/postUser")
public User post(){
User user = new User("yangchen","21","from user entity");
User user_response = restTemplate.postForObject(serviceName+"/responseUser",user,User.class);
return user_response;
}
将原先的数据集合替换为实体对象进行传值。
结果为:
restTemplate调用传输JSON
消费者端:
传输JSON格式的字符串时,需要使用HttpEntity 对JSON
进行包装才可以传输。
//restTemplate调用传输JSON
@RequestMapping("/postJson")
public User postJson(){
//定义JSON字符串
String json = "{\"name\": \"yangchen\",\"age\": \"21\",\"msg\": \"from json\"}";
HttpHeaders headers = new HttpHeaders();
//设定格式为JSON格式
headers.setContentType(MediaType.APPLICATION_JSON);
//将http头和json字符串使用HttpEntity进行包装然后传输
HttpEntity<String> entity = new HttpEntity<String>(json,headers);
User user_json=restTemplate.postForObject(serviceName+"/responseUser",entity,User.class);
return user_json;
}
服务者端:
同样使用注解@RequestBody 进行接收:
@RequestMapping("/responseUser")
public User responseUser(@RequestBody User user){
return user;
}
结果如下:
tips:如果restTemplate.postForObject( )方法中除了传输的对象还存在另外需要传的值,如:token,只需在后面继续添加,逗号隔开即可。因为在底层这边维护的是一个可变数组
服务端可以通过@RequestParam注解的方式进行获取:
Put请求--一般作为修改使用
同样修改时的相关参数需要存在MultiValueMap<String,Object> dataMap = new LinkedMultiValueMap<String,Object>( );中。因为Put请求和Post请求类似,甚至可以使用Post请求去代替Put请求。
服务端:
@RequestMapping("/getUser")
public User getUser(@RequestParam("name")String name,
@RequestParam("age")String age,
@RequestParam("msg")String msg){
return new User(name,age,msg);
}
不能使用@RequestBody【User user】 方法接收。
消费者端:
//PUT请求
@RequestMapping("/put")
public String put(){
//只能使用MultiValueMap存,和Post一样
MultiValueMap<String,Object> multiValueMap = new LinkedMultiValueMap<String, Object>();
multiValueMap.add("name","yangchen");
multiValueMap.add("age","21");
multiValueMap.add("msg","from put");
//调用远程服务,put方法没有返回值
restTemplate.put(serviceName+"/getUser",multiValueMap);
return "success";
}
Delete请求
delete请求可以使用get请求代替。参数绑定方式和get请求类似:
消费端接口:
//delete请求
@RequestMapping("/delete")
public String delete(){
//数组存取参数
String[] param_array = {"yangchen","21","from delete array"};
//远程调用服务
restTemplate.delete(serviceName+"/getUser?name={0}&age={1}&msg={2}",param_array);
//====================================================
//Map存参数
Map<String,Object> paramMap = new ConcurrentHashMap<String, Object>();
paramMap.put("name","yangchen");
paramMap.put("age","21");
paramMap.put("msg","from param map");
restTemplate.delete(serviceName+"/getUser?name={name}&age={age}&msg={msg}",paramMap);
return "success";
}
与GET请求类似,但是restTemplate.delete( )方法没有返回值。
RestTemplate+Ribbon实现负载均衡
Ribbon
ribbon是客户端负载均衡
使用Ribbon需要添加依赖(客户端):
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-ribbon</artifactId>
</dependency>
我们在启动类中进行添加:
@EnableDiscoveryClient
@EnableFeignClients
@SpringBootApplication
public class NacosConsumerApplication {
public static void main(String[] args) {
SpringApplication.run(NacosConsumerApplication.class, args);
}
@LoadBalanced //如果使用了loadBalancerClient进行了代理,则已经做过一次负载均衡,此处的注解不能加
@Bean
public RestTemplate restTemplate(){
return new RestTemplate();
}
}
Spring Cloud Alibaba - RestTemplate的更多相关文章
- Spring Cloud Alibaba基础教程:支持的几种服务消费方式(RestTemplate、WebClient、Feign)
通过<Spring Cloud Alibaba基础教程:使用Nacos实现服务注册与发现>一文的学习,我们已经学会如何使用Nacos来实现服务的注册与发现,同时也介绍如何通过LoadBal ...
- Spring Cloud Alibaba Sentinel对RestTemplate的支持
Spring Cloud Alibaba Sentinel 支持对 RestTemplate 的服务调用使用 Sentinel 进行保护,在构造 RestTemplate bean的时候需要加上 @S ...
- Spring Cloud Alibaba(2)---RestTemplate微服务项目
RestTemplate微服务项目 前言 因为要运用 Spring Cloud Alibaba 开源组件到分布式项目中,所以这里先搭建一个不通过 Spring Cloud只通过 RestTemplat ...
- Spring Cloud Alibaba 新版本发布:众多期待内容整合打包加入!
在Nacos 1.0.0 Release之后,Spring Cloud Alibaba也终于发布了最新的版本.该版本距离上一次发布,过去了整整4个月!下面就随我一起看看,这个大家期待已久的版本都有哪些 ...
- Spring Cloud Alibaba基础教程:Nacos配置的加载规则详解
前情回顾: <Spring Cloud Alibaba基础教程:使用Nacos实现服务注册与发现> <Spring Cloud Alibaba基础教程:支持的几种服务消费方式(Res ...
- Spring Cloud Alibaba基础教程:使用Nacos作为配置中心
通过本教程的前两篇: <Spring Cloud Alibaba基础教程:使用Nacos实现服务注册与发现> <Spring Cloud Alibaba基础教程:支持的几种服务消费方 ...
- Spring Cloud Alibaba基础教程:使用Nacos实现服务注册与发现
自Spring Cloud Alibaba发布第一个Release以来,就备受国内开发者的高度关注.虽然Spring Cloud Alibaba还没能纳入Spring Cloud的主版本管理中,但是凭 ...
- 阿里巴巴开源 Spring Cloud Alibaba,加码微服务生态建设
本周,Spring Cloud联合创始人Spencer Gibb在Spring官网的博客页面宣布:阿里巴巴开源 Spring Cloud Alibaba,并发布了首个预览版本.随后,Spring Cl ...
- Spring Cloud Alibaba | 序言
目录 Spring Cloud Alibaba | 序言 1. Spring Cloud Alibaba是什么? 2. 主要功能 3. 组件 4. 版本说明 4.1 版本依赖关系 4.2 组件版本关系 ...
随机推荐
- 2、linux防火墙的使用(firewalld)
2.1.说明: 1.在 RHEL7 里有几种防火墙共存,firewalld.iptables,默认是使用 firewalld 来管理 netfilter 子系统,不过底层调用的命令仍然是 iptabl ...
- 14.4、web排错三部曲
1.在客户端上ping服务器端ip : ping 服务器ip地址 :#排除线路问题: 2.在客户端上telnet服务器端Ip 端口号: telnet 服务器ip地址 端口号:#排除防火墙的影响: 3. ...
- AcWing 1141. 局域网
某个局域网内有n台计算机和m条 双向 网线,计算机的编号是1~n由于搭建局域网时工作人员的疏忽, 现在局域网内的连接形成了回路,我们知道如果局域网形成回路那么数据将不停的在回路内传输,造成网络卡的现象 ...
- jenkins send files or publish
1.创建一个自由风格项目 2.添加用户凭据 3.配置git 4.配置构建方式 这里选择 send files or execute command over SSH 5.配置远程发布脚本 6.构建 7 ...
- mac 下彻底卸载node和npm
以下链接可供参考: https://segmentfault.com/a/1190000007445643 https://www.cnblogs.com/ChenGuangW/p/11398367. ...
- 《PHP基础知识总结》系列分享专栏
总结PHP基础知识,对初学者还是高手都值得参考巩固. <PHP基础知识总结>已整理成PDF文档,点击可直接下载至本地查阅https://www.webfalse.com/read/2017 ...
- Java | 字符串缓冲区(StringBuilder)
为什么要出现字符缓冲区 我们都知道,String类是不可变的,但是有的时候,我们要用到字符串的拼接,如果拼接的数量小的时候,还可以,但是如果拼接的数据量太大的话,内存的占用就太大了,所以这个时候再用S ...
- 源码解析Java Attach处理流程
前言 当Java程序运行时出现CPU负载高.内存占用大等异常情况时,通常需要使用JDK自带的工具jstack.jmap查看JVM的运行时数据,并进行分析. 什么是Java Attach 那么JVM自带 ...
- 『动善时』JMeter基础 — 55、使用非GUI模式运行JMeter(命令行模式)
目录 1.JMeter的非GUI模式说明 2.为什么使用非GUI模式运行JMeter 3.使用非GUI模式运行JMeter (1)非GUI模式运行JMeter步骤 (2)其它参数说明 4.CLI模式运 ...
- C语言:预处理命令总结
预处理指令是以#号开头的代码行,# 号必须是该行除了任何空白字符外的第一个字符.# 后是指令关键字,在关键字和 # 号之间允许存在任意个数的空白字符,整行语句构成了一条预处理指令,该指令将在编译器进行 ...