dubbo源码分析-Dubbo SPI

package org.apache.dubbo.demo.consumer;

import org.apache.dubbo.common.extension.ExtensionLoader;

public class DubboSPITest {

    public static void main(String[] args) {
    	// 获取 extensionLoader。
        ExtensionLoader<Robot> extensionLoader =
                ExtensionLoader.getExtensionLoader(Robot.class);
        
        // 获取扩展类
        Robot optimusPrime = extensionLoader.getExtension("optimusPrime");
        optimusPrime.sayHello();
        
        Robot bumblebee = extensionLoader.getExtension("bumblebee");
        bumblebee.sayHello();
    }
}

在dubbo-common模块
在这里插入图片描述

我们首先通过 ExtensionLoader 的 getExtensionLoader 方法获取一个 ExtensionLoader 实例,然后再通过 ExtensionLoader 的 getExtension 方法获取拓展类对象。这其中,getExtensionLoader 方法用于从缓存中获取与拓展类对应的 ExtensionLoader,若缓存未命中,则创建一个新的实例。

// 类静态变量,所有实例共享这一个。
private static final ConcurrentMap<Class<?>, ExtensionLoader<?>> EXTENSION_LOADERS = new ConcurrentHashMap<>(64);

public static <T> ExtensionLoader<T> getExtensionLoader(Class<T> type) {
    if (type == null) { // type = interface org.apache.dubbo.demo.consumer.Robot
        throw new IllegalArgumentException("Extension type == null");
    }
    // 看看是不是接口,只能对接口拓展
    if (!type.isInterface()) {
        throw new IllegalArgumentException("Extension type (" + type + ") is not an interface!");
    }
    // 看看要拓展接口上有没有注解@SPI,这个必须要加上。
    if (!withExtensionAnnotation(type)) {
        throw new IllegalArgumentException("Extension type (" + type +
                ") is not an extension, because it is NOT annotated with @" + SPI.class.getSimpleName() + "!");
    }
    // 就是一个CHM,key = interface org.apache.dubbo.demo.consumer.Robot
    // value = org.apache.dubbo.common.extension.ExtensionLoader[org.apache.dubbo.demo.consumer.Robot]
    return (ExtensionLoader<T>) EXTENSION_LOADERS.computeIfAbsent(type, k -> new ExtensionLoader<T>(type));
}

下面我们从 ExtensionLoader 的 getExtension方法作为入口,对拓展类对象的获取过程进行详细的分析。

首先看ExtensionLoader的一些公共属性


// 缓存对象名,class=接口实现类,String=接口实现类的对象名(就是在配置文件中写的key=value的key)。
ConcurrentMap<Class<?>, String> cachedNames = new ConcurrentHashMap<>();

// 保存名字到Class的映射,其中Map<String, Class<?>>保存对象名,用Holder包装是为了线程安全
// "optimusPrime" -> "class org.apache.dubbo.demo.consumer.OptimusPrime"
Holder<Map<String, Class<?>>> cachedClasses = new Holder<>();

// 每个实现类一个Holder用来保存它们的实例(也就是上面的Holder)
ConcurrentMap<String, Holder<Object>> cachedInstances = new ConcurrentHashMap<>();

进入getExtension(String name)

public T getExtension(String name) {
	// 
	if (StringUtils.isEmpty(name)) {
	    throw new IllegalArgumentException("Extension name == null");
	}
	// 获取默认的拓展实现类
	if ("true".equals(name)) {
	    return getDefaultExtension();
	}
	// Holder,顾名思义,用于持有目标对象,进去看看是怎么获取的
	final Holder<Object> holder = getOrCreateHolder(name);
	// 双重检查锁生成单例对象。
	Object instance = holder.get();
	if (instance == null) {
	    synchronized (holder) {
	        instance = holder.get();
	        if (instance == null) {
	        	// 创建拓展实例,下一步进去看看
	            instance = createExtension(name);
	            // 设置实例到Holder中
	            holder.set(instance);
	        }
	    }
	}
	return (T) instance;
}

