Spring学习03-[Spring容器核心技术IOC学习进阶]

IOC学习进阶

@Order

可以改变自动注入的顺序

比如有个animal的接口,里面两个实现类 Cat和Dog。这两个都注册为bean。此时注入的时候,就可以用list来进行接收,表示两个都进行注入。
如果还用animal来进行接收,那就会报错,因为@Autowired根据类型找到两个,然后根据名称找,此时找不到就报错

  • 类型接口
  • animal
java 复制代码
public interface Animal {
    void eat();
}
  • 实现类
  • Cat
java 复制代码
@Component
public class Cat implements Animal{
    @Override
    public void eat() {
        System.out.println("猫吃老鼠");
    }
}
  • Dog
java 复制代码
@Component
public class Dog implements Animal{
    @Override
    public void eat() {
        System.out.println("狗吃肉");
    }
}
  • 使用
java 复制代码
  @Autowired
    private List<Animal> animal;

    @Test
    public void testOrder(){
        System.out.println(animal);
    }


可以看到先输出Cat,再输出Dog(先注入了Cat,再注入了Dog)

使用@Order改变注入顺序

此时我们可以用@Order注解自定义注入顺序

用法:@Order(数值)---数值小的先注入
让Dog先注入,Cat后注入

java 复制代码
@Component
@Order(1)
public class Dog implements Animal{
    @Override
    public void eat() {
        System.out.println("狗吃肉");
    }
}
java 复制代码
@Component
@Order(2)
public class Cat implements Animal{
    @Override
    public void eat() {
        System.out.println("猫吃老鼠");
    }
}
  • 测试
    可以和上面未加@Order进行对比,加了@Order后Dog先进行了注入

实现Ordered接口,重写getOrder方法来改变自动注入顺序

这种方式了解即可

java 复制代码
@Component
@Order(1)
public class Dog implements Animal, Ordered {
    @Override
    public void eat() {
        System.out.println("狗吃肉");
    }

    @Override
    public int getOrder() {
        return 1;
    }
}

@DependsOn

改变bean的创建顺序

比如下面的 B负责数据库连接,A负责数据库查询,肯定是B先创建bean,此时就可以用@DependsOn进行调整两个bean的创建顺序

使用

  • 不加 @DependsOn注解
  • A:数据库查询
java 复制代码
@Component
public class A {
    public A() {
        System.out.println("数据库查询");
    }
}
  • B:数据库连接
java 复制代码
@Component
public class B {
    public B() {
        System.out.println("数据库连接");
    }
}


此时可以看到在不加@DependsOn注解的时候,数据库查询bean先创建,明显不对

  • 加 @DependsOn注解
    因为数据库查询依赖数据库连接所以在数据库查询上加@DependsOn注解
    表示:A依赖B,在创建A的时候先创建B
java 复制代码
@Component
@DependsOn("b")
public class A {
    public A() {
        System.out.println("数据库查询");
    }
}
  • 测试
    可以看到,数据库连接先创建,数据库查询后创建

@Lazy

懒加载bean,默认是true:表示懒加载,可以设置成false,表示非懒加载

全局设置-设置所有bean启动时候懒加载

通过配置文件进行设置

@Scope

默认是单例的

bean是单例的,会不会有线程安全问题


正常情况下应该是1000块钱

模拟两个线程取钱和存钱

  • AccountService
java 复制代码
@Component
public class AccountService {

    public int balance=1000;

    // 存钱
    public void add(int money) throws InterruptedException {
        int newMoney = balance + money;
        TimeUnit.SECONDS.sleep(2);
        balance=newMoney;
    }

    //取钱
    public void sub(int money) throws InterruptedException {
        int newMoney = balance - money;
        TimeUnit.SECONDS.sleep(1);
        balance=newMoney;
    }
}
  • 测试类
java 复制代码
@SpringBootTest(classes = StudyApplicaiton.class)
public class AccountServiceTeset {

    @Test
    public void test(@Autowired AccountService accountService, @Autowired AccountService accountService1) throws InterruptedException {
        new Thread(()->{
            try {
                accountService.add(500);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        }).start();


        new Thread(()->{
            try {
                accountService1.sub(500);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        }).start();

        TimeUnit.SECONDS.sleep(3);

        System.out.println(accountService.balance);
        System.out.println(accountService1.balance);
    }
}

可以看到出现问题了,正确应该是1000

代码解释:存钱线程先执行,存钱后睡两秒钟。然后取钱线程执行,执行完之后banlance为500,此时存钱线程睡眠结束,把banlance更新为1500

这就是典型的多线程多任务的多线程案例。存钱任务对应一个线程,取钱任务对应一个任务。

火车售票是典型的多线程单任务多线程案例。多个线程对应一个售票任务。

解决方式

修改bean为多例模式,或者用threadlocal在每个线程复制一份工作变量

@Conditional:条件注解

用于动态决定某个bean是否生效

使用:@Conditional(xxx.class)条件注解,里面带一个实现了Condition接口的类。这个实现了Condition接口的类会重写matches方法,返回值是boolean,如果返回true,满足条件bean可以注册,否则不满足条件无法注册bean

使用

  • 实现了condition的类
java 复制代码
public class UserCondition implements Condition {

    @Override
    public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
        return true;
    }
}
  • bean
java 复制代码
@Conditional(UserCondition.class)
@Component
public class User {

}
  • 测试

bean的生命周期回调方法

什么是bean的生命周期



初始化

bean的初始化可以实现InitializingBean方法也可以在方法上加@PostConstruct注解

循环依赖



需要注意的是原生Spring支持循环依赖,但是SpringBoot不支持循环依赖。如果要让SpringBoot也支持循环依赖,需要在配置文件中开启循环依赖

yaml 复制代码
spring:
  main:
    allow-circular-references: true

解决循环依赖

常用的有spring的三级缓存,或者对某个依赖bean加@lazy注解

相关推荐
奶香臭豆腐8 分钟前
C++ —— 模板类具体化
开发语言·c++·学习
hanbarger18 分钟前
mybatis框架——缓存,分页
java·spring·mybatis
cdut_suye26 分钟前
Linux工具使用指南:从apt管理、gcc编译到makefile构建与gdb调试
java·linux·运维·服务器·c++·人工智能·python
苹果醋338 分钟前
2020重新出发,MySql基础,MySql表数据操作
java·运维·spring boot·mysql·nginx
小蜗牛慢慢爬行40 分钟前
如何在 Spring Boot 微服务中设置和管理多个数据库
java·数据库·spring boot·后端·微服务·架构·hibernate
azhou的代码园43 分钟前
基于JAVA+SpringBoot+Vue的制造装备物联及生产管理ERP系统
java·spring boot·制造
波音彬要多做1 小时前
41 stack类与queue类
开发语言·数据结构·c++·学习·算法
wm10431 小时前
java web springboot
java·spring boot·后端
smile-yan1 小时前
Provides transitive vulnerable dependency maven 提示依赖存在漏洞问题的解决方法
java·maven
老马啸西风1 小时前
NLP 中文拼写检测纠正论文-01-介绍了SIGHAN 2015 包括任务描述,数据准备, 绩效指标和评估结果
java