什么是远程调用
如上图:目前有订单模块和用户模块,订单模块只能查询到订单信息,用户模块只能看到用户信息,但我们此时需要根据订单id查询到订单id的同时也可以看到用户信息。此时订单模块是无法直接访问用户模块的数据库来查询用户信息,想要完成这个功能就需要让订单模块远程调用用户模块,通过用户模块暴露出来的接口来完成对用户信息的查询。
RestTemplate
Resttemplate是一个Spring提供的用来发送HTTP请求的工具。使用时只需要在springboot启动类中添加上对应的bean即可
@SpringBootApplication
public class OrderApplication {
public static void main(String[] args) {
SpringApplication.run(OrderApplication.class, args);
}
@Bean
public RestTemplate restTemplate() {
return new RestTemplate();
}
}
再在需要订单模块的service层编写如下代码:
@Service
public class OrderService {
@Autowired
private OrderMapper orderMapper;
@Autowired
private RestTemplate restTemplate;
public Order selectOrderByOrderId(Long orderId) {
// 1.查询订单
Order order = orderMapper.selectByOrderId(orderId);
// 2.利用RestTemplate发起http请求,查询用户
// 直接编写url地址,不推荐,因为这样直接将服务器地址和端口号写死了,更推荐的是使用Nacos等服务注册中心对服务进行管理后使用下面的方式。
// String url = "http://localhost:8080/user/" + order.getUserId();
// 编写url路径 userservice是我们要调用的另外一个服务名称,指向对应服务器地址 url路径:{user/{id}
String url = "http://userservice/user/" + order.getUserId();
// 以get方式发送http请求,实现远程调用,返回结果应该是User类型
User user = restTemplate.getForObject(url, User.class);
// 3.封装user到Order
order.setUser(user);
// 4.返回
return order;
}
}
什么是Feign
Feign是SpringCloud提供的一个声明式客户端,进一步优化了远程调用的实现方法,增强代码可读性。
RestTemplate存在问题:
- 代码可读性差,编写代码过程中突然出现一段url地址
- 如果url参数过长,维护起来异常困难
Feign简单使用:
- 在项目中引入Feign的依赖
<!--feign客户端依赖-->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
- 在启动类上添加
@EnableFeignClients
启动Feign的功能
@EnableFeignClients
@SpringBootApplication
public class OrderApplication {
public static void main(String[] args) {
SpringApplication.run(OrderApplication.class, args);
}
}
- 编写Feign客户端,声明一个远程调用
@FeignClient(value = "userservice")
public interface UserClient {
@GetMapping("/user/{id}")
User findById(@PathVariable("id") Long id);
}
- 服务名称:userservice
- 请求方式:GET
- 请求路径:/user/{id}
- 请求参数:Long id
- 返回值类型:User
可以发现Feign客户端编写的方式和编写controller的方式很像,并且RestTemplate进行远程调用时涉及到的内容都在这里进行声明了
// RestTmeplate发送远程调用请求
// 编写url路径 userservice是我们要调用的另外一个服务名称,指向对应服务器地址 url路径:{user/{id}
String url = "http://userservice/user/" + order.getUserId();
// 以get方式发送http请求,实现远程调用,返回结果应该是User类型
User user = restTemplate.getForObject(url, User.class);
- 通过Feign在OrderService中进行远程调用
可以发现,通过Feign进行远程调用代码可读性更高,远程调用需要的信息都存放在UserClient客户端中,需要远程调用时直接调用对应的方法即可完成远程调用。public Order queryOrderById(Long orderId) { // 1.查询订单 Order order = orderMapper.findById(orderId); // 2.用Feign远程调用 User user = userClient.findById(order.getUserId()); // 3.封装user到Order order.setUser(user); // 4.返回 return order; }