package org.apache.dubbo.common.utils;

public class Holder<T> {
    private volatile T value;
    public void set(T value) {
        this.value = value;
    }
    public T get() {
        return value;
    }
}

进入getOrCreateHolder(name);

ConcurrentMap<String, Holder<Object>> cachedInstances = new ConcurrentHashMap<>();

private Holder<Object> getOrCreateHolder(String name) {
	// cachedInstances就是一个实例名对应一个Holder,这个Holder用来保存对应实例
     Holder<Object> holder = cachedInstances.get(name);
     if (holder == null) {
         cachedInstances.putIfAbsent(name, new Holder<>());
         holder = cachedInstances.get(name);
     }
     return holder;
}

进入createExtension(name);

// 类静态变量,所有实例共享这一个。
private static final ConcurrentMap<Class<?>, Object> EXTENSION_INSTANCES = new ConcurrentHashMap<>(64);

private T createExtension(String name) {
    // 进入看看,大概意思是获取拓展类。应该是放到了缓存里面,从缓存里面获取
    Class<?> clazz = getExtensionClasses().get(name);
    if (clazz == null) {
        throw findException(name);
    }
    try {
    	// 从缓存中根据类型获取实例
        T instance = (T) EXTENSION_INSTANCES.get(clazz);
        if (instance == null) {
        	// 通过反射创建实例
        	// 用key=类型,value=实例对象
            EXTENSION_INSTANCES.putIfAbsent(clazz, clazz.newInstance());
            instance = (T) EXTENSION_INSTANCES.get(clazz);
        }
        // 向实例中注入依赖,dubbo的IOC
        injectExtension(instance);
        Set<Class<?>> wrapperClasses = cachedWrapperClasses;
        if (CollectionUtils.isNotEmpty(wrapperClasses)) {
        	// 循环创建 Wrapper 实例
            for (Class<?> wrapperClass : wrapperClasses) {
            // 将当前 instance 作为参数传给 Wrapper 的构造方法,并通过反射创建 Wrapper 实例。
            // 然后向 Wrapper 实例中注入依赖,最后将 Wrapper 实例再次赋值给 instance 变量
                instance = injectExtension((T) wrapperClass.getConstructor(type).newInstance(instance));
            }
        }
        initExtension(instance);
        return instance;
    } catch (Throwable t) {
        throw new IllegalStateException("Extension instance (name: " + name + ", class: " +
                type + ") couldn't be instantiated: " + t.getMessage(), t);
    }
}

createExtension 方法的逻辑稍复杂一下,包含了如下的步骤:

  1. 通过 getExtensionClasses 获取所有的拓展类
  2. 通过反射创建拓展对象
  3. 向拓展对象中注入依赖
  4. 将拓展对象包裹在相应的 Wrapper 对象中
    以上步骤中,第一个步骤是加载拓展类的关键,第三和第四个步骤是 Dubbo IOC 与 AOP 的具体实现。

进入getExtensionClasses()

// 缓存类信息
private final Holder<Map<String, Class<?>>> cachedClasses = new Holder<>();

// 首先看返回值,是个map,证明了就是先获取缓存,再从这个缓存中取出类
private Map<String, Class<?>> getExtensionClasses() {
     Map<String, Class<?>> classes = cachedClasses.get();
     if (classes == null) {
         synchronized (cachedClasses) {
             classes = cachedClasses.get();
             if (classes == null) {
             	// 加载拓展类
                 classes = loadExtensionClasses();
                 cachedClasses.set(classes);
             }
         }
     }
     return classes;
}

进入

private static final String SERVICES_DIRECTORY = "META-INF/services/";

private static final String DUBBO_DIRECTORY = "META-INF/dubbo/";

private static final String DUBBO_INTERNAL_DIRECTORY = DUBBO_DIRECTORY + "internal/";
private static LoadingStrategy DUBBO_INTERNAL_STRATEGY =  () -> DUBBO_INTERNAL_DIRECTORY;
private static LoadingStrategy DUBBO_STRATEGY = () -> DUBBO_DIRECTORY;
private static LoadingStrategy SERVICES_STRATEGY = () -> SERVICES_DIRECTORY;

