• SpringCloud之熔断器Hystrix及服务监控Dashboard


    目的:  

         服务雪崩效应

      服务熔断服务降级

      Hystrix默认超时时间设置

      Hystrix服务监控Dashboard


    服务雪崩效应

      雪崩效应就是一种不稳定的平衡状态也是加密算法的一种特征,它指明文或密钥的少量变化会引起密文的很大变化,就像雪崩前,山上看上去很平静,但是只要有一点问题,就会造成一片大崩溃

    正常情况下的访问 

    但是,当请求的服务中出现无法访问、异常、超时等问题时(图中的I),那么用户的请求将会被阻塞。

    如果多个用户的请求中,都存在无法访问的服务,那么他们都将陷入阻塞的状态中。

    Hystrix的引入,可以通过服务熔断和服务降级来解决这个问题


    服务熔断服务降级

    • Hystrix断路器简介

    Hystrix 是 Netflix 针对微服务分布式系统采用的熔断保护中间件,相当于电路中的保险丝。

    在分布式环境中,许多服务依赖项中的一些必然会失败。Hystrix 是一个库,通过添加延迟容忍和容错逻辑,帮助你控制这些分布式服务之间的交互。Hystrix 通过隔离服务之间的访问点、停止级联失败和提供回退选项来实现这一点,所有这些都可以提高系统的整体弹性。

    在微服务架构下,很多服务都相互依赖,如果不能对依赖的服务进行隔离,那么服务本身也有可能发生故障,Hystrix 通过 HystrixCommand 对调用进行隔离,这样可以阻止故障的连锁效应,能够让接口调用快速失败并迅速恢复正常,或者回退并优雅降级。

    Hystrix服务熔断服务降级@HystrixCommand fallbackMethod

    熔断机制是应对雪崩效应的一种微服务链路保护机制。

    当某个服务不可用或者响应时间超时,会进行服务降级,进而熔断该节点的服务调用,快速返回自定义的错误影响页面信息。

    我们来写个工程测试一下

    我们写一个新的带服务熔断的服务提供者项目 microservice-student-provider-hystrix-1004

    配置和 代码 都复制一份到这个项目里

     在新工程中的pom依赖中添加hystrix依赖:

            <!--Hystrix相关依赖-->
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-starter-hystrix</artifactId>
            </dependency>

    application.yml修改下端口和实例名称

    server:
      port: 1004
      context-path: /
    spring:
      datasource:
        type: com.alibaba.druid.pool.DruidDataSource
        driver-class-name: com.mysql.jdbc.Driver
        url: jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=utf8
        username: root
        password: root
      jpa:
        hibernate:
          ddl-auto: update
        show-sql: true
      application:
        name: microservice-student
      profiles: provider-hystrix-1004
    
    eureka:
      instance:
        hostname: localhost
        appname: microservice-student
        instance-id: microservice-student:1004
        prefer-ip-address: true
      client:
        service-url:
          defaultZone: http://eureka2001.ht.com:2001/eureka/,http://eureka2002.ht.com:2002/eureka/,http://eureka2003.ht.com:2003/eureka/
    
    info:
      groupId: com.ht.htSpringCloud
      artifactId: microservice-student-provider-hystrix-1004
      version: 1.0-SNAPSHOT
      userName: http://ht.com
      phone: 123456

    启动类StudentProviderHystrixApplication_1004中加下注解支持 @EnableCircuitBreaker

    package com.ht.microservicestudentproviderhystrix1004;
    
    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    import org.springframework.boot.autoconfigure.domain.EntityScan;
    import org.springframework.cloud.client.circuitbreaker.EnableCircuitBreaker;
    import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
    
    @EnableCircuitBreaker
    @EntityScan("com.ht.*.*")
    @EnableEurekaClient
    @SpringBootApplication
    public class MicroserviceStudentProviderHystrix1004Application {
    
        public static void main(String[] args) {
            SpringApplication.run(MicroserviceStudentProviderHystrix1004Application.class, args);
        }
    
    }

    服务提供者1004中controller新增

    package com.ht.microservicestudentproviderhystrix1004.controller;
    
    import com.ht.microservicecommon.entity.Student;
    import com.ht.microservicestudentproviderhystrix1004.service.StudentService;
    import com.netflix.hystrix.contrib.javanica.annotation.HystrixCommand;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.beans.factory.annotation.Value;
    import org.springframework.web.bind.annotation.*;
    
    import java.util.HashMap;
    import java.util.List;
    import java.util.Map;
    
    @RestController
    @RequestMapping("/student")
    public class StudentProviderController {
     
        @Autowired
        private StudentService studentService;
         
        @PostMapping(value="/save")
        public boolean save(Student student){
            try{
                studentService.save(student);
                return true;
            }catch(Exception e){
                return false;
            }
        }
         
        @GetMapping(value="/list")
        public List<Student> list(){
            return studentService.list();
        }
         
        @GetMapping(value="/get/{id}")
        public Student get(@PathVariable("id") Integer id){
            return studentService.findById(id);
        }
         
        @GetMapping(value="/delete/{id}")
        public boolean delete(@PathVariable("id") Integer id){
            try{
                studentService.delete(id);
                return true;
            }catch(Exception e){
                return false;
            }
        }
        @Value("${server.port}")
        private String port;
    
        @RequestMapping("/ribbon")
        public String ribbon(){
            return "工号【"+port+"】正在为您服务";
        }
        /**
         * 测试Hystrix服务降级
         * @return
         * @throws InterruptedException
         */
        @ResponseBody
        @GetMapping(value="/hystrix")
        @HystrixCommand(fallbackMethod="hystrixFallback")
        public Map<String,Object> hystrix() throws InterruptedException{
    //        Thread.sleep(2000);
            Map<String,Object> map=new HashMap<String,Object>();
            map.put("code", 200);
            map.put("info","工号【"+port+"】正在为您服务");
            return map;
        }
    
        public Map<String,Object> hystrixFallback() throws InterruptedException{
            Map<String,Object> map=new HashMap<String,Object>();
            map.put("code", 500);
            map.put("info", "系统【"+port+"】繁忙,稍后重试");
            return map;
        }
    }

    microservice-student-consumer-80项目也要对应的加个方法

    /**
         * 测试Hystrix服务降级
         * @return
         */
        @GetMapping(value="/hystrix")
        @ResponseBody
        public Map<String,Object> hystrix(){
            return restTemplate.getForObject(SERVER_IP_PORT+"/student/hystrix/", Map.class);
        }

    然后我们来测试下

    先启动三个eureka,再启动带hystrix的provider,最后启动普通的consumer;

    浏览器:http://localhost/student/hystrix

    Hystrix默认请求时间为1秒,从上面代码可以看到用线程给他设置了2秒,它就自动进入了我们自定义错误的方法,那么就防止了系统雪崩的问题

    这里给他设置了splee为一秒,就访问成功


    Hystrix默认超时时间设置

    Hystrix默认超时时间是1秒,我们可以通过hystrix源码看到,

    找到 hystrix-core.jar com.netflix.hystrix包下的HystrixCommandProperties类

    default_executionTimeoutInMilliseconds属性局势默认的超时时间

     用ctrl+shift+R在输入类名 就能快速找到

     默认1000毫秒 1秒

    我们系统里假如要自定义设置hystrix的默认时间的话;

    就在application.yml配置文件加上,这里设置了3秒

    hystrix:
      command:
        default:
          execution:
            isolation:
              thread:
                timeoutInMilliseconds: 3000


    Hystrix服务监控Dashboard

    Hystrix服务监控Dashboard仪表盘

    Hystrix提供了 准实时的服务调用监控项目Dashboard,能够实时记录通过Hystrix发起的请求执行情况,

    可以通过图表的形式展现给用户看。

    我们新建项目:microservice-student-consumer-hystrix-dashboard-90

     

     增加项目依赖pom

    
    
    <?xml version="1.0" encoding="UTF-8"?>
    <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 https://maven.apache.org/xsd/maven-4.0.0.xsd">
        <modelVersion>4.0.0</modelVersion>
        <parent>
            <groupId>com.ht</groupId>
            <artifactId>htSpringCloud</artifactId>
            <version>1.0-SNAPSHOT</version>
        </parent>
        <artifactId>microservice-student-consumer-hystrix-dashboard-90</artifactId>
    
        <properties>
            <java.version>1.8</java.version>
        </properties>
    
        <dependencies>
            <!--Hystrix服务监控Dashboard依赖-->
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-starter-hystrix</artifactId>
            </dependency>
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-starter-hystrix-dashboard</artifactId>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-actuator</artifactId>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-test</artifactId>
                <scope>test</scope>
            </dependency>
        </dependencies>
    
        <build>
            <plugins>
                <plugin>
                    <groupId>org.springframework.boot</groupId>
                    <artifactId>spring-boot-maven-plugin</artifactId>
                </plugin>
            </plugins>
        </build>
    
    </project>

    application.yml配置

    server:
      port: 90
      context-path: /

    在启动类加注解:@EnableHystrixDashboard

    package com.example.demo;
    
    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
    import org.springframework.boot.autoconfigure.orm.jpa.HibernateJpaAutoConfiguration;
    import org.springframework.cloud.netflix.hystrix.dashboard.EnableHystrixDashboard;
    
    @SpringBootApplication(exclude={DataSourceAutoConfiguration.class, HibernateJpaAutoConfiguration.class})
    @EnableHystrixDashboard
    public class DemoApplication {
    
        public static void main(String[] args) {
            SpringApplication.run(DemoApplication.class, args);
        }
    
    }

    启动项目在浏览器中访问http://localhost:90/hystrix

    成功进入

    然后我们来测试下;

    我们启动三个eureka,然后再启动microservice-student-provider-hystrix-1004

    我们直接请求http://localhost:1004/student/hystrix

    返回正常业务

    我们监控的话,http://localhost:1004/hystrix.stream 这个路径即可;

    一直是ping,然后data返回数据;

     

    谢谢观看!

  • 相关阅读:
    Python的logging模块
    Python中的json模块
    Python的re模块
    NoSQL简介
    单例设计模式
    基于配置文件的方式配置AOP
    重用切点表达式
    切面优先级
    返回通知、异常通知、环绕通知
    后置通知
  • 原文地址:https://www.cnblogs.com/huangting/p/11914229.html
Copyright © 2020-2023  润新知