spring InitializingBean接口

简介:

最近工作需要得到sping中的每个事物需要执行的sql,称机会简单研究了一下spring的事务,项目中管理事务比较简单,用TransactionTemplate,就直接以TransactionTemplate为入口开始学习。 
TransactionTemplate的源码如下: 
public class TransactionTemplate extends DefaultTransactionDefinition 
        implements TransactionOperations, InitializingBean{ 
        . 
        . 
        . 
        } 
TransactionTemplate继承了DefaultTransactionDefinition,实现了TransactionOperations,InitializingBean接口。先研究InitializingBean接口 
InitializingBean接口为bean提供了初始化方法的方式,它只包括afterPropertiesSet方法,凡是继承该接口的类,在初始化bean的时候会执行该方法。 
测试程序如下: 
import org.springframework.beans.factory.InitializingBean; 
public class TestInitializingBean implements InitializingBean{ 

    @Override 
    public void afterPropertiesSet() throws Exception { 
        System.out.println("ceshi InitializingBean");         
    } 
    public void testInit(){ 
        System.out.println("ceshi init-method");         
    } 

配置文件如下: 


    xmlns:context="http://www.springframework.org/schema/context" 
    xmlns:jdbc="http://www.springframework.org/schema/jdbc" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"         
    xmlns:p="http://www.springframework.org/schema/p" xmlns:aop="http://www.springframework.org/schema/aop" 
    xmlns:tx="http://www.springframework.org/schema/tx" 
    xsi:schemaLocation=" 
    http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd"> 
    

Main主程序如下: 
public class Main { 
    public static void main(String[] args){ 
        ApplicationContext context = new FileSystemXmlApplicationContext("/src/main/java/com/beans.xml"); 
    } 

运行Main程序,打印如下结果: 
ceshi InitializingBean   
这说明在spring初始化bean的时候,如果bean实现了InitializingBean接口,会自动调用afterPropertiesSet方法。 
问题:实现InitializingBean接口与在配置文件中指定init-method有什么不同? 
修改配置文件,加上init-method配置,修改如下: 


