《一起学sentinel》二、初探sentinel的Slot

简介: 随着微服务的流行,服务和服务之间的稳定性变得越来越重要。Sentinel 以流量为切入点,从流量控制、熔断降级、系统负载保护等多个维度保护服务的稳定性。

slot概述

在 Sentinel 里面,所有的资源都对应一个资源名称(resourceName),每次资源调用都会创建一个 Entry 对象。Entry 可以通过对主流框架的适配自动创建,也可以通过注解的方式或调用 SphU API 显式创建。Entry 创建候,同时也会创建一系列功能插槽(slot chain),这些插槽有不同的职责,例如:

  • NodeSelectorSlot 负责收集资源的路径,并将这些资源的调用路径,以树状结构存储起来,用于根据调用路径来限流降级
  • ClusterBuilderSlot 则用于存储资源的统计信息以及调用者信息,例如该资源的 RT, QPS, thread count 等等,这些信息将用作为多维度限流,降级的依据
  • StatisticSlot 则用于记录、统计不同纬度的 runtime 指标监控信息
  • FlowSlot 则用于根据预设的限流规则以及前面 slot 统计的状态,来进行流量控制
  • AuthoritySlot 则根据配置的黑白名单和调用来源信息,来做黑白名单控制
  • DegradeSlot 则通过统计信息以及预设的规则,来做熔断降级
  • SystemSlot 则通过系统的状态,例如 load1 等,来控制总的入口流量

下面是关系结构图

ProcessorSlot子类及实现类.png


solt的基本逻辑及代码演示

每个Slot执行完业务逻辑处理后,会调用fireEntry()方法,该方法将会触发下一个节点的entry方法,下一个节点又会调用他的fireEntry,以此类推直到最后一个Slot,由此就形成了sentinel的责任链。

  • 工作流概述:
    image.png

------

下面我会根据slot 的基本实现processorSlot讲一下slot 的基本结构及用法

------

  • 先看看顶层接口ProcessorSlot

    public interface ProcessorSlot<T> {
        void entry(....); //开始入口
        void fireEntry(....);//finish意味着结束
        void exit(....);//退出插槽
        void fireExit(....);//退出插槽结束
    }

​ 这个接口有4个方法,entry,fireEntry,exit,fireExit

  • ProcessorSlot 的抽象实现 AbstractLinkedProcessorSlot

    public abstract class AbstractLinkedProcessorSlot<T> implements ProcessorSlot<T> {
    
        private AbstractLinkedProcessorSlot<?> next = null;
    
        @Override
        public void fireEntry(... ) throws Throwable {
            //当业务执行完毕后,如果还有下一个slot
            if (next != null) {
                next.transformEntry(context, resourceWrapper, obj, count, prioritized, args);
            }
        }
    
        @SuppressWarnings("unchecked")
        //指向下一个slot的entry,每一个slot根据自己的职责不同,有自己的实现
        void transformEntry(... ) throws Throwable {
            T t = (T)o;
            entry(context, resourceWrapper, t, count, prioritized, args);
        }
    
        @Override
        public void fireExit(... ) {
            //当一个slot的exit执行完毕后,如果还有下一个未关闭slot
            if (next != null) {
                //指向下一个slot的exit
                next.exit(context, resourceWrapper, count, args);
            }
        }
    
        public AbstractLinkedProcessorSlot<?> getNext() {
            return next;
        }
    
        public void setNext(AbstractLinkedProcessorSlot<?> next) {
            this.next = next;
        }
    
    }
    • DefaultProcessorSlotChain实现了上述的chain(setNext和getNext)

      public class DefaultProcessorSlotChain extends ProcessorSlotChain {
          //直接实现了AbstractLinkedProcessorSlot的实例并作为first,可以理解为当前slot
          AbstractLinkedProcessorSlot<?> first = new AbstractLinkedProcessorSlot<Object>() {
      
              @Override
              public void entry(... )
                  throws Throwable {
                  super.fireEntry(context, resourceWrapper, t, count, prioritized, args);
              }
      
              @Override
              public void exit(... ) {
                  super.fireExit(context, resourceWrapper, count, args);
              }
      
          };
          //默认的end(可以理解为当前的后一个slot)
          AbstractLinkedProcessorSlot<?> end = first;
      
          @Override
          public void addFirst(AbstractLinkedProcessorSlot<?> protocolProcessor) {
              protocolProcessor.setNext(first.getNext());
              first.setNext(protocolProcessor);
              //如果当前为最后一个
              if (end == first) {
                  end = protocolProcessor;
              }
          }
      
          @Override
          public void addLast(AbstractLinkedProcessorSlot<?> protocolProcessor) {
              //将后一个slot放进当前slot的next
              end.setNext(protocolProcessor);
              //将end指向后一个slot
              end = protocolProcessor;
          }
      }
  • AbstractLinkedProcessorSlot 的实例 DemoSlot :

    public class DemoSlot extends AbstractLinkedProcessorSlot<DefaultNode> {
    
        //开始入口
        @Override
        public void entry(Context context, ResourceWrapper resourceWrapper, DefaultNode node, int count, boolean prioritized, Object... args)
                throws Throwable {
            //finish意味着结束
            fireEntry(context, resourceWrapper, node, count, prioritized, args);
        }
        //退出插槽
        @Override
        public void exit(Context context, ResourceWrapper resourceWrapper, int count, Object... args) {
            //退出插槽结束
            fireExit(context, resourceWrapper, count, args);
        }
    }