private static LoadingStrategy[] strategies = new LoadingStrategy[] { DUBBO_INTERNAL_STRATEGY, DUBBO_STRATEGY, SERVICES_STRATEGY };


private Map<String, Class<?>> loadExtensionClasses() {
	// 在下面,进去看一下,缓存默认拓展名
    cacheDefaultExtensionName();
	// 拓展
    Map<String, Class<?>> extensionClasses = new HashMap<>();
	// 加载指定文件夹下的配置文件
    for (LoadingStrategy strategy : strategies) {
        loadDirectory(extensionClasses, strategy.directory(), type.getName(), strategy.preferExtensionClassLoader(), strategy.excludedPackages());
        loadDirectory(extensionClasses, strategy.directory(), type.getName().replace("org.apache", "com.alibaba"), strategy.preferExtensionClassLoader(), strategy.excludedPackages());
    }

    return extensionClasses;
}

private void cacheDefaultExtensionName() {
    // 获取 SPI 注解,这里的 type 变量是在调用 getExtensionLoader 方法时传入的
    final SPI defaultAnnotation = type.getAnnotation(SPI.class);
    if (defaultAnnotation == null) {
        return;
    }

    String value = defaultAnnotation.value();
    if ((value = value.trim()).length() > 0) {
        // 对 SPI 注解内容进行切分
        String[] names = NAME_SEPARATOR.split(value);
        // 检测 SPI 注解内容是否合法,不合法则抛出异常
        if (names.length > 1) {
            throw new IllegalStateException("More than 1 default extension name on extension " + type.getName()
                    + ": " + Arrays.toString(names));
        }
        // 设置默认名称,参考 getDefaultExtension 方法
        if (names.length == 1) {
            cachedDefaultName = names[0];
        }
    }
}

loadExtensionClasses 方法总共做了两件事情,一是对 SPI 注解进行解析,二是调用 loadDirectory 方法加载指定文件夹配置文件。

private void loadDirectory(Map<String, Class<?>> extensionClasses, String dir, String type,
                               boolean extensionLoaderClassLoaderFirst, String... excludedPackages) {
    // 文件夹路径 + type 接口全限定名 
    // fileName =  META-INF/dubbo/org.apache.dubbo.demo.consumer.Robot               
    String fileName = dir + type;
    try {
        Enumeration<java.net.URL> urls = null;
        // 获取到类加载器,进入下面的那个方法
        ClassLoader classLoader = findClassLoader();

        // try to load from ExtensionLoader's ClassLoader first
        if (extensionLoaderClassLoaderFirst) {
            ClassLoader extensionLoaderClassLoader = ExtensionLoader.class.getClassLoader();
            if (ClassLoader.getSystemClassLoader() != extensionLoaderClassLoader) {
                urls = extensionLoaderClassLoader.getResources(fileName);
            }
        }
		// 进入到这一步
        if(urls == null || !urls.hasMoreElements()) {
            // 根据文件名加载所有的同名文件
            if (classLoader != null) {
                urls = classLoader.getResources(fileName);
            } else {
                urls = ClassLoader.getSystemResources(fileName);
            }
        }
		// 加载资源
        if (urls != null) {
            while (urls.hasMoreElements()) {
                java.net.URL resourceURL = urls.nextElement();
                loadResource(extensionClasses, classLoader, resourceURL, excludedPackages);
            }
        }
    } catch (Throwable t) {
        logger.error("Exception occurred when loading extension class (interface: " +
                type + ", description file: " + fileName + ").", t);
    }
}

private static ClassLoader findClassLoader() {
   return ClassUtils.getClassLoader(ExtensionLoader.class);
}

获取到文件名
在这里插入图片描述
获取到类加载器
在这里插入图片描述

进入classLoader.getResources(fileName);获取所有的同名文件
在这里插入图片描述

首先拿到资源位置,然后去加载资源文件。进入loadResource方法。

