国庆中秋特辑系列文章:
国庆中秋特辑(八)Spring Boot项目如何使用JPA
国庆中秋特辑(七)Java软件工程师常见20道编程面试题
国庆中秋特辑(六)大学生常见30道宝藏编程面试题
国庆中秋特辑(五)MySQL如何性能调优?下篇
国庆中秋特辑(四)MySQL如何性能调优?上篇
国庆中秋特辑(三)使用生成对抗网络(GAN)生成具有节日氛围的画作,深度学习框架 TensorFlow 和 Keras 来实现
国庆中秋特辑(二)浪漫祝福方式 使用生成对抗网络(GAN)生成具有节日氛围的画作
国庆中秋特辑(一)浪漫祝福方式 用循环神经网络(RNN)或长短时记忆网络(LSTM)生成祝福诗词
Spring IOC(Inversion of Control,控制反转)是 Spring 框架的核心特性之一,它通过解耦和依赖注入的方式简化了应用的组件开发和维护。在 Spring 框架中,有两个主要的 IOC 容器实现:一个是基于 XML 配置文件的 BeanFactory,另一个是基于 Java 类的 ApplicationContext。
这里我们以一个简单的案例来分析 Spring IOC 的工作原理。假设我们有一个简单的 Java 程序,需要用到一个数据持久层(DataAccess)和一个业务层(Service)。
首先,我们需要创建一个 Spring 配置文件(如:applicationContext.xml),在这个文件中,我们将 DataAccess 和 Service 作为 Bean 定义:
在这个配置文件中,我们定义了两个 Bean:一个是 DataAccess 类型的 Bean,另一个是 Service 类型的 Bean。Service Bean 中的 dataAccess 属性通过 ref 属性指定为 DataAccess Bean。
接下来,我们需要在 Java 程序中创建一个 Spring 的 IOC 容器,并从中获取 DataAccess 和 Service Bean:
import com.example.DataAccess; import com.example.Service; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class Main { public static void main(String[] args) { ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml"); DataAccess dataAccess = (DataAccess) context.getBean("dataAccess"); Service service = (Service) context.getBean("service"); // 使用 Service 进行业务操作 service.doSomething(); } }
在这个 Java 程序中,我们首先创建了一个 Spring 的 IOC 容器(ApplicationContext),然后通过 getBean 方法获取了 DataAccess 和 Service Bean。注意,由于 Service Bean 的 dataAccess 属性是通过 ref 属性指定为 DataAccess Bean,所以在获取 Service Bean 时,Spring 会自动将 DataAccess Bean 注入到 Service Bean 中。
现在,我们可以通过 Service 类的 doSomething 方法来调用 DataAccess 类的相关方法进行数据持久操作。这个过程就是 Spring IOC 的工作原理。Spring IOC 容器负责管理 Bean 的创建和管理,以及 Bean 之间的依赖关系,我们只需要关注业务逻辑的实现即可。
总结一下,Spring IOC 的核心思想是:不再由对象自己创建和管理它所依赖的对象,而是由外部(如 Spring 容器)负责注入依赖对象。这样可以大大简化对象的创建和管理,提高代码的可维护性和可扩展性。
// Resource 加载 public Resource getResourceByPath(String path) { if (path.startsWith("/")) { path = path.substring(1); } Resource[] resources = getResources(); for (Resource resource : resources) { if (resource.getPath().equals(path)) { return resource; } } return null; }
// BeanDefinitionReader 读取 BeanDefinition public void readBeanDefinitions(Resource resource) throws BeansException, IOException { if (!resource.exists()) { return; } try (InputStream reader = resource.getInputStream()) { while (reader.markSupported()) { int marker = reader.mark(); if (marker == XMLBeanDefinitionReader.ROOT_ELEMENT_START_MARKER) { // 创建 BeanDefinitionReader 对象 BeanDefinitionReader beanDefinitionReader = new BeanDefinitionReader(this); // 开始解析 XML 文件中的 BeanDefinition beanDefinitionReader.parse(reader); } else if (marker == XMLBeanDefinitionReader.ROOT_ELEMENT_END_MARKER) { // 解析结束 break; } } } }
// BeanFactory 创建 Bean public Object getBean(String name) throws BeansException { // 根据 Bean 名称获取 BeanDefinition BeanDefinition beanDefinition = getBeanDefinition(name); // 如果 BeanDefinition 不存在,抛出异常 if (beanDefinition == null) { throw new NoSuchBeanDefinitionException(name); } // 创建 Bean Object bean = createBean(name, beanDefinition); // 返回 Bean return bean; }
// DependencyAutowire 注入依赖 public void setDependency(String propertyName, @Qualifier @Nullable String[] qualifiedClassNames) { // 获取 BeanDefinition BeanDefinition beanDefinition = getBeanDefinition(propertyName); // 如果 BeanDefinition 不存在,抛出异常 if (beanDefinition == null) { throw new NoSuchBeanDefinitionException(propertyName); } // 获取依赖的 Bean @Qualifier("'" + beanDefinition.getBeanClassName() + "'") @Autowired Object dependency = getBeanFactory().getBean(propertyName, Object.class); // 设置依赖 setter.setValue(this, dependency); }
// Lifecycle 接口 public interface Lifecycle { void start() throws BeansException; void stop() throws BeansException; boolean isStarted(); boolean isStopped(); void destroy() throws BeansException; }
由于 Spring IOC 核心代码较长,无法在这里全部展示。但我可以简要介绍一下 Spring IOC 的核心部分,并提供部分关键代码示例。
BeanFactory 是 Spring IOC 容器的基本实现,它负责管理 Bean 的创建和管理。创建过程主要包括以下几个步骤:
以下是创建 BeanFactory 的示例代码:
public class BeanFactoryImpl implements BeanFactory { //... 省略其他代码... @Override protected void createBeanFactory(String name, BeanDefinitionRegistry registry) throws BeansException { // 创建并初始化一个 DefaultListableBeanFactory 实例 DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory(); beanFactory.setBeanDefinitionRegistry(registry); beanFactory.setResourceAccess(new ClassPathResource(name, getClassLoader())); beanFactory.setEnvironment(this.environment); beanFactory.setApplicationContextName(name); beanFactory.setParentBeanFactory(this); beanFactory.setSingletonObjects(new ConcurrentHashMap()); beanFactory.setDefaultSingletonBeanName(name); beanFactory.registerBeanDefinition(new RootBeanDefinition(beanFactory, name)); } }
Bean 的生命周期主要包括以下几个阶段:
以下是 Bean 生命周期的示例代码:
public class BeanFactoryImpl implements BeanFactory { //... 省略其他代码... @Override protected Object createBean(String name, BeanDefinition beanDefinition, Object[] args) throws BeansException { // 根据 BeanDefinition 创建 Bean 实例 Object bean = beanDefinition.getBean(); // 属性填充 for (PropertyValue propertyValue : beanDefinition.getPropertyValues()) { bean = propertyValue.resolve(bean); } // 初始化 bean = beanDefinition.initialize(bean); // 依赖注入 if (bean instanceof ConfigurableBeanFactory) { ((ConfigurableBeanFactory) bean).mergeBeanDefinitions(beanDefinition.getResourceDescription()); } // 将 Bean 添加到 singletonObjects 中,以 BeanName 为键 synchronized (this.singletonObjects) { this.singletonObjects.put(name, bean); } return bean; } }
Spring IOC 容器通过依赖注入(Dependency Injection,DI)的方式将 Bean 的依赖关系注入到 Bean 实例上。依赖注入主要有两种方式:构造器注入和 setter 方法注入。
以下是依赖注入的示例代码:
public class BeanFactoryImpl implements BeanFactory { //... 省略其他代码... @Override protected Object createBean(String name, BeanDefinition beanDefinition, Object[] args) throws BeansException { //... 省略其他代码... // 依赖注入 if (bean instanceof ConfigurableBeanFactory) { ((ConfigurableBeanFactory) bean).mergeBeanDefinitions(beanDefinition.getResourceDescription()); } //... 省略其他代码... } }
以上是 Spring IOC 核心代码的简要分析和示例。要了解更多关于 Spring IOC 的详细信息,建议参考 Spring 官方文档和相关教程。