到这里我们看完了Slot的基本执行过程,总结一下

  • 1.初始化firstendslot
  • 2.开始执行entry
  • 3.开始执行fireEntry并查询是否下一个slot,如果有则执行第2步
  • 4.开始执行exit
  • 5.开始执行fireExit并查询是否有下一个slot,如果有则执行第4步
  • 6.结束

我们使用Slot方式进行处理时,需要实现一个类似tomcat 的lifeCycle,但是差异是tomcatlifeCycle是一个使用异步事件的方式执行容器内逻辑,而sentinel 使用的是一种子父依赖关系的链式调用,强调了顺序性执行。

默认的各个插槽之间的顺序是固定的,因为有的插槽需要依赖其他的插槽计算出来的结果才能进行工作。

下面我们看看是如何保证顺序的

--

SLOT的加载

1.定义顺序

sentinel在每个实例化的slot上面备注了顺序的参数,如

@SpiOrder(-10000)
public class NodeSelectorSlot extends AbstractLinkedProcessorSlot<Object> {

这是一个自定义的注解,保存的内容主要就是上面的(-10000)作为顺序权重

2.SPI加载

默认的chain会调用sentinel的类加载工具SpiLoaderloadPrototypeInstanceListSorted(ProcessorSlot.class);

这个方法会将所有实现了ProcessorSlot的类,用SPI的方式加载

slot-spi-class.png

@SpiOrder(-10000)
public class NodeSelectorSlot 
@SpiOrder(-9000)
public class ClusterBuilderSlot
@SpiOrder(-8000)
public class LogSlot
@SpiOrder(-7000)    
public class StatisticSlot
@SpiOrder(-5000)
public class SystemSlot
@SpiOrder(-6000)
public class AuthoritySlot
@SpiOrder(-2000)
public class FlowSlot
@SpiOrder(-1000)
public class DegradeSlot

3.加载完后排序