在这里插入图片描述

进入loadResource方法


private void loadResource(Map<String, Class<?>> extensionClasses, ClassLoader classLoader,
                              java.net.URL resourceURL, String... excludedPackages) {
    try {
        try (BufferedReader reader = new BufferedReader(new InputStreamReader(resourceURL.openStream(), StandardCharsets.UTF_8))) {
            String line;
            // 按行读取配置内容
            while ((line = reader.readLine()) != null) {
                // 定位 # 字符,代表后面的是注释,需要删除
                final int ci = line.indexOf('#');
                if (ci >= 0) {
                     // 截取 # 之前的字符串,# 之后的内容为注释,需要忽略
                    line = line.substring(0, ci);
                }
                line = line.trim();
                if (line.length() > 0) {
                    try {
                        String name = null;
                         // 以等于号 = 为界,截取键与值
                        int i = line.indexOf('=');
                        if (i > 0) {
                            name = line.substring(0, i).trim();
                            line = line.substring(i + 1).trim();
                        }
                        if (line.length() > 0 && !isExcluded(line, excludedPackages)) {
                            // 加载类,并通过 loadClass 方法对类进行缓存,Class.forName是Java类库自带的(通过反射加载类),但是loadClass是dubbo自己的,进去看看怎么缓存的
                            loadClass(extensionClasses, resourceURL, Class.forName(line, true, classLoader), name);
                        }
                    } catch (Throwable t) {
                        IllegalStateException e = new IllegalStateException("Failed to load extension class (interface: " + type + ", class line: " + line + ") in " + resourceURL + ", cause: " + t.getMessage(), t);
                        exceptions.put(line, e);
                    }
                }
            }
        }
    } catch (Throwable t) {
        logger.error("Exception occurred when loading extension class (interface: " +
                type + ", class file: " + resourceURL + ") in " + resourceURL, t);
    }
}

loadResource 方法用于读取和解析配置文件,并通过反射加载类,最后调用 loadClass 方法进行其他操作。loadClass 方法用于主要用于操作缓存。

很明显就是我先拿到资源位置,就是我这个文件,然后一行行读取文件
在这里插入图片描述

进入loadClass方法, dubbo自己的,不是Java类库里面的那个


private void loadClass(Map<String, Class<?>> extensionClasses, java.net.URL resourceURL, Class<?> clazz, String name) throws NoSuchMethodException {
    if (!type.isAssignableFrom(clazz)) {
        throw new IllegalStateException("Error occurred when loading extension class (interface: " +
                type + ", class line: " + clazz.getName() + "), class "
                + clazz.getName() + " is not subtype of interface.");
    }
    // 检测目标类上是否有 Adaptive 注解
    if (clazz.isAnnotationPresent(Adaptive.class)) {
        // 设置 cachedAdaptiveClass缓存
        cacheAdaptiveClass(clazz);
    } else if (isWrapperClass(clazz)) {  // 检测 clazz 是否是 Wrapper 类型
        // 存储 clazz 到 cachedWrapperClasses 缓存中
        cacheWrapperClass(clazz);
    } else {
         // 程序进入此分支,表明 clazz 是一个普通的拓展类
         // 检测 clazz 是否有默认的构造方法,如果没有,则抛出异常
        clazz.getConstructor();
        if (StringUtils.isEmpty(name)) {
           // 如果 name 为空,则尝试从 Extension 注解中获取 name,或使用小写的类名作为 name
            name = findAnnotationName(clazz);
            if (name.length() == 0) {
                throw new IllegalStateException("No such extension name for the class " + clazz.getName() + " in the config " + resourceURL);
            }
        }
		 // 切分 name
        String[] names = NAME_SEPARATOR.split(name);
        if (ArrayUtils.isNotEmpty(names)) {
          // 如果类上有 Activate 注解,则使用 names 数组的第一个元素作为键,
          // 存储 name 到 Activate 注解对象的映射关系
            cacheActivateClass(clazz, names[0]);
            for (String n : names) {
                // 存储 Class 到名称的映射关系
                cacheName(clazz, n);
                 // 存储名称到 Class 的映射关系
                saveInExtensionClass(extensionClasses, clazz, n);
            }
        }
    }
}

