Spring基于注解式启动原理分析

bean的定义与注册源码相对比较简单,本文主要根据Spring启动流程 介绍spring配置类的注册与定义

一、先看demo

/**
 * spring生命周期配置类<后续会继续分析生命周期>
 **/
@Configuration
@ComponentScan("com.jj.stu.spring.cycle")
@Conditional(MyCondition.class)
public class SpringCycleConfig  implements InitializingBean {
    public SpringCycleConfig() {
        System.out.println("spring配置类实例化");
    }

    @Override
    public void afterPropertiesSet() throws Exception {
        System.out.println("SpringCycleConfig bean初始化接口 InitializingBean" );

    }
}

/**
 * 条件判断类
 **/
public class MyCondition implements Condition {
    /**
     * 根据条件判断上下文context
     */
    public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
        // 1.获取当前环境配置
        Environment environment = context.getEnvironment();
        String osName = environment.getProperty("os.name");
        if (osName.equals("Windows 7")) {
            // 返回true 就是能够创建该bean
            return true;
        }
        return false;
    }
}


//启动类
public class SpringCycleMain {
    public static void main(String[] args) {
        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(SpringCycleConfig.class);

        context.close();
    }
}

二、AnnotationConfigApplicationContext源码分析

直接看代码,以构造函数为切入点

//1.调用无参构造器 2.注册配置类 3.调用refresh方法
 public AnnotationConfigApplicationContext(Class<?>... componentClasses) {
        this();
        this.register(componentClasses);    
        this.refresh();
    }

关于配置类bean的定义与注册看前两个方法就好
顺便说一下refresh方法,对bean生命周期进行管理主要是通过这个方法,后续会讲

1.this()方法源码


public AnnotationConfigApplicationContext() {
		//这个是AnnotatedBeanDefinitionReader,beandefinition读取器 重点分析
        this.reader = new AnnotatedBeanDefinitionReader(this);
        this.scanner = new ClassPathBeanDefinitionScanner(this);
    }

//重点有两条
public AnnotatedBeanDefinitionReader(BeanDefinitionRegistry registry, Environment environment) {
        this.beanNameGenerator = AnnotationBeanNameGenerator.INSTANCE;
        this.scopeMetadataResolver = new AnnotationScopeMetadataResolver();
        Assert.notNull(registry, "BeanDefinitionRegistry must not be null");
        Assert.notNull(environment, "Environment must not be null");
        this.registry = registry;
        //1.条件评估器  在注册时进行条件判断,后文解析doRegisterBean()时再分析源码
        this.conditionEvaluator = new ConditionEvaluator(registry, environment, (ResourceLoader)null);
        2.注册spring内置的后置处理器,生命周期管理基本全靠这些处理器
        AnnotationConfigUtils.registerAnnotationConfigProcessors(this.registry);
    }

AnnotationConfigUtils.registerAnnotationConfigProcessors(this.registry)源码

 public static Se
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring框架中提供了事务管理功能,使得数据库的操作可以具备原子性、一致性、隔离性和持久性四个特性。但是,在应用程序运行过程中,经常会遇到Spring事务失效的情况,因为Spring框架对事务的管理是基于Spring容器的。当容器启动的时候,Spring会使用AOP技术根据注解或配置文件中的信息,在需要进行事务管理的方法上动态地生成一个代理类,这个类负责执行事务控制。如果Spring容器没有正确启动,或者容器没有正确加载,那么就有可能导致事务失败。 以下是常见的导致Spring事务失效的原因: 1、事务管理声明不正确导致事务无法被启动Spring事务管理中,声明事务管理通常是通过注解或XML配置文件来完成的。如果这些配置信息不正确,或者没有正确应用在目标方法上,就会导致事务无法被启动。例如,在注解配置方中,如果目标方法没有被@Transactional注解修饰,那么就无法启动事务。 2、外部方法调用导致事务被绕过 在某些情况下,Spring事务管理可能会被绕过。比如,外部调用一个带有事务控制的目标方法时,如果调用方自身已经被标注为@Transactional,那么目标方法就不会被拦截,事务也就不会被启动。 3、运行时异常抛出导致事务失效 Spring事务管理默认情况下只会在运行时异常上回滚事务。如果目标方法中抛出的异常不是运行时异常,或者在方法中捕获并处理了异常,那么就不会回滚事务。 4、跨事务的方法引起事务失效 如果在一个方法的事务中调用了另一个带有事务控制的方法,那么默认情况下,Spring会使用同一个事务进行管理。如果在这个过程中,目标方法出现了异常而导致事务回滚,则整个事务都会回滚,包括外部方法的事务。这也就是所谓的悬挂事务,会导致事务失效。 总之,Spring事务管理是一项非常重要的功能,但是在使用过程中也需要仔细分析和处理可能出现的问题,以保证事务操作的正确性和准确性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值