 public static <T> List<T> loadPrototypeInstanceListSorted(Class<T> clazz) {
    //这里就是第二步的加载
    ServiceLoader<T> serviceLoader = ServiceLoaderUtil.getServiceLoader(clazz);

    List<SpiOrderWrapper<T>> orderWrappers = new ArrayList<>();
    //循环遍历加载
    for (T spi : serviceLoader) {
        //查询对应类的顺序(第一步)
        int order = SpiOrderResolver.resolveOrder(spi);
        //将顺序和类插入List(手动有序数组)
        SpiOrderResolver.insertSorted(orderWrappers, spi, order);
    }
}



//排序方法很简答
private static <T> void insertSorted(List<SpiOrderWrapper<T>> list, T spi, int order) {
    int idx = 0;
    for (; idx < list.size(); idx++) {
        //循环遍历定长的list,一次比对大小
        if (list.get(idx).getOrder() > order) {
            break;如果发现当前索引大于
        }
    }
    list.add(idx, new SpiOrderWrapper<>(order, spi));
}
目录
相关文章
|
8月前
|
监控 开发者 Sentinel
Sentinel解密之SlotChain中的各大SLot
Sentinel解密之SlotChain中的各大SLot
73 0
|
8月前
|
数据可视化 Sentinel 微服务
Sentinel解密:SlotChain中的SLot大揭秘
Sentinel解密:SlotChain中的SLot大揭秘
93 0
M4Y
|
存储 监控 Java
《一起学sentinel》六、Slot的子类及实现之FlowSlot和DegradeSlot
随着微服务的流行,服务和服务之间的稳定性变得越来越重要。Sentinel 以流量为切入点,从流量控制、熔断降级、系统负载保护等多个维度保护服务的稳定性。
M4Y
816 0
《一起学sentinel》六、Slot的子类及实现之FlowSlot和DegradeSlot
M4Y
|
存储 监控 算法
《一起学sentinel》五、Slot的子类及实现之AuthoritySlot和SystemSlot
随着微服务的流行,服务和服务之间的稳定性变得越来越重要。Sentinel 以流量为切入点,从流量控制、熔断降级、系统负载保护等多个维度保护服务的稳定性。
M4Y
919 0
《一起学sentinel》五、Slot的子类及实现之AuthoritySlot和SystemSlot
M4Y
|
存储 监控 API
《一起学sentinel》四、Slot的子类及实现之LogSlot和StatisticSlot
随着微服务的流行,服务和服务之间的稳定性变得越来越重要。Sentinel 以流量为切入点,从流量控制、熔断降级、系统负载保护等多个维度保护服务的稳定性。
M4Y
539 0
《一起学sentinel》四、Slot的子类及实现之LogSlot和StatisticSlot
M4Y
|
存储 监控 API
《一起学sentinel》三、Slot的子类及实现之NodeSelectorSlot和ClusterBuilderSlot
随着微服务的流行,服务和服务之间的稳定性变得越来越重要。Sentinel 以流量为切入点,从流量控制、熔断降级、系统负载保护等多个维度保护服务的稳定性。
M4Y
887 0
《一起学sentinel》三、Slot的子类及实现之NodeSelectorSlot和ClusterBuilderSlot
|
5月前
|
Java UED Sentinel
微服务守护神:Spring Cloud Sentinel,让你的系统在流量洪峰中稳如磐石!
【8月更文挑战第29天】Spring Cloud Sentinel结合了阿里巴巴Sentinel的流控、降级、熔断和热点规则等特性,为微服务架构下的应用提供了一套完整的流量控制解决方案。它能够有效应对突发流量,保护服务稳定性,避免雪崩效应,确保系统在高并发下健康运行。通过简单的配置和注解即可实现高效流量控制,适用于高并发场景、依赖服务不稳定及资源保护等多种情况,显著提升系统健壮性和用户体验。
104 1
|
7月前
|
监控 Java Sentinel
使用Sentinel进行服务调用的熔断和限流管理(SpringCloud2023实战)
Sentinel是面向分布式、多语言异构化服务架构的流量治理组件,主要以流量为切入点,从流量路由、流量控制、流量整形、熔断降级、系统自适应过载保护、热点流量防护等多个维度来帮助开发者保障微服务的稳定性。
176 3
|
3月前
|
负载均衡 算法 Java
蚂蚁面试:Nacos、Sentinel了解吗?Springcloud 核心底层原理,你知道多少?
40岁老架构师尼恩分享了关于SpringCloud核心组件的底层原理,特别是针对蚂蚁集团面试中常见的面试题进行了详细解析。内容涵盖了Nacos注册中心的AP/CP模式、Distro和Raft分布式协议、Sentinel的高可用组件、负载均衡组件的实现原理等。尼恩强调了系统化学习的重要性,推荐了《尼恩Java面试宝典PDF》等资料,帮助读者更好地准备面试,提高技术实力,最终实现“offer自由”。更多技术资料和指导,可关注公众号【技术自由圈】获取。
蚂蚁面试:Nacos、Sentinel了解吗?Springcloud 核心底层原理,你知道多少?
|
4月前
|
监控 Java Nacos
SpringCloud基础5——微服务保护、Sentinel
sentinel、雪崩问题、流量控制、隔离和降级、授权规则、规则持久化
SpringCloud基础5——微服务保护、Sentinel