本套mode使用的是Alibaba的2021.0.1版,springboot的2.6.3实现(适用于springboot 2.7~2.4),使用到的组件包括OpenFeign(实现远程调用)、loadbalancer(负载均衡)、hystrix(熔断降级)、nacos(注册中心)
nacos的安装和基本配置使用可以查看我这篇博客,或者关注下面的专栏后续会持续更新其他的Spring Cloud 组件介绍以及使用
Nacos下载与安装配置(2.2.3版本示例)
Spring Cloud Alibaba专栏
openFeign是要声明式的web服务客户端,或叫做声明式REST客户端,它让编写web服务客户端变得简单。
能够干什么?
可以让我们实现调用别的服务模块像调用本地模块接口一样方便,并且在配合上
loadbalancer
可以实现负载均衡的效果,实现集群化部署,避免请求全部到一个服务;并且通过hystrix
实现一个服务降级到效果,可以当请求过多无法按时返回时,实现一个服务降级到效果,以及通过nacos
实现一个服务注册的效果。
实现原理
OpenFeign在进行远程调用时主要是采用一种动态代理的方式实现,项目启动时其实就会把被 @FeignClient修饰的接口生成一个代理对象,全部注入到spring容器中,然后在被调用时,在内部会通过这个代理对象把接口调用转化为一个远程调用到 的Request,并发送给目标服务。
具体代理实现原理可以看下面这个博主讲解,相对比较详细
OpenFeign基本介绍和原理了解
- 首先建立一个父包,用于指定统一版本,避免出现版本
⭐记得使用maven方式建立,然后去自己加依赖指定版本,不同版本之间差距很大
记得依赖父包,便于指定版本
spring的 Spring Boot 3.0,Spring Boot 2.7~2.4和 2.4 是变化比较大的版本,我的这个架构依赖适用于Spring Boot 2.7-2.4之间的版本的,因为
SpringCloudFeign
在Hoxton.M2RELEASED版本之后抛弃了Ribbon,使用了spring-cloud-loadbalancer
,变动会比较大,也就是Spring Boot 2.4以后都使用的spring-cloud-loadbalancer
做的远程调用
- 因为ribbon以及部分组件目前已经没有更新了,所以不建议使用2.4之前的版本,而3.0的最新的也不太建议使用,因为版本过新,相对bug比较多,选择中间的就好
具体需要查看自己spring对应的组件版本可以查看
springcloud组件版本对应说明
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0modelVersion>
<groupId>org.examplegroupId>
<artifactId>springcloud-testartifactId>
<packaging>pompackaging>
<version>1.0-SNAPSHOTversion>
<modules>
<module>service-ordermodule>
<module>service-productmodule>
<module>service-utilmodule>
modules>
<parent>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-parentartifactId>
<version>2.6.3version>
<relativePath/>
parent>
<properties>
<java.version>1.8java.version>
<project.build.sourceEncoding>UTF-8project.build.sourceEncoding>
<cloud.version>2021.0.1cloud.version>
<alibaba.version>2021.0.1.0alibaba.version>
<mysql.version>8.0.30mysql.version>
properties>
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-dependenciesartifactId>
<version>${cloud.version}version>
<type>pomtype>
<scope>importscope>
dependency>
<dependency>
<groupId>com.alibaba.cloudgroupId>
<artifactId>spring-cloud-alibaba-dependenciesartifactId>
<version>${alibaba.version}version>
<type>pomtype>
<scope>importscope>
dependency>
dependencies>
dependencyManagement>
project>
因为为了测试方便,我没有使用到数据库,所以就没加sql相关的依赖和配置,如果自己需要使用到,自行添加就行
loadbalancer
依赖的,但之后的版本弃用了,所以需要添加
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<parent>
<artifactId>springcloud-testartifactId>
<groupId>org.examplegroupId>
<version>1.0-SNAPSHOTversion>
parent>
<modelVersion>4.0.0modelVersion>
<artifactId>service-orderartifactId>
<properties>
<maven.compiler.source>8maven.compiler.source>
<maven.compiler.target>8maven.compiler.target>
properties>
<dependencies>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-webartifactId>
<scope>providedscope>
dependency>
<dependency>
<groupId>org.projectlombokgroupId>
<artifactId>lombokartifactId>
dependency>
<dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-starter-openfeignartifactId>
<scope>providedscope>
dependency>
<dependency>
<groupId>com.alibaba.cloudgroupId>
<artifactId>spring-cloud-starter-alibaba-nacos-discoveryartifactId>
dependency>
<dependency>
<groupId>org.examplegroupId>
<artifactId>service-utilartifactId>
<version>1.0-SNAPSHOTversion>
dependency>
<dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-starter-netflix-hystrixartifactId>
<version>2.2.9.RELEASEversion>
dependency>
<dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-starter-loadbalancerartifactId>
dependency>
dependencies>
project>
server:
port: 8201
spring:
application:
name: service-order
profiles:
active: dev
# nacos配置
cloud:
nacos:
discovery:
server-addr: localhost:8848
# 开启负载均衡(默认是不开启的)
feign:
circuitbreaker:
enabled: true
# 熔断配置
hystrix:
command:
default:
execution.isolation.thread.timeoutInMilliseconds: 3000 # 设置Hystrix超时时间(毫秒)
ProductFeignClient#getProducts:
execution.isolation.thread.timeoutInMilliseconds: 1000 # 为特定的Feign方法设置超时时间
-mavne的依赖和xml和前面的order模块一样,就是端口和name不同
server:
port: 8203
spring:
application:
name: service-product
# 注入到nacos
cloud:
nacos:
discovery:
server-addr: localhost:8848
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<parent>
<artifactId>springcloud-testartifactId>
<groupId>org.examplegroupId>
<version>1.0-SNAPSHOTversion>
parent>
<modelVersion>4.0.0modelVersion>
<artifactId>service-utilartifactId>
<properties>
<maven.compiler.source>8maven.compiler.source>
<maven.compiler.target>8maven.compiler.target>
properties>
<dependencies>
<dependency>
<groupId>org.projectlombokgroupId>
<artifactId>lombokartifactId>
<version>1.18.16version>
dependency>
dependencies>
project>
@Data
public class OrderInfo {
private Long id;
/**订单号*/
private String orderNo;
/**产品列表*/
private List<ProductInfo> productInfoList;
/**收货人姓名*/
private String receiverName;
}
@Data
@AllArgsConstructor
@NoArgsConstructor
public class ProductInfo {
private Long id;
/**销量*/
private Integer sale;
/**价格*/
private Integer price;
/**名称*/
private String name;
/**类型*/
private Integer type;
/**端口*/
private Integer port;
}
@Data
public class Result<T> {
//状态码
private Integer code;
//信息
private String message;
//数据
private T data;
private Result(){}
//设置数据,返回对象的方法
public static <T> Result<T> build(T data, ResultCodeEnum resultCodeEnum) {
//创建Result对象,设置值,返回对象
Result<T> result = new Result<>();
//判断返回结果中是否需要数据
if (data != null) {
//设置数据到result对象
result.setData(data);
}
//设置其他值
result.setCode(resultCodeEnum.getCode());
result.setMessage(resultCodeEnum.getMessage());
//返回设置值之后的对象
return result;
}
//成功的方法
public static <T> Result<T> ok(T data) {
return build(data, ResultCodeEnum.SUCCESS);
}
//成功的方法
public static <T> Result<T> ok() {
return build(null, ResultCodeEnum.SUCCESS);
}
//失败的方法
public static <T> Result<T> fail(T data) {
return build(data, ResultCodeEnum.FAIL);
}
}
@Getter
public enum ResultCodeEnum {
SUCCESS(200,"成功"),
FAIL(201, "失败"),
;
private Integer code;
private String message;
ResultCodeEnum(Integer code, String message) {
this.code = code;
this.message = message;
}
}
@SpringBootApplication
@EnableDiscoveryClient //启动nacos注解
@EnableFeignClients //开启feign的客户端
public class ServiceOrderApplication {
public static void main(String[] args) {
SpringApplication.run(ServiceOrderApplication.class,args);
}
}
用于配置负载均衡,负载均衡的策略这里写了两个,也可以使用权重或者自定义策略;还有远程调用时的重试策略相关配置,
Default()
方法可以传参数自行配置
@Configuration
public class LoadBalancedConfig {
/**
* 配置负载均衡策略
* @param environment
* @param loadBalancerClientFactory
* @return ReactorLoadBalancer
*/
@Bean
ReactorLoadBalancer<ServiceInstance> randomLoadBalancer(Environment environment,
LoadBalancerClientFactory loadBalancerClientFactory) {
String name = environment.getProperty(LoadBalancerClientFactory.PROPERTY_NAME);
// 轮询负载均衡
RoundRobinLoadBalancer roundRobinLoadBalancer = new RoundRobinLoadBalancer(loadBalancerClientFactory
.getLazyProvider(name, ServiceInstanceListSupplier.class),
name);
// 随机
// RandomLoadBalancer randomLoadBalancer = new RandomLoadBalancer(loadBalancerClientFactory
// .getLazyProvider(name, ServiceInstanceListSupplier.class),
// name);
return roundRobinLoadBalancer;
}
/**
* 请求失败时,默认重试策略(默认重试5次)
* @Param * @param null
* @return
*/
@Bean
public Retryer feignRetryer(){
//每次重试之间等待 500 毫秒。
//最大等待时间为 1 毫秒。
// 最多尝试 3 次。
return new Retryer.Default(500,1,3);
}
}
添加了两个接口,一个是用于测试,当前模块是否可以正常被调用接口的,一个是测试远程调用是否会有问题,并且也没有写service层,直接进行调用feign进行远程调用
@RestController
@RequestMapping("/admin/acl/index")
public class index {
@Resource
private ProductFeignClient productFeignClient;
/**
* 测试订单服务是否有问题
* @author yingfeng
* @return * @return Result
*/
@PostMapping("/login")
public Result login(){
HashMap<String, String> map = new HashMap<>(16);
map.put("token","token-admin");
System.out.println();
return Result.ok(map);
}
/**
* 测试远程调用接口的接口
* @version 1.0
*/
@GetMapping("/getorderinfo/{type}")
public Result getrderinfo(@PathVariable Integer type){
OrderInfo orderInfo = new OrderInfo();
//因为返回的是list,所以如果远程调用超时,或者报错时,进入熔断所以会返回一个空集合
List<ProductInfo> productInfoList = productFeignClient.getProductInfoList(type);
if(CollectionUtils.isEmpty(productInfoList)){
return Result.ok("当前人数过多,稍后重试!");
}
orderInfo.setOrderNo("dd131312");
orderInfo.setId(1234l);
orderInfo.setProductInfoList(productInfoList);
orderInfo.setReceiverName("张三");
return Result.ok(orderInfo);
}
}
@FeignClient
:启用远程调用(启动类需要有@EnableFeignClients
,才会生效)
- name/value:指定注册到注册中心的服务的名称,如果项目使用了负载均衡,name属性会作为微服务的名称,用于服务发现
- url: url一般用于调试,可以手动指定
@FeignClient
调用的地址- fallback: 定义容错的处理类,当调用远程接口失败或超时时,会调用对应接口的容错逻辑,fallback指定的类必须实现
@FeignClient
标记的接口(也就是发生熔断时,会执行该类)
@GetMapping地址需要记得写全,,方法名称要和被调用的接口一致
@FeignClient(value = "service-product",fallback = ProductFeignClientImpl.class)
@LoadBalancerClient(name = "service-product",configuration = LoadBalancedConfig.class)
public interface ProductFeignClient {
@GetMapping("/admin/Product/innet/getProductInfoList/{type}")
public List<ProductInfo> getProductInfoList(@PathVariable Integer type);
}
@Slf4j
@Service
public class ProductFeignClientImpl implements ProductFeignClient {
@Override
public List<ProductInfo> getProductInfoList(Integer type) {
log.info("请求接口失败,进入熔断!");
List<ProductInfo> productInfoList = new ArrayList<>();
return productInfoList;
}
}
和上面的一样,占位置就不出现放上来了
用于给order进行远程调用接口,因为是内部调用便于取出,就直接返回对应的参数,不需要返回统一result处理类
@RestController
@RequestMapping("/admin/Product")
public class ProductInfoController {
@Resource
private ProductInfoService productInfoService;
@GetMapping("/innet/getProductInfoList/{type}")
public List<ProductInfo> getProductInfoList(@PathVariable Integer type){
List<ProductInfo> productInfoList = productInfoService.getProductInfoList(type);
System.out.println("查询产品详情:"+productInfoList);
//如果传入的类型未查询到,自定义返回一个运行时异常
if (CollectionUtils.isEmpty(productInfoList)){
throw new RuntimeException();
}
return productInfoList;
}
}
public interface ProductInfoService {
List<ProductInfo> getProductInfoList(Integer type);
}
@Service
public class ProductInfoServiceImpl implements ProductInfoService {
// 获取当前服务的端口
@Value("${server.port}")
private Integer port;
@Override
public List<ProductInfo> getProductInfoList(Integer type) {
ProductInfo productInfo3 = new ProductInfo(3l,800,20,"哈密瓜1",1,port);
ProductInfo productInfo1 = new ProductInfo(1l,50,8,"苹果1",1,port);
ProductInfo productInfo2 = new ProductInfo(2l,200,13,"牛肉1",2,port);
ProductInfo productInfo4 = new ProductInfo(4l,50,9,"青菜1",2,port);
/* 实际项目中,会从数据库查出数据 */
List<ProductInfo> productInfoList = Arrays.asList(productInfo1,productInfo2,productInfo3,productInfo4);
// 根据传入的类型返回指定类型
List<ProductInfo> result = productInfoList.stream()
.filter(productInfo -> productInfo.getType() == type)
.collect(Collectors.toList());
return result;
}
}
service-order启动一个、service-product启动两个
设置访问可以并向启动