第三章 dubbo内核之ioc源码解析
dubbo的IOC具體實(shí)現(xiàn)在:T injectExtension(T instance)方法中。該方法只在三個(gè)地方被使用:
1 createAdaptiveExtension() 2 --injectExtension((T) getAdaptiveExtensionClass().newInstance()) //為創(chuàng)建好的AdaptiveExtensionClass實(shí)例進(jìn)行屬性注入 3 4 createExtension(String name) 5 --injectExtension(instance) //為創(chuàng)建好的Extension實(shí)例進(jìn)行屬性注入 6 --injectExtension((T) wrapperClass.getConstructor(type).newInstance(instance)) //為創(chuàng)建好的wrapperClass實(shí)例進(jìn)行屬性注入來看一下源碼:
1 /** 2 * dubbo-IOC的核心 3 */ 4 private T injectExtension(T instance) { 5 try { 6 if (objectFactory != null) { 7 for (Method method : instance.getClass().getMethods()) { 8 if (method.getName().startsWith("set") && method.getParameterTypes().length == 1 9 && Modifier.isPublic(method.getModifiers())) {//一個(gè)參數(shù)的public的setXXX(T param)方法.例如,setName(String name) 10 Class<?> pt = method.getParameterTypes()[0];//參數(shù)param的類型T,eg.String 11 try { 12 String property = method.getName().length() > 3 13 ? method.getName().substring(3, 4).toLowerCase() + method.getName().substring(4) : "";//獲取屬性名XXX, eg.name 14 Object object = objectFactory.getExtension(pt, property);//實(shí)例化參數(shù) 15 if (object != null) { 16 //執(zhí)行instance.method(object)方法,這里就是執(zhí)行instance的setter方法,進(jìn)行setter注入 17 method.invoke(instance, object); 18 } 19 } catch (Exception e) { 20 logger.error("fail to inject via method " + method.getName() + " of interface " 21 + type.getName() + ": " + e.getMessage(), 22 e); 23 } 24 } 25 } 26 } 27 } catch (Exception e) { 28 logger.error(e.getMessage(), e); 29 } 30 return instance; 31 }整個(gè)方法的作用就是通過instance對(duì)象實(shí)例的setter方法為instance的屬性賦值,完成setter注入,即IOC的最經(jīng)典的注入方式。
詳細(xì)步驟:
- 獲取instance的setter方法,通過setter方法獲取屬性名稱property和屬性類型pt(即paramType的簡(jiǎn)寫)
- 使用objectFactory創(chuàng)建一個(gè)property名稱(類型為pt)的對(duì)象實(shí)例
- 執(zhí)行instance的setter方法,注入property實(shí)例
其中,比較重要的就是:Object object = objectFactory.getExtension(pt, property);這個(gè)方法。其中的objectFactory=AdaptiveExtensionFactory實(shí)例,其屬性factories = [SpringExtensionFactory實(shí)例, SpiExtensionFactory實(shí)例]。
看一下源碼:
1 private final List<ExtensionFactory> factories; 2 3 public <T> T getExtension(Class<T> type, String name) { 4 /** 5 * 先調(diào)用SpiExtensionFactory來實(shí)例化; 6 * 如果不行,再使用SpringExtensionFactory來實(shí)例化 7 */ 8 for (ExtensionFactory factory : factories) { 9 T extension = factory.getExtension(type, name); 10 if (extension != null) { 11 return extension; 12 } 13 } 14 return null; 15 }看一下SpiExtensionFactory的源碼:
1 public class SpiExtensionFactory implements ExtensionFactory { 2 public <T> T getExtension(Class<T> type, String name) { 3 if (type.isInterface() && type.isAnnotationPresent(SPI.class)) {//type是接口且必須具有@SPI注解 4 ExtensionLoader<T> loader = ExtensionLoader.getExtensionLoader(type); 5 if (loader.getSupportedExtensions().size() > 0) {//獲取type的所有ExtensionClasses實(shí)現(xiàn)的key 6 return loader.getAdaptiveExtension();//獲取type的裝飾類,如果有@Adaptive注解的類,則返回該類的實(shí)例,否則返回一個(gè)動(dòng)態(tài)代理類的實(shí)例(例如Protocol$Adpative的實(shí)例) 7 } 8 } 9 return null; 10 } 11 }從這里我們可以看出dubbo-SPI的另外一個(gè)好處:可以為SPI實(shí)現(xiàn)類注入SPI的裝飾類或動(dòng)態(tài)代理類。
看一下SpringExtensionFactory的源碼:
1 public class SpringExtensionFactory implements ExtensionFactory { 2 private static final Set<ApplicationContext> contexts = new ConcurrentHashSet<ApplicationContext>(); 3 4 public static void addApplicationContext(ApplicationContext context) { 5 contexts.add(context); 6 } 7 8 public static void removeApplicationContext(ApplicationContext context) { 9 contexts.remove(context); 10 } 11 12 @SuppressWarnings("unchecked") 13 public <T> T getExtension(Class<T> type, String name) { 14 for (ApplicationContext context : contexts) { 15 if (context.containsBean(name)) {//該context是否包含name的bean 16 Object bean = context.getBean(name);//獲取name的bean,如果是懶加載或多例的bean,此時(shí)會(huì)實(shí)例化name的bean 17 if (type.isInstance(bean)) {//如果obj的類型是type或其子類,與instanceof相同 18 return (T) bean; 19 } 20 } 21 } 22 return null; 23 } 24 }至此,IOC就干完了。但是有一個(gè)遺留問題,ApplicationContext是什么時(shí)候加入到contexts中呢?當(dāng)講解ServiceBean的時(shí)候來說。
轉(zhuǎn)載于:https://www.cnblogs.com/java-zhao/p/7469786.html
總結(jié)
以上是生活随笔為你收集整理的第三章 dubbo内核之ioc源码解析的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 404 not found ------
- 下一篇: CCF 201703-3 Markdow