• springcloud(六) Hystrix 熔断,限流


    Hystrix 熔断:

      首先仍然启动Eureka,这里就不说了。

    OrderController.java:

    package com.tuling.cloud.study.user.controller;
    
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.PathVariable;
    import org.springframework.web.bind.annotation.RestController;
    import org.springframework.web.client.RestTemplate;
    
    import com.netflix.hystrix.contrib.javanica.annotation.HystrixCommand;
    import com.tuling.cloud.study.user.entity.User;
    
    @RestController
    public class OrderController {
      private static final Logger logger = LoggerFactory.getLogger(OrderController.class);
      @Autowired
      private RestTemplate restTemplate;
      
    
      @HystrixCommand(fallbackMethod = "findByIdFallback")
      @GetMapping("/user/{id}")
      public User findById(@PathVariable Long id) {
    	logger.error("================请求用户中心接口,用户id:" + id + "==============");
        return restTemplate.getForObject("http://microservice-provider-user/" + id, User.class);
      }
    
      //降级方法
      public User findByIdFallback(Long id) { 
        User user = new User();
        user.setId(-1L);
        user.setName("默认用户");
        return user;
      }
    
    }
    

      

     order 服务和上一章一样唯一修改的是yml文件:

    server:
      port: 9010
    spring:
      application:
        name: microservice-consumer-order
    eureka:
      client:
        serviceUrl:
          defaultZone: http://localhost:8761/eureka/
      instance:
        prefer-ip-address: true
    hystrix:
      command:
        default:
          circuitBreaker:
            requestVolumeThreshold: 3 #默认20,熔断的阈值,如何user服务报错满足3次,熔断器就会打开,就算order之后请求正确的数据也不行。
            sleepWindowInMilliseconds: 5000 #默认5S , 等5S之后熔断器会处于半开状态,然后下一次请求的正确和错误讲决定熔断器是否真的关闭和是否继续打开
    

      user服务修改UserController.java其余不变

    package com.tuling.cloud.study.controller;
    
    import java.util.Random;
    
    import org.apache.log4j.Logger;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.cloud.client.serviceregistry.Registration;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.PathVariable;
    import org.springframework.web.bind.annotation.RestController;
    
    import com.tuling.cloud.study.entity.User;
    import com.tuling.cloud.study.repository.UserRepository;
    
    @RestController
    public class UserController {
    	
      private final Logger logger = Logger.getLogger(getClass());
    	
      @Autowired
      private UserRepository userRepository;
      @Autowired
      private Registration registration;
      
    
      @GetMapping("/{id}")
      public User findById(@PathVariable Long id) throws Exception {
    	  logger.info("用户中心接口:查询用户"+ id +"信息");
    	  //测试熔断,传入不存在的用户id模拟异常情况
    	  if (id == 10) {
    	    throw new NullPointerException();
    	  }
    	  User findOne = userRepository.findOne(id);
    	  return findOne;
      }
      
      @GetMapping("/getIpAndPort")
      public String findById() {
    	  return registration.getHost() + ":" + registration.getPort();
      }
    }
    

      user服务模拟接口报错,order服务在调用的时候如果id传入的是10 ,就会导致user服务报错,那么满足3次报错之后,熔断器就会打开。注意:之后在5S内浏览器继续请求order服务的findById()接口是不会进入的,hystrix会直接执行降级方法。

    等5S过去之后,hytrix不会全打开,而是处于半开状态,接下来的第一个请求决定熔断器是否继续打开,还是关闭。

    演示:

    特别注意“:user服务报错满足3次,就导致调用方order的 findById() 进不去了,而是直接进入降级方法。这就是熔断。

    Hystrix 限流:

      Eureka 还是用同样的(略)

    order工程截图:

    pom.xml 和上一章一样(略)

    OrderController.java:

    package com.jiagoushi.cloud.study.user.controller;
    
    import com.jiagoushi.cloud.study.user.entity.User;
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.PathVariable;
    import org.springframework.web.bind.annotation.RestController;
    import org.springframework.web.client.RestTemplate;
    
    import com.netflix.hystrix.contrib.javanica.annotation.HystrixCommand;
    import com.netflix.hystrix.contrib.javanica.annotation.HystrixProperty;
    
    @RestController
    public class OrderController {
    
    	private static final Logger logger = LoggerFactory.getLogger(OrderController.class);
    
    	@Autowired
    	private RestTemplate restTemplate;
    
    	@HystrixCommand(fallbackMethod = "findByIdFallback", 
    					groupKey = "orderUserGroup", 
    					threadPoolKey = "orderUserIdThreadPool", 
    					threadPoolProperties = {
    	        			@HystrixProperty(name = "coreSize", value = "2"), 
    	        			@HystrixProperty(name = "maxQueueSize", value = "2"),
    	        			@HystrixProperty(name = "queueSizeRejectionThreshold", value = "1") }) //maxQueueSize和queueSizeRejectionThreshold 简单理解两者取最小值做为队列长度
    	@GetMapping("/user/{id}")
    	public User findById(@PathVariable Long id) {
    		logger.info("================请求用户中心接口,用户id:" + id + "==============");
    		return restTemplate.getForObject("http://microservice-provider-user/" + id, User.class);
    	}
    
    	@HystrixCommand(fallbackMethod = "findByIdFallback",
    			groupKey = "orderUserGroup",
    			threadPoolKey = "orderUserIdThreadPool",
    			threadPoolProperties = {
    					@HystrixProperty(name = "coreSize", value = "2"), //配置线程池线程数量
    					@HystrixProperty(name = "maxQueueSize", value = "2"),
    					@HystrixProperty(name = "queueSizeRejectionThreshold", value = "1") })
    	@GetMapping("/user/{userName}")
    	public User findByUserName(@PathVariable String userName) {
    		logger.info("================请求用户中心接口,用户userName:" + userName + "==============");
    		return restTemplate.getForObject("http://microservice-provider-user/" + userName, User.class);
    	}
    
    	//降级方法
    	public User findByIdFallback(Long id) {
    		User user = new User();
    		user.setId(-1L);
    		user.setName("默认用户");
    		return user;
    	}
    	
    }
    

      说明:

    1.  hystix 默认线程池大小是10。
    2. groupKey 是 服务分组 , threadPoolKey 是 线程池标识 , 也就是说当groupKey和threadPoolKey 同时修饰findById() 和findByUserName() 时 ,他们共用一个线程池,大小共10。
    3. ThreadPoolProperties:配置线程池参数,coreSize配置核心线程池大小和线程池最大大 小,keepAliveTimeMinutes是线程池中空闲线程生存时间(如果不进行动态配置,那么是没 有任何作用的),maxQueueSize配置线程池队列最大大小, queueSizeRejectionThreshold限定当前队列大小,即实际队列大小由这个参数决定,通过 改变queueSizeRejectionThreshold可以实现动态队列大小调整。 

    applciation.xml:

    server:
      port: 9010
    spring:
      application:
        name: microservice-consumer-order
    eureka:
      client:
        serviceUrl:
          defaultZone: http://localhost:8761/eureka/
      instance:
        prefer-ip-address: true
    hystrix:
      command:
        default:
          execution:
            isolation:
              thread:
                timeoutInMilliseconds: 20000 #命令执行超时时间,默认1000ms,就是调接口的响应时间超过20S就执行降级,不管提供者是否挂机还是延迟超过时间就走降级
    

          

    user工程截图:

    pom.xml 和上一章一样(略)

    UserController.java:

    package com.jiagoushi.cloud.study.controller;
    
    import com.jiagoushi.cloud.study.entity.User;
    import org.apache.log4j.Logger;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.cloud.client.serviceregistry.Registration;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.PathVariable;
    import org.springframework.web.bind.annotation.RestController;
    
    import com.jiagoushi.cloud.study.repository.UserRepository;
    
    @RestController
    public class UserController {
    	
      private final Logger logger = Logger.getLogger(getClass());
    	
      @Autowired
      private UserRepository userRepository;
      @Autowired
      private Registration registration;
      
    
      @GetMapping("/{id}")
      public User findById(@PathVariable Long id) throws Exception {
    	  logger.info("用户中心接口:查询用户"+ id +"信息");
    // 配合限流演示,模拟业务耗时3S    Thread.sleep(3000);
    User findOne = userRepository.findOne(id); return findOne; } @GetMapping("/getIpAndPort") public String findById() { return registration.getHost() + ":" + registration.getPort(); } }

      application.yml:

    server:
      port: 8002
    spring:
      application:
        name: microservice-provider-user
      jpa:
        generate-ddl: false
        show-sql: true
        hibernate:
          ddl-auto: none
      datasource:                           # 指定数据源
        platform: h2                        # 指定数据源类型
        schema: classpath:schema.sql        # 指定h2数据库的建表脚本
        data: classpath:data.sql            # 指定h2数据库的数据脚本
    logging:                                # 配置日志级别,让hibernate打印出执行的SQL
      level:
        root: INFO
        org.hibernate: INFO
        org.hibernate.type.descriptor.sql.BasicBinder: TRACE
        org.hibernate.type.descriptor.sql.BasicExtractor: TRACE
    eureka:
      client:
        serviceUrl:
          defaultZone: http://localhost:8761/eureka/
      instance:
        prefer-ip-address: true
    

      使用jemeter 来演示 hystrix 的限流:

      先默认order接口什么都不配置:

    上图:

     

     结果发现: 12个线程有10成功,2个被降级处理,说明hystrix 默认的线程池大小是10,

       接下来配置一下order接口:

    上图:

     结果发现:12个线程访问只有3个成功,9个被降级。因为hystrix 线程池被配置成2个,队列长度1,所以9个线程立即被降级。

     就好比商品详情服务一共100个线程,只允许20线程可以调用评论接口,如果并发是50,那么其他30就被降级,线程就立即回收,防止服务雪崩

     欢迎来QQ群:592495675 搞事情

  • 相关阅读:
    JSP动作元素<jsp:include>和<jsp:param>的搭配使用
    js去除字符串中的空格
    js 判断字符串中是否包含某个字符串(转载)
    window+R
    eclipse中ctrl+K失效
    图片转二进制——各种方法汇总(转载)
    spring mvc使用ModelAndView时发生No request handling method with name '方法 名' in class [类名]的错误
    类中main函数的快捷创建
    java中的中文字符转码技术
    SPOJ QTREE6
  • 原文地址:https://www.cnblogs.com/smallFishs/p/10601300.html
Copyright © 2020-2023  润新知