在这里插入图片描述
就是缓存标注Activate注解的类,但是当前这个实现类并没有,所以跳过了。
在这里插入图片描述

cacheName缓存名字
在这里插入图片描述

private final ConcurrentMap<Class<?>, String> cachedNames = new ConcurrentHashMap<>();

 /**
  * cache name
  */
 private void cacheName(Class<?> clazz, String name) {
     if (!cachedNames.containsKey(clazz)) {
         cachedNames.put(clazz, name);
     }
 }

saveInExtensionClass保存名字到Class的映射,其实也就是初始化了extensionClass这个map,然后通过map获取到Class。在最开始的地方,也就是private Map<String, Class<?>> getExtensionClasses()。到这个地方,就是缓存这一块已经ok了,
在这里插入图片描述

之后回到这个方法,我的一个思考如图。

在这里插入图片描述

进入injectExtension(instance);

在这里插入图片描述

private T injectExtension(T instance) {
	
    if (objectFactory == null) {
        return instance;
    }

    try {
    	// 遍历目标类的所有方法
        for (Method method : instance.getClass().getMethods()) {
        	 // 检测方法是否以 set 开头,且方法仅有一个参数,且方法访问级别为 public
            if (!isSetter(method)) {
                continue;
            }
            /**
             * Check to see if we need auto injection for this property
             */
            if (method.getAnnotation(DisableInject.class) != null) {
                continue;
            }
            // 获取 setter 方法参数类型
            Class<?> pt = method.getParameterTypes()[0];
            if (ReflectUtils.isPrimitives(pt)) {
                continue;
            }

            try {
               // 获取属性名,比如 setName 方法对应属性名 name
                String property = getSetterProperty(method);
                 // 从 ObjectFactory 中获取依赖对象
                Object object = objectFactory.getExtension(pt, property);
                if (object != null) {
                   // 通过反射调用 setter 方法设置依赖
                    method.invoke(instance, object);
                }
            } catch (Exception e) {
                logger.error("Failed to inject via method " + method.getName()
                        + " of interface " + type.getName() + ": " + e.getMessage(), e);
            }

        }
    } catch (Exception e) {
        logger.error(e.getMessage(), e);
    }
    return instance;
}


private boolean isSetter(Method method) {
    return method.getName().startsWith("set")
            && method.getParameterTypes().length == 1
            && Modifier.isPublic(method.getModifiers());
}

在上面代码中,objectFactory 变量的类型为 AdaptiveExtensionFactory,AdaptiveExtensionFactory 内部维护了一个 ExtensionFactory 列表,用于存储其他类型的 ExtensionFactory。Dubbo 目前提供了两种 ExtensionFactory,分别是 SpiExtensionFactory 和 SpringExtensionFactory。前者用于创建自适应的拓展,后者是用于从 Spring 的 IOC 容器中获取所需的拓展。

dubbo的


public <T> T getExtension(Class<T> type, String name) {
    if (type.isInterface() && type.isAnnotationPresent(SPI.class)) {
        ExtensionLoader<T> loader = ExtensionLoader.getExtensionLoader(type);
        if (!loader.getSupportedExtensions().isEmpty()) {
            return loader.getAdaptiveExtension();
        }
    }
    return null;
}

Spring的

 public <T> T getExtension(Class<T> type, String name) {

    //SPI should be get from SpiExtensionFactory
    if (type.isInterface() && type.isAnnotationPresent(SPI.class)) {
        return null;
    }

    for (ApplicationContext context : CONTEXTS) {
        T bean = BeanFactoryUtils.getOptionalBean(context, name, type);
        if (bean != null) {
            return bean;
        }
    }

    logger.warn("No spring extension (bean) named:" + name + ", try to find an extension (bean) of type " + type.getName());

    return null;
}

在这里插入图片描述

在这里插入图片描述

loadResource()方法
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值