程序员

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

2020-07-20  本文已影响0人  M4Y

一、slot详解

slot概述

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

下面是关系结构图

ProcessorSlot子类及实现类.png

solt的基本逻辑及代码演示

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

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

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

我们使用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));
}
上一篇下一篇

猜你喜欢

热点阅读