04_dubbo_ioc
【dubbo的IOC实现方法】
dubbo的IOC具体实现在:T injectExtension( T instance )方法中,该方法在3个地方被使用:
ExtensionLoader.getExtensionLoader(Protocol.class)
--new ExtensionLoader<T>(type)
----getAdaptiveExtension()
------createAdaptiveExtension()
/**1.为创建好的AdaptiveExtensionClass实例进行属性注入**/
--------injectExtension((T) getAdaptiveExtensionClass().newInstance()) /**2.为创建好的Extension实例进行属性注入**/
ExtensionLoader.getExtension("dubbo")
--createExtension("dubbo")
----injectExtension(instance);
/**3.为创建好的wrapperClass实例进行属性注入**/
----injectExtension((T) wrapperClass.getConstructor(type).newInstance(instance)) //
【 injectExtension( T instance )源码】
injectExtension( T instance )方法的作用是通过instance对象的实例的setXxx(object)方法为instance的属性进行赋值,完成setter注入,类似Spring的ioc的经典注入方式。
详细步骤:
1.获取instance的setXxx方法,通过setXxx方法获取属性名称property和属性类型pt。
2.使用objectFactory创建属性property的对象实例object
3.执行method.invoke( instance, object ),注入property实例,即执行setXxx(object)方法
private T injectExtension(T instance) {
try {
if (objectFactory != null) {
for (Method method : instance.getClass().getMethods()) {
//成员变量的setXxx(T param)方法,例如:setName(String name)
if (method.getName().startsWith("set")
&& method.getParameterTypes().length == 1
&& Modifier.isPublic(method.getModifiers())) {
//获取参数param的类型的Class对象
Class<?> pt = method.getParameterTypes()[0];
try {
//通过方法名setXxx获取成员变量名xxx,例如setName-->name
String property = method.getName().length() > 3 ? method.getName().substring(3, 4).toLowerCase() + method.getName().substring(4) : "";
//实例化参数object
Object object = objectFactory.getExtension(pt, property);
if (object != null) {
//执行instance实例对应的setXxx(object)方法,参数是object
method.invoke(instance, object);
}
} catch (Exception e) {
logger.error("fail to inject via method " + method.getName()
+ " of interface " + type.getName() + ": " + e.getMessage(), e);
}
}
}
}
} catch (Exception e) {
logger.error(e.getMessage(), e);
}
return instance;
}
其中,比较重要的是
Object object = objectFactory.getExtension(pt, property);
这个方法,其中的ExtensionFactory objectFactory实例为 AdaptiveExtensionFactory实例,即
//伪代码
ExtensionFactory objectFactory = AdaptiveExtensionFactory实例
AdaptiveExtensionFactory的成员变量
factories = [SpringExtensionFactory实例, SpiExtensionFactory实例]
【 AdaptiveExtensionFactory 类源代码】
/**
* AdaptiveExtensionFactory
*
* @author william.liangf
*/
@Adaptive
public class AdaptiveExtensionFactory implements ExtensionFactory { private final List<ExtensionFactory> factories; public AdaptiveExtensionFactory() {
ExtensionLoader<ExtensionFactory> loader = ExtensionLoader.getExtensionLoader(ExtensionFactory.class);
List<ExtensionFactory> list = new ArrayList<ExtensionFactory>();
for (String name : loader.getSupportedExtensions()) {
list.add(loader.getExtension(name));
}
factories = Collections.unmodifiableList(list);
} /**
* 先调用SpiExtensionFactory来实例化
* 如果不行,再使用SpringExtensionFactory来实例化
**/
public <T> T getExtension(Class<T> type, String name) {
for (ExtensionFactory factory : factories) {
T extension = factory.getExtension(type, name);
if (extension != null) {
return extension;
}
}
return null;
} }
【SpiExtensionFactory源码】
/**
* SpiExtensionFactory
*
* @author william.liangf
*/
public class SpiExtensionFactory implements ExtensionFactory { public <T> T getExtension(Class<T> type, String name) {
//type是接口类型,且必须含有@SPI注解
if (type.isInterface() && type.isAnnotationPresent(SPI.class)) {
ExtensionLoader<T> loader = ExtensionLoader.getExtensionLoader(type);
//获取type的所有ExtensionClasses实现的key(例如:type是Protocol.class)
if (loader.getSupportedExtensions().size() > 0) {
//获取type的装饰类,如果有@Adaptive注解的类,则返回该类的实例,否则返回一个动态代理的实例(如:Protocol$Adaptive 的实例)
return loader.getAdaptiveExtension();
}
}
return null;
} }
结论:dubbo-spi相对于JDK-spi的一个好处,可以为SPI实现类注入SPI的装饰类或动态代理类。
【 SpringExtendsionFactory的源码 】
/**
* SpringExtensionFactory
*
* @author william.liangf
*/
public class SpringExtensionFactory implements ExtensionFactory { private static final Set<ApplicationContext> contexts = new ConcurrentHashSet<ApplicationContext>(); public static void addApplicationContext(ApplicationContext context) {
contexts.add(context);
} public static void removeApplicationContext(ApplicationContext context) {
contexts.remove(context);
} @SuppressWarnings("unchecked")
public <T> T getExtension(Class<T> type, String name) {
for (ApplicationContext context : contexts) {
//该context是否包含name的bean
if (context.containsBean(name)) {
//获取该name对应的bean,如果是懒加载或者多实例的bean,此时会实例化name对应的bean
Object bean = context.getBean(name);
//如果obj的类型是typ或者其子类,与instance相同
if (type.isInstance(bean)) {
return (T) bean;
}
}
}
return null;
} }
【参考文章】
https://www.cnblogs.com/java-zhao/p/7469786.html
04_dubbo_ioc的更多相关文章
随机推荐
- SD与SE的关系,以及异常值
很多刚进入实验室的同学对实验数据的标准差(SD)与标准误(SE)的含义搞不清,不知道自己的数据报告到底该用SD还是SE.这里对这两个概念进行一些介绍. 标准差(SD)强调raw data的Variat ...
- AS添加依赖库提示Manifest merger failed解决办法
今天在学习<Android权威编程指南>时 在project structure中添加recyclerview时提示错误 按照提示添加tools:replace标签还是报错 然后切换至bu ...
- Angular material mat-icon 资源参考_Maps
ul,li>ol { margin-bottom: 0 } dt { font-weight: 700 } dd { margin: 0 1.5em 1.5em } img { height: ...
- Keras中间层输出的两种方式,即特征图可视化
训练好的模型,想要输入中间层的特征图,有两种方式: 1. 通过model.get_layer的方式.创建新的模型,输出为你要的层的名字. 创建模型,debug状态可以看到模型中,base_model/ ...
- HDU 5442 后缀自动机(从环字符串选定一个位置 , 时针或顺时针走一遍,希望得到字典序最大)
http://acm.hdu.edu.cn/showproblem.php?pid=5442 题目大意: 给定一个字符串,可理解成环,然后选定一位置,逆时针或顺时针走一遍,希望得到字典序最大,如果同样 ...
- springcloud(三)-Eureka
Eureka是Netflix开源的一款提供服务注册和发现的产品,它提供了完整的Service Registry和Service Discovery实现.也是springcloud体系中最重要最核心的组 ...
- telent三种认证及vlan简单划分
实验一 telent三种认证方式登录 实验拓扑图如下: 操作过程: 1.认证模式为none R1操作: 1.system-view进入系统试图2.telnet server enable开 ...
- js小仓库
1.千分位分隔符 const toDecimalMark = num => num.toLocaleString("en-US"); console.log(toDecima ...
- UGUI Slider的onValueChanged事件
在本文,你将学到如何将UGUI Slider的onValueChanged事件进行统一管理. using System; using UnityEngine; using UnityEngine.UI ...
- maven——添加插件和添加依赖有什么区别?
依赖:运行时开发时都需要用到的jar包,比如项目中需要一个Json的jar包,就要添加一个依赖,这个依赖在项目运行时也需要,因此在项目打包时需要把这些依赖也打包进项目里: 插件:在项目开的发时需要,但 ...