简单实现Spring的IOC原理详解
控制反转(InversionofControl,缩写为IoC)
简单来说就是当自己需要一个对象的时候不需要自己手动去new一个,而是由其他容器来帮你提供;Spring里面就是IOC容器。
例如:
在Spring里面经常需要在Service这个装配一个Dao,一般是使用@Autowired注解:类似如下
public Class ServiceImpl{ @Autowired Dao dao; public void getData(){ dao.getData(); }
在这里未初始化Dao直接使用是会报出空指针异常的,那么在Spring里面的做法就是通过反射来将需要的类帮你加载进来。
下面是一个例子模拟了Spring的DI和IOC
首先写两个注解模拟Spring的注解:
Entity注解代表的是Spring的@Service @Target(ElementType.TYPE) // 类 @Retention(RetentionPolicy.RUNTIME) public @interface Entity { } 代表的是Spring里面的@Autowrid @Target(ElementType.FIELD) //描述方法的 @Retention(RetentionPolicy.RUNTIME) // 仅运行时保留 public @interface Resources { }
当注解建立完成之后再建立两个类:
Rain类代表的是需要从其他地方获取天气数据(数据库或者服务器)
public class Rain { public void rain(){ System.out.println("正在下雨"); // 为了方便直接写了 } }
Weather类代表的是获取到的天气数据
@Entity public class Weather { @Resources Rain rain; // 这里在后面通过反射直接注入rain public void weather_rain() { rain.rain(); }
下面是通过反射来直接注入:
首先遍历指定的包名:这一步先省略,
首先是建立一个List模拟Spring的bean容器,即将已经装初始化好的带有Entity注解的类全部初始化
public class Weather_reflec { List<Object> objectList ; // 模拟Spring容器 public Weather_reflec() { objectList= new ArrayList<Object>(); } // 在这里其实最好的做法是先找出带有注解的类,判断带有Entity注解再传入。但是为了方便直接省略了 public void get_ref(Object object) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException { Class<?> clazz =object.getClass(); if(clazz.isAnnotationPresent(Entity.class)){ //判断是否带有Entity注解 Field[] fields =clazz.getDeclaredFields(); //获取其变量 for (Field field :fields){ if(field.isAnnotationPresent(Resources.class)){ //判断是否需要注入 Class<?> rainClass=Class.forName(field.getType().getName(),false,Thread.currentThread().getContextClassLoader()); // 这里先将Rain类加载 field.set(object,rainClass.newInstance()); //赋给rain objectList.add(object); //最后将已将赋值后的Weather保存进容器 } } } } public List<Object> returnList(){ return objectList; //返回容器方便以后使用 }
最后也就是模拟Controller里面直接使用的
public class WeatherPrediction { public static void main(String args[]) throws ClassNotFoundException, NoSuchMethodException, InstantiationException, IllegalAccessException, InvocationTargetException { WeatherPrediction weatherPrediction =new WeatherPrediction(); Weather weather =(Weather)weatherPrediction.springDo(); weather.weather_rain(); // 这里如果是普通调用会报空指针异常,而容器却为其将rain这个变量赋值了,所以可以正常输出 } /* 模拟Spring启动过程,这一步其实可以单独写一个类,这一步是容器该做的,而我们并不需要去管 */ public Object springDo() throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, InstantiationException, IllegalAccessException { Weather_reflec weather_reflec =new Weather_reflec(); // 启动的时候就需要加载的 Weather weather =new Weather(); //扫描类注解后new操作然后进行下一步 weather_reflec.get_ref(weather); // 将其类里面的变量进行new操作并放入容器 Object object =weather_reflec.returnList().get(0); return object; } 运行后输出:正在下雨
在WeatherPrediction里面并没有对Rain进行一个new操作但是却可以使用,这应该是最简单的一个模拟Spring的IOC例子了,当然Spring的IOC容器比这个强大太多了,比如需要考虑线程安全,以及各种的细节问题
总结
以上就是本文关于简单实现Spring的IOC原理详解的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站:
如有不足之处,欢迎留言指出。感谢朋友们对本站的支持!
相关文章
SpringBoot项目中配置application.yml中server.port不生效的问题
这篇文章主要介绍了SpringBoot项目中配置application.yml中server.port不生效的问题及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教2023-12-12SpringBoot @CompentScan excludeFilters配置无效的解决方案
这篇文章主要介绍了SpringBoot @CompentScan excludeFilters配置无效的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教2021-11-11SpringBoot 集成 ShedLock 分布式锁的示例详解
ShedLock是一个在分布式环境中使用的定时任务框架,用于解决在分布式环境中的多个实例的相同定时任务在同一时间点重复执行的问题,本文重点给大家介绍SpringBoot 分布式锁ShedLock的相关知识,感兴趣的朋友一起看看吧2021-08-08
最新评论