5、Hystrix 请求缓存(request cache)
requestcontext:
1、 在一个请求执行之前,都必须先初始化一个requestcontext;
HystrixRequestContext context = HystrixRequestContext.initializeContext();
然后在请求结束之后,需要关闭request context
context.shutdown();
一般来说,在java web来的应用中,都是通过filter过滤器来实现的
filter:
/**
* hystrix请求上下文过滤器
* @author 张三丰
*
*/
public class HystrixRequestContextFilter implements Filter {
public void init(FilterConfig config) throws ServletException {
}
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
HystrixRequestContext context = HystrixRequestContext.initializeContext();
try {
chain.doFilter(request, response);
} catch (Exception e) {
e.printStackTrace();
} finally {
context.shutdown();
}
}
public void destroy() {
}
}
command:
/**
* 获取商品名称的command
* @author 张三丰
*
*/
public class GetProductNameCommand extends HystrixCommand<String> {
private Long productId;
public GetProductNameCommand(Long productId) {
super(Setter.withGroupKey(HystrixCommandGroupKey.Factory.asKey("BrandInfoService"))
.andCommandKey(HystrixCommandKey.Factory.asKey("GetproductNameCommand"))
.andThreadPoolKey(HystrixThreadPoolKey.Factory.asKey("GetproductInfoPool"))
.andThreadPoolPropertiesDefaults(HystrixThreadPoolProperties.Setter()//配置线程池
.withCoreSize(15)
.withQueueSizeRejectionThreshold(10))
.andCommandPropertiesDefaults(HystrixCommandProperties.Setter()//配置信号量(这个参数设置了HystrixCommand.getFallback()最大允许的tomcat并发请求数量,默认值是10,也是通过semaphore信号量的机制去限流如果超出了这个最大值,那么直接被reject)
.withFallbackIsolationSemaphoreMaxConcurrentRequests(15))
);
this.productId = productId;
}
@Override
protected String run() throws Exception {
// 调用一个商品服务的接口(对于同一个请求,会被缓存,比如productId=100的商品请求两次,那么第二次会直接查缓存,不会进到这里)
// 如果调用失败了,报错了,那么就会去调用fallback降级机制
throw new Exception();
}
//抛出异常,走降级接口
@Override
protected String getFallback() {
//从缓存拿数据,尽量别走网络,如果非得走网络,需要再次调用command
System.out.println("从本地缓存获取商品数据,productId=" + productId);
return "商品名称";
}
//开启请求缓存
@Override
protected String getCacheKey() {
return "product_info_" + productId;
}
}
controller中:
@RequestMapping("/getProductInfos")
@ResponseBody
public String getProductInfos(Long productId) {
GetProductInfoCommand getProductInfoCommand = new GetProductInfoCommand(productId);
ProductInfo productInfo = getProductInfoCommand.execute();
System.out.println(productInfo);
System.out.println(getProductInfoCommand.isResponseFromCache());
return "success";
}