讲一讲Spring Ioc的底层原理 并手写一套

时间: 2024-05-11 21:13:51 浏览: 12
Spring Ioc的底层原理是基于反射和依赖注入实现的。在Spring容器启动时,会扫描所有的类,找到其中被注解为@Component、@Service、@Controller、@Repository等的类,并将其实例化为Bean对象,同时将其放入Bean工厂中进行管理。在Bean实例化的过程中,Spring使用反射机制调用构造方法或工厂方法来创建对象,并通过setter或构造方法注入依赖的其他Bean对象。这样,通过Ioc容器的管理,实现了对象之间的解耦和灵活性。 手写一套简单的Spring Ioc框架: 1. 定义注解@ComponentScan用于扫描指定包下的所有类,将被注解为@Component、@Service、@Controller、@Repository等的类实例化为Bean对象,并将其放入Bean工厂中进行管理。 ```java @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) public @interface ComponentScan { String[] basePackages(); } ``` 2. 定义注解@Component用于标注需要实例化为Bean对象的类。 ```java @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) public @interface Component { String value() default ""; } ``` 3. 定义Bean工厂类,存储所有的Bean对象。 ```java public class BeanFactory { private Map<String, Object> beans = new ConcurrentHashMap<>(); public Object getBean(String name) { return beans.get(name); } public void addBean(String name, Object bean) { beans.put(name, bean); } } ``` 4. 定义Bean扫描器,扫描指定包下的所有类,并将被注解为@Component、@Service、@Controller、@Repository等的类实例化为Bean对象,并将其放入Bean工厂中进行管理。 ```java public class BeanScanner { private BeanFactory beanFactory; public BeanScanner(BeanFactory beanFactory) { this.beanFactory = beanFactory; } public void scan(String... basePackages) { for (String basePackage : basePackages) { String path = basePackage.replaceAll("\\.", "/"); File file = new File(this.getClass().getResource("/" + path).getFile()); if (!file.exists()) { continue; } if (file.isDirectory()) { for (File subFile : file.listFiles()) { scan(basePackage + "." + subFile.getName().replaceAll(".class", "")); } } else { String className = basePackage + "." + file.getName().replaceAll(".class", ""); try { Class<?> clazz = Class.forName(className); if (clazz.isAnnotationPresent(Component.class)) { Object bean = clazz.getDeclaredConstructor().newInstance(); beanFactory.addBean(clazz.getName(), bean); } } catch (Exception e) { e.printStackTrace(); } } } } } ``` 5. 定义注解@Autowired用于依赖注入。 ```java @Target(ElementType.FIELD) @Retention(RetentionPolicy.RUNTIME) public @interface Autowired { } ``` 6. 定义Bean注入器,通过反射机制注入依赖的Bean对象。 ```java public class BeanInjector { private BeanFactory beanFactory; public BeanInjector(BeanFactory beanFactory) { this.beanFactory = beanFactory; } public void inject() { for (Object bean : beanFactory.beans.values()) { Field[] fields = bean.getClass().getDeclaredFields(); for (Field field : fields) { if (field.isAnnotationPresent(Autowired.class)) { try { field.setAccessible(true); Object dependencyBean = beanFactory.getBean(field.getType().getName()); field.set(bean, dependencyBean); } catch (IllegalAccessException e) { e.printStackTrace(); } } } } } } ``` 7. 定义Spring Ioc框架的入口类,启动容器,并扫描指定包下的所有类,实例化为Bean对象,并将其放入Bean工厂中进行管理,最后注入依赖的Bean对象。 ```java public class SpringIoc { public static void run(Class<?> clazz, String[] args) { BeanFactory beanFactory = new BeanFactory(); BeanScanner beanScanner = new BeanScanner(beanFactory); ComponentScan componentScan = clazz.getDeclaredAnnotation(ComponentScan.class); beanScanner.scan(componentScan.basePackages()); BeanInjector beanInjector = new BeanInjector(beanFactory); beanInjector.inject(); } } ``` 使用示例: 1. 定义两个类,一个依赖另一个。 ```java @Component public class UserService { public void sayHello() { System.out.println("Hello, World!"); } } @Component public class UserController { @Autowired private UserService userService; public void hello() { userService.sayHello(); } } ``` 2. 定义启动类,指定需要扫描的包,启动Spring Ioc容器。 ```java @ComponentScan(basePackages = "com.example") public class Application { public static void main(String[] args) { SpringIoc.run(Application.class, args); UserController userController = (UserController) BeanFactory.getBean("com.example.UserController"); userController.hello(); } } ``` 输出结果: ``` Hello, World! ```

相关推荐

最新推荐

recommend-type

Spring实现一个简单的SpringIOC容器

本篇文章主要介绍了Spring实现一个简单的SpringIOC容器,具有一定的参考价值,感兴趣的小伙伴们可以参考一下。
recommend-type

Spring中IoC优点与缺点解析

主要为大家详细解析了Spring中IoC优点与缺点,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
recommend-type

Spring为IOC容器注入Bean的五种方式详解

主要介绍了Spring为IOC容器注入Bean的五种方式详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
recommend-type

grpcio-1.63.0-cp38-cp38-linux_armv7l.whl

Python库是一组预先编写的代码模块,旨在帮助开发者实现特定的编程任务,无需从零开始编写代码。这些库可以包括各种功能,如数学运算、文件操作、数据分析和网络编程等。Python社区提供了大量的第三方库,如NumPy、Pandas和Requests,极大地丰富了Python的应用领域,从数据科学到Web开发。Python库的丰富性是Python成为最受欢迎的编程语言之一的关键原因之一。这些库不仅为初学者提供了快速入门的途径,而且为经验丰富的开发者提供了强大的工具,以高效率、高质量地完成复杂任务。例如,Matplotlib和Seaborn库在数据可视化领域内非常受欢迎,它们提供了广泛的工具和技术,可以创建高度定制化的图表和图形,帮助数据科学家和分析师在数据探索和结果展示中更有效地传达信息。
recommend-type

zigbee-cluster-library-specification

最新的zigbee-cluster-library-specification说明文档。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

MATLAB柱状图在信号处理中的应用:可视化信号特征和频谱分析

![matlab画柱状图](https://img-blog.csdnimg.cn/3f32348f1c9c4481a6f5931993732f97.png) # 1. MATLAB柱状图概述** MATLAB柱状图是一种图形化工具,用于可视化数据中不同类别或组的分布情况。它通过绘制垂直条形来表示每个类别或组中的数据值。柱状图在信号处理中广泛用于可视化信号特征和进行频谱分析。 柱状图的优点在于其简单易懂,能够直观地展示数据分布。在信号处理中,柱状图可以帮助工程师识别信号中的模式、趋势和异常情况,从而为信号分析和处理提供有价值的见解。 # 2. 柱状图在信号处理中的应用 柱状图在信号处理
recommend-type

用Spring boot和vue写一个登录注册界面

好的,这是一个比较常见的技术问题,我可以为您提供一些参考资料和思路。首先,您可以使用Spring Boot作为后端框架来处理用户认证和注册的逻辑,比如使用Spring Security实现用户登录认证。同时,您还需要设计相应的数据模型和数据库表结构来存储用户信息。在前端方面,您可以使用Vue.js作为框架来构建登录注册页面,使用Axios来发起API请求并和后端进行交互。当然,在实现过程中,还需要考虑一些具体细节,比如数据校验、安全性和用户体验等方面。希望这些信息能够帮助到您。
recommend-type

JSBSim Reference Manual

JSBSim参考手册,其中包含JSBSim简介,JSBSim配置文件xml的编写语法,编程手册以及一些应用实例等。其中有部分内容还没有写完,估计有生之年很难看到完整版了,但是内容还是很有参考价值的。
recommend-type

"互动学习:行动中的多样性与论文攻读经历"

多样性她- 事实上SCI NCES你的时间表ECOLEDO C Tora SC和NCESPOUR l’Ingén学习互动,互动学习以行动为中心的强化学习学会互动,互动学习,以行动为中心的强化学习计算机科学博士论文于2021年9月28日在Villeneuve d'Asq公开支持马修·瑟林评审团主席法布里斯·勒菲弗尔阿维尼翁大学教授论文指导奥利维尔·皮耶昆谷歌研究教授:智囊团论文联合主任菲利普·普雷教授,大学。里尔/CRISTAL/因里亚报告员奥利维耶·西格德索邦大学报告员卢多维奇·德诺耶教授,Facebook /索邦大学审查员越南圣迈IMT Atlantic高级讲师邀请弗洛里安·斯特鲁布博士,Deepmind对于那些及时看到自己错误的人...3谢谢你首先,我要感谢我的两位博士生导师Olivier和Philippe。奥利维尔,"站在巨人的肩膀上"这句话对你来说完全有意义了。从科学上讲,你知道在这篇论文的(许多)错误中,你是我可以依