完成DI 依赖注入功能
生活随笔
收集整理的這篇文章主要介紹了
完成DI 依赖注入功能
小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.
在之前的源碼分析中,我們已經(jīng)了解到,依賴注入的入口是從getBean()方法開始的,前面的IOC 手寫部分基本流程已通。先在V1ApplicationContext 中定義好IOC 容器,一個是V1BeanWrapper,一個是單例對象緩存
public class V1ApplicationContext extends V1DefaultListableBeanFactory implements V1BeanFactory {private String [] configLocations;private V1BeanDefinitionReader reader;//用來保證注冊式單例的容器private Map<String,Object> singletonBeanCacheMap = new HashMap<String, Object>();//用來存儲所有的被代理過的對象private Map<String,V1BeanWrapper> beanWrapperMap = new ConcurrentHashMap<String,V1BeanWrapper>();... }從getBean()開始
下面,我們從完善getBean()方法開始:
//依賴注入,從這里開始,通過讀取BeanDefinition中的信息 //然后,通過反射機制創(chuàng)建一個實例并返回 //Spring做法是,不會把最原始的對象放出去,會用一個BeanWrapper來進行一次包裝 //裝飾器模式: //1、保留原來的OOP關(guān)系 //2、我需要對它進行擴展,增強(為了以后AOP打基礎(chǔ)) public Object getBean(String beanName) throws Exception {V1BeanDefinition V1BeanDefinition = this.beanDefinitionMap.get(beanName);Object instance = null;//這個邏輯還不嚴謹,自己可以去參考Spring源碼//工廠模式 + 策略模式V1BeanPostProcessor postProcessor = new V1BeanPostProcessor();postProcessor.postProcessBeforeInitialization(instance,beanName);instance = instantiateBean(beanName,V1BeanDefinition);//3、把這個對象封裝到BeanWrapper中V1BeanWrapper beanWrapper = new V1BeanWrapper(instance);//singletonObjects//factoryBeanInstanceCache//4、把BeanWrapper存到IOC容器里面 // //1、初始化// //class A{ B b;} // //class B{ A a;} // //先有雞還是先有蛋的問題,一個方法是搞不定的,要分兩次//2、拿到BeanWraoper之后,把BeanWrapper保存到IOC容器中去this.factoryBeanInstanceCache.put(beanName,beanWrapper);postProcessor.postProcessAfterInitialization(instance,beanName);// //3、注入populateBean(beanName,new V1BeanDefinition(),beanWrapper);return this.factoryBeanInstanceCache.get(beanName).getWrappedInstance(); }private void populateBean(String beanName, V1BeanDefinition V1BeanDefinition, V1BeanWrapper V1BeanWrapper) {Object instance = V1BeanWrapper.getWrappedInstance();// V1BeanDefinition.getBeanClassName();Class<?> clazz = V1BeanWrapper.getWrappedClass();//判斷只有加了注解的類,才執(zhí)行依賴注入if(!(clazz.isAnnotationPresent(V1Controller.class) || clazz.isAnnotationPresent(V1Service.class))){return;}//獲得所有的fieldsField[] fields = clazz.getDeclaredFields();for (Field field : fields) {if(!field.isAnnotationPresent(V1Autowired.class)){ continue;}V1Autowired autowired = field.getAnnotation(V1Autowired.class);String autowiredBeanName = autowired.value().trim();if("".equals(autowiredBeanName)){autowiredBeanName = field.getType().getName();}//強制訪問field.setAccessible(true);try {//為什么會為NULL,先留個坑if(this.factoryBeanInstanceCache.get(autowiredBeanName) == null){ continue; } // if(instance == null){ // continue; // }field.set(instance,this.factoryBeanInstanceCache.get(autowiredBeanName).getWrappedInstance());} catch (IllegalAccessException e) {e.printStackTrace();}}} public class V1BeanPostProcessor {public Object postProcessBeforeInitialization(Object bean, String beanName) throws Exception {return bean;}public Object postProcessAfterInitialization(Object bean, String beanName) throws Exception {return bean;} }至此,DI 部分就完成了。
?
?
總結(jié)
以上是生活随笔為你收集整理的完成DI 依赖注入功能的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 高仿真的类-ApplicationCon
- 下一篇: MVC 顶层设计-HandlerMapp