以前利用 RestTemplate 发起远程调用:
String url = "http://user-service/user/" + order.getUserId();
User user = restTemplate.getForObject(url,User.class);
存在的问题:
Feign 是一个声明式的http客户端,其作用就是帮助我们优雅的实现http请求的发送,解决上面提到的问题
在服务模块的 pom.xml 引入依赖
<dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-starter-openfeignartifactId>
dependency>
在启动类添加注解 @EnableFeignClients 开启Feign功能
@EnableFeignClients //开启Feign功能
@MapperScan("cn.itcast.order.mapper") //扫描包,配置接口实现类
@SpringBootApplication
public class OrderApplication {
public static void main(String[] args) {
SpringApplication.run(OrderApplication.class, args);
}
}
建立服务提供者接口,远程调用服务
@FeignClient("user-service") // 服务名
public interface UserClient {
@GetMapping("/user/{id}") // 路径
User findById(@PathVariable("id") Long id);
}
主要是基于SpringMVC的注解来声明远程调用信息,比如:
服务名称:user-service
请求方式:GET
请求路径:/user/{id}
请求参数:Long id
返回值类型:User
最后通过自动注入对象来调用该接口的方法
一般只需要修改日志配置,日志默认级别为 NONE,修改级别:
全局生效
feign:
client:
config:
default: # 这里用default就是全局配置
loggerLevel: HEADERS # 日志级别
局部生效
feign:
client:
config:
user-service: # 写服务名称,则是针对某个微服务的配置
loggerLevel: HEADERS # 日志级别
创建一个Feign的配置类
public class FeignClientConfiguration {
@Bean
public Logger.Level feignLogLevel() {
return Logger.Level.BASIC;
}
}
全局配置,将配置类放入注解 @EnableFeignClients 中
@EnableFeignClients(defaultConfiguration = FeignClientConfiguration.class)
局部配置,将配置类放入注解
@FeignClient(value = "user-service",configuration = FeignClientConfiguration.class)
日志级别越高,对性能影响越大
Feign 底层的客户端实现:
因此优化Feign的性能包括:
引入依赖:
<dependency>
<groupId>io.github.openfeigngroupId>
<artifactId>feign-httpclientartifactId>
dependency>
配置连接池:
feign:
client:
config:
user-service:
# Feign日志级别修改
loggerLevel: BASIC
httpclient:
enabled: true # 开启feign对HttpClient的支持
max-connections: 200 # 最大连接数
max-connections-per-route: 50 # 每个路径的最大连接数
若有多个服务想调用同一个功能模块时,每个服务都要写一次相同的调用请求,造成代码冗杂,这时可以创建一个专门的api模块集中管理调用,而各个服务只需要引用该模块就能完成调用了
创建一个模块,命名为 feign-api,然后引入Feign的相关依赖
建立好feign远程调用要有的相关配置
在服务消费者模块引入feign-api依赖
注意:调用会报错
自动进行注入时该对象没找到,因为该模块在启动时是扫描自己包下的对象,而 UserClient 是 feign-api 模块包内的,所以扫描不到生成不了对应对象
解决办法:
方式一:指定 FeignClient 所在包
@EnableFeignClients(basePackages = "cn.cyx.feign.client") //开启Feign功能
方式二:指定具体类对应的字节码
@EnableFeignClients(clients = {UserClient.class}) //开启Feign功能