Feign(服务调用):
前提条件:把互相调用的服务在Nacos进行注册;
1)引入相关依赖
<!--服务调用-->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
2)在调用端的启动类添加注解:@EnableFeignClients
3)创建包和接口:
创建client包
@FeignClient注解用于指定从哪个服务中调用功能 ,名称与被调用的服务名保持一致。
@GetMapping注解用于对被调用的微服务进行地址映射。
@PathVariable注解一定要指定参数名称,否则出错
@Component注解防止,在其他位置注入CodClient时idea报错
package com.guli.edu.client;
@FeignClient("service-vod")
@Component
public interface VodClient {
@DeleteMapping(value = "/eduvod/vod/video/{videoId}")
public R removeVideo(@PathVariable("videoId") String videoId);
}
4)调用微服务:在调用端的VideoServiceImpl中调用client中的方法
@Override
public boolean removeVideoById(String id) {
//查询云端视频id
Video video = baseMapper.selectById(id);
String videoSourceId = video.getVideoSourceId();
//删除视频资源
if(!StringUtils.isEmpty(videoSourceId)){
vodClient.removeVideo(videoSourceId);
}
Integer result = baseMapper.deleteById(id);
return null != result && result > 0;
}
feign结合Hystrix使用:
1)在调用端添加依赖:
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-ribbon</artifactId>
</dependency>
<!--hystrix依赖,主要是用 @HystrixCommand -->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-hystrix</artifactId>
</dependency>
<!--服务注册-->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
</dependency>
<!--服务调用-->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
2)在调用端配置文件中添加hystrix配置
#开启熔断机制
feign.hystrix.enabled=true
#设置hystrix超时时间,默认1000ms
hystrix.command.default.execution.isolation.thread.timeoutInMilliseconds=60
3)在调用端的client包里面创建熔断器的实现类:要么是调用服务的降级操作,要么简单的给个回应。
@Component
public class VodFileDegradeFeignClient implements VodClient {
@Override
public R removeVideo(String videoId) {
return R.error().message("time out");
}
@Override
public R removeVideoList(List videoIdList) {
return R.error().message("time out");
}
}
4)修改Client接口的注解
@FeignClient(name = "service-vod", **fallback =VodFileDegradeFeignClient.class**)
@Component
public interface VodClient {
@DeleteMapping(value = "/eduvod/vod/{videoId}")
public R removeVideo(@PathVariable("videoId") String videoId);
@DeleteMapping(value = "/eduvod/vod/delete-batch")
public R removeVideoList(@RequestParam("videoIdList") List videoIdList);
}
Hystrix:
Hystrix的设计原则包括:
资源隔离:例如三个模块拥有100线程,把线程按照一定数目进行分配,防止雪崩。
熔断器:熔断器开关由关闭到打开的状态转换是通过当前服务健康状况和设定阈值比较决定的.
1)当熔断器开关关闭时, 请求被允许通过熔断器. 如果当前健康状况高于设定阈值, 开关继续保持关闭. 如果当前健康状况低于设定阈值, 开关则切换为打开状态.
2)当熔断器开关打开时, 请求被禁止通过.
3)当熔断器开关处于打开状态, 经过一段时间后, 熔断器会自动进入半开状态, 这时熔断器只允许一个请求通过. 当该请求调用成功时, 熔断器恢复到关闭状态. 若该请求失败, 熔断器继续保持打开状态, 接下来的请求被禁止通过.
命令模式
Hystrix大概流程:
构建Hystrix的Command对象, 调用执行方法.
Hystrix检查当前服务的熔断器开关是否开启, 若开启, 则执行降级服务getFallback方法.
若熔断器开关关闭, 则Hystrix检查当前服务的线程池是否能接收新的请求, 若超过线程池已满, 则执行降级服务getFallback方法.
若线程池接受请求, 则Hystrix开始执行服务调用具体逻辑run方法.
若服务执行失败, 则执行降级服务getFallback方法, 并将执行结果上报Metrics更新服务健康状况.
若服务执行超时, 则执行降级服务getFallback方法, 并将执行结果上报Metrics更新服务健康状况.
若服务执行成功, 返回正常结果.
若服务降级方法getFallback执行成功, 则返回降级结果.
若服务降级方法getFallback执行失败, 则抛出异常.
fallback方法自己书写:可以是除去边边角角只保留核心服务的降级(最好是这个),也可以是简简单单的回应。