文章目录
- 前言
- 正文
- 一、项目结构
- 二、服务调用链路说明
- 三、Rpc调用链路说明
- 四、项目代码
- 4.1 client 模块中的feign接口
- 4.2 client 中的rest接口
- 4.3 client 中的启动类
- 4.4 server中的rest接口
- 4.5 server中的配置文件
- 五、调试
- 附录
- 附1:本系列文章链接
前言
本篇是SpringCloud原理系列的 OpenFeign 模块的第一篇。主要内容是搭建一个极简的Spring Cloud OpenFeign 调用链路。
项目代码仓库地址:https://gitee.com/fengsoshuai/springcloud-openfeign-demo
正文
本次项目使用java 17,spring cloud 4.0.4,springboot 3.1.4。
maven 环境编译,idea开发。
一、项目结构
本次项目分为3个模块。
二、服务调用链路说明
- 使用IDEA/Postman/Apifox等工具进行触发client服务的接口
- client内部通过feign调用server接口
- server执行业务逻辑
- server返回执行结果到client
- client返回调用结果到触发方
三、Rpc调用链路说明
两个服务之间,使用远程调用。
基本都是需要URL,请求头,请求报文,请求方式(Get\Post 等)等基本信息的。
下图简单说明rpc调用时的链路。
其中,调用方,相当于发起远程调用的一方,对比本项目的话,相当于使用postman等工具触发后,client模块的操作。
只是特殊的一点在于,调用方中的红色虚线框内的部分,被openFeign 封装了,不再是我们手动去处理他们。而这也正是本系列研究的重点。
中间部分,就是形如 RestTemplate,WebClient的功能,只是发出请求,接受响应。
服务方,就是一个提供rest接口的普通应用。
四、项目代码
本文全部代码托管在gitte仓库中,地址已经在文章开头给出。
这里只粘贴出比较重要的几个文件。
4.1 client 模块中的feign接口
package org.feng.feigns;
import org.feng.common.dto.HelloRequest;
import org.feng.common.dto.HelloResponse;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
/**
* hello-feign客户端接口
*
* @version v1.0
* @author: fengjinsong
* @date: 2023年11月20日 21时25分
*/
@FeignClient(name = "helloFeignClient", url = "http://localhost:10080")
public interface HelloFeignClient {
@PostMapping("/hello/post")
HelloResponse postHello(@RequestBody HelloRequest helloRequest);
}
4.2 client 中的rest接口
package org.feng.controller;
import jakarta.annotation.Resource;
import jakarta.servlet.http.HttpServletRequest;
import lombok.extern.slf4j.Slf4j;
import org.feng.common.dto.HelloRequest;
import org.feng.common.dto.HelloResponse;
import org.feng.feigns.HelloFeignClient;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import java.time.LocalDateTime;
import java.util.Objects;
/**
* HelloFeignClientController
*
* @version v1.0
* @author: fengjinsong
* @date: 2023年11月20日 21时45分
*/
@Slf4j
@RestController
@RequestMapping("/helloclient")
public class HelloFeignClientController {
@Resource
private HelloFeignClient helloFeignClient;
@PostMapping("/postHello")
public HelloResponse postHello(@RequestBody HelloRequest helloRequest) {
if(Objects.isNull(helloRequest.getLocalDateTime())){
helloRequest.setLocalDateTime(LocalDateTime.now());
}
ServletRequestAttributes requestAttributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
HttpServletRequest request = requestAttributes.getRequest();
String localAddr = request.getLocalAddr();
int serverPort = request.getServerPort();
helloRequest.setHost(localAddr);
helloRequest.setPort(serverPort);
log.info("helloRequest {}", helloRequest);
return helloFeignClient.postHello(helloRequest);
}
}
4.3 client 中的启动类
指定扫描包为 org.feng.feigns
。
package org.feng;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.openfeign.EnableFeignClients;
@EnableFeignClients(basePackages = "org.feng.feigns")
@SpringBootApplication
public class ClientApplication {
public static void main(String[] args) {
SpringApplication.run(ClientApplication.class, args);
}
}
4.4 server中的rest接口
package org.feng.controller;
import lombok.extern.slf4j.Slf4j;
import org.feng.common.dto.HelloRequest;
import org.feng.common.dto.HelloResponse;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
/**
* openfeign 控制器
*
* @author feng
*/
@Slf4j
@RequestMapping("/hello")
@RestController
public class HelloOpenFeignController {
@PostMapping("/post")
public HelloResponse postHello(@RequestBody HelloRequest helloRequest) {
log.info("request:{}", helloRequest);
HelloResponse response = new HelloResponse();
response.setTitle(helloRequest.getTitle());
response.setLocalDateTime(helloRequest.getLocalDateTime());
response.setFromHost(helloRequest.getHost());
response.setFromPort(helloRequest.getPort());
log.info("response: {}", response);
return response;
}
}
4.5 server中的配置文件
spring.application.name=openserver
server.port=10080
五、调试
启动server 和 client 服务。
在idea中触发client 的服务:
POST http://localhost:8080/helloclient/postHello
Content-Type: application/json
{
"title": "托尔斯泰"
}
响应报文如下:
{
"fromHost": "127.0.0.1",
"fromPort": 8080,
"title": "托尔斯泰",
"localDateTime": "2023-11-21T14:07:18.537384"
}
server中的服务,打印出来的日志如下:
以上就是通过onepfeign 进行rpc 调用的完整示例了。
可以看到,我们只在client中定义了接口,并没有实现。但是在调用时,没有报错,同时也调用到了server服务。而这,就是spring cloud 中的 openfeign 封装了远程调用,帮我们处理的部分,也是我们后续研究其原理的核心部分。
附录
附1:本系列文章链接
SpringCloud原理-OpenFeign篇(一、Hello OpenFeign项目示例)
SpringCloud原理-OpenFeign篇(二、OpenFeign包扫描和FeignClient的注册原理)
SpringCloud原理-OpenFeign篇(三、FeignClient的动态代理原理)