    xmlns:context="http://www.springframework.org/schema/context" 
    xmlns:jdbc="http://www.springframework.org/schema/jdbc" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"         
    xmlns:p="http://www.springframework.org/schema/p" xmlns:aop="http://www.springframework.org/schema/aop" 
    xmlns:tx="http://www.springframework.org/schema/tx" 
    xsi:schemaLocation=" 
    http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd"> 
    

在配置文件中加入init-method="testInit"。 
运行Main程序,打印如下结果: 
ceshi InitializingBean 
ceshi init-method 
由结果可看出,在spring初始化bean的时候,如果该bean是实现了InitializingBean接口,并且同时在配置文件中指定了init-method,系统则是先调用afterPropertiesSet方法,然后在调用init-method中指定的方法。 

这方式在spring中是怎么实现的? 
通过查看spring的加载bean的源码类(AbstractAutowireCapableBeanFactory)可看出其中奥妙 
AbstractAutowireCapableBeanFactory类中的invokeInitMethods讲解的非常清楚,源码如下: 
protected void invokeInitMethods(String beanName, final Object bean, RootBeanDefinition mbd) 
            throws Throwable { 
//判断该bean是否实现了实现了InitializingBean接口,如果实现了InitializingBean接口,则只掉调用bean的afterPropertiesSet方法         
        boolean isInitializingBean = (bean instanceof InitializingBean); 
        if (isInitializingBean && (mbd == null || !mbd.isExternallyManagedInitMethod("afterPropertiesSet"))) { 
            if (logger.isDebugEnabled()) { 
                logger.debug("Invoking afterPropertiesSet() on bean with name '" + beanName + "'"); 
            } 

            if (System.getSecurityManager() != null) { 
                try { 
                    AccessController.doPrivileged(new PrivilegedExceptionAction 
                        public Object run() throws Exception { 
   //直接调用afterPropertiesSet 
                            ((InitializingBean) bean).afterPropertiesSet(); 
                            return null; 
                        } 
                    },getAccessControlContext()); 
                } catch (PrivilegedActionException pae) { 
                    throw pae.getException(); 
                } 
            }                 
            else { 
//直接调用afterPropertiesSet 
                ((InitializingBean) bean).afterPropertiesSet(); 
            } 
        } 
        if (mbd != null) { 
            String initMethodName = mbd.getInitMethodName(); 
 //判断是否指定了init-method方法,如果指定了init-method方法,则再调用制定的init-method 
            if (initMethodName != null && !(isInitializingBean && "afterPropertiesSet".equals(initMethodName)) && 
                    !mbd.isExternallyManagedInitMethod(initMethodName)) { 
//进一步查看该方法的源码,可以发现init-method方法中指定的方法是通过反射实现 
                invokeCustomInitMethod(beanName, bean, mbd); 
            } 
        } 
    } 
总结: 
1:spring为bean提供了两种初始化bean的方式,实现InitializingBean接口,实现afterPropertiesSet方法,或者在配置文件中同过init-method指定,两种方式可以同时使用 
2:实现InitializingBean接口是直接调用afterPropertiesSet方法,比通过反射调用init-method指定的方法效率相对来说要高点。但是init-method方式消除了对spring的依赖 
3:如果调用afterPropertiesSet方法时出错,则不调用init-method指定的方法。 
4:TransactionTemplate实现InitializingBean接口,主要是判断transactionManager是否已经初始化,如果没有则抛出异常。源码如下: 
     public void afterPropertiesSet() { 
        if (this.transactionManager == null) { 
            throw new IllegalArgumentException("Property 'transactionManager' is required"); 
        } 
    } 

本文转自二郎三郎博客园博客,原文链接:http://www.cnblogs.com/haore147/p/5213362.html,如需转载请自行联系原作者
相关文章
|
24天前
|
Java API 微服务
【Spring Boot系列】通过OpenAPI规范构建微服务服务接口
【4月更文挑战第5天】通过OpenAPI接口构建Spring Boot服务RestAPI接口
|
2月前
|
前端开发 安全 Java
Spring Boot 三招组合拳,手把手教你打出优雅的后端接口
Spring Boot 三招组合拳,手把手教你打出优雅的后端接口
44 0
|
2月前
|
传感器 Java API
Spring揭秘:Aware接口应用场景及实现原理!
Aware接口赋予了Bean更多自感知的能力,通过实现不同的Aware接口,Bean可以轻松地获取到Spring容器中的其他资源引用,像ApplicationContext、BeanFactory等。 这样不仅增强了Bean的功能,还提高了代码的可维护性和扩展性,从而让Spring的IoC容器变得更加强大和灵活。
132 0
Spring揭秘:Aware接口应用场景及实现原理!
|
6天前
|
Java Spring
spring boot访问接口报500
spring boot访问接口报500
12 2
|
18天前
|
Java 数据库连接 数据库
spring+mybatis_编写一个简单的增删改查接口
spring+mybatis_编写一个简单的增删改查接口
16 2
|
1月前
|
算法 NoSQL Java
限流艺术:Spring Boot接口限流的实用指南
限流艺术:Spring Boot接口限流的实用指南
68 0
限流艺术:Spring Boot接口限流的实用指南
|
2月前
|
数据采集 前端开发 NoSQL
Spring Boot反爬虫,防止接口盗刷
Spring Boot反爬虫,防止接口盗刷
26 1
|
2月前
ssm(Spring+Spring mvc+mybatis)Dao接口——IDeptDao
ssm(Spring+Spring mvc+mybatis)Dao接口——IDeptDao
9 0
|
2月前
|
Java Spring
使用JDBCTemplate实现与Spring结合,方法公用 ——接口(BaseDao)
使用JDBCTemplate实现与Spring结合,方法公用 ——接口(BaseDao)
9 0