• Spring修炼之旅(3)自动装配与注解开发


    一、自动装配说明

    1.1概述

    • 自动装配是使用spring满足bean依赖的一种方法

    • spring会在应用上下文中为某个bean寻找其依赖的bean。

    1.2装配机制

    Spring中bean有三种装配机制,分别是:

    1. xml中显式配置;

    2. 在java中显式配置;

    3. 隐式的bean发现机制和自动装配。

    这里我们主要讲第三种:自动化的装配bean。

    Spring的自动装配需要从两个角度来实现,或者说是两个操作:

    1. 组件扫描(component scanning):spring会自动发现应用上下文中所创建的bean;

    2. 自动装配(autowiring):spring自动满足bean之间的依赖,也就是我们说的IoC/DI;

    组件扫描和自动装配组合发挥巨大威力,使得显示的配置降低到最少。

    1.3测试环境搭建

    新建实体类

    1. package com.yanyu.pojo;
    2. public class Cat {
    3. public void shout() {
    4. System.out.println("miao~");
    5. }
    6. }
    1. package com.yanyu.pojo;
    2. public class Dog {
    3. public void shout() {
    4. System.out.println("wang~");
    5. }
    6. }
    1. package com.yanyu.pojo;
    2. import lombok.Data;
    3. @Data
    4. public class User {
    5. private Cat cat;
    6. private Dog dog;
    7. private String str;
    8. }

    编写Spring配置文件

    1. "1.0" encoding="UTF-8"?>
    2. <beans xmlns="http://www.springframework.org/schema/beans"
    3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    4. xsi:schemaLocation="http://www.springframework.org/schema/beans
    5. http://www.springframework.org/schema/beans/spring-beans.xsd">
    6. <bean id="dog" class="com.yanyu.pojo.Dog"/>
    7. <bean id="cat" class="com.yanyu.pojo.Cat"/>
    8. <bean id="user" class="com.yanyu.pojo.User">
    9. <property name="cat" ref="cat"/>
    10. <property name="dog" ref="dog"/>
    11. <property name="str" value="yanyu"/>
    12. bean>
    13. beans>

    测试

    1. import com.yanyu.pojo.User;
    2. import org.junit.Test;
    3. import org.springframework.context.ApplicationContext;
    4. import org.springframework.context.support.ClassPathXmlApplicationContext;
    5. public class MyTest {
    6. @Test
    7. public void testMethodAutowire() {
    8. ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
    9. User user = (User) context.getBean("user");
    10. user.getCat().shout();
    11. user.getDog().shout();
    12. }
    13. }

    二、自动装配

    2.1byName

    autowire byName (按名称自动装配)

    由于在手动配置xml过程中,常常发生字母缺漏和大小写等错误,而无法对其进行检查,使得开发效率降低。

    采用自动装配将避免这些错误,并且使配置简单化。

    1. 修改bean配置,增加一个属性  autowire="byName"
      1. <bean id="user" class="com.yanyu.pojo.User" autowire="byName">
      2. <property name="str" value="yabyu"/>
      3. bean>
    2. 再次测试,结果依旧成功输出!
    3. 我们将 cat 的bean id修改为 catXXX
    4. 再次测试, 执行时报空指针java.lang.NullPointerException。因为按byName规则找不对应set方法,真正的setCat就没执行,对象就没有初始化,所以调用时就会报空指针错误。

    当一个bean节点带有 autowire byName的属性时。

    1. 将查找其类中所有的set方法名,例如setCat,获得将set去掉并且首字母小写的字符串,即cat。

    2. 去spring容器中寻找是否有此字符串名称id的对象。

    3. 如果有,就取出注入;如果没有,就报空指针异常。

    2.2byType

    autowire byType (按类型自动装配)

    使用autowire byType首先需要保证:同一类型的对象,在spring容器中唯一。如果不唯一,会报不唯一的异常。

    NoUniqueBeanDefinitionException

    测试:

    1、将user的bean配置修改一下 : autowire="byType"

    2、测试,正常输出

    3、在注册一个cat 的bean对象!

    1. <bean id="dog" class="com.yanyu.pojo.Dog"/>
    2. <bean id="cat" class="com.yanyu.pojo.Cat"/>
    3. <bean id="cat2" class="com.yanyu.pojo.Cat"/>
    4. <bean id="user" class="com.yanyu.pojo.User" autowire="byType">
    5. <property name="str" value="qinjiang"/>
    6. bean>

    4、测试,报错:NoUniqueBeanDefinitionException

    5、删掉cat2,将cat的bean名称改掉!测试!因为是按类型装配,所以并不会报异常,也不影响最后的结果。甚至将id属性去掉,也不影响结果。

    这就是按照类型自动装配!

    2.3使用注解

    配置文件

    1. "1.0" encoding="UTF-8"?>
    2. <beans xmlns="http://www.springframework.org/schema/beans"
    3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    4. xmlns:context="http://www.springframework.org/schema/context"
    5. xmlns:aop="http://www.springframework.org/schema/aop"
    6. xsi:schemaLocation="http://www.springframework.org/schema/beans
    7. https://www.springframework.org/schema/beans/spring-beans.xsd
    8. http://www.springframework.org/schema/context
    9. https://www.springframework.org/schema/context/spring-context.xsd
    10. http://www.springframework.org/schema/aop
    11. https://www.springframework.org/schema/aop/spring-aop.xsd">
    12. <bean id="dog" class="com.yanyu.pojo.Dog"/>
    13. <bean id="cat" class="com.yanyu.pojo.Cat"/>
    14. <bean id="user" class="com.yanyu.pojo.User" autowire="byType">
    15. <property name="str" value="yabyu"/>
    16. bean>
    17. beans>

    2、开启属性注解支持!

    <context:annotation-config/>

    @Autowired

    • @Autowired是按类型自动转配的,不支持id匹配。

    • 需要导入 spring-aop的包!

    直接在属性上使用也可以在set方式上使用!
    使用Autowired我们可以不用编写Set方法了,前提是你这个自动装配的属性在I0C (Spring) 容器中存在,且符合名字byname!

    测试:

    1、将User类中的set方法去掉,使用@Autowired注解

    1. package com.yanyu.pojo;
    2. import lombok.Data;
    3. import org.springframework.beans.factory.annotation.Autowired;
    4. import org.springframework.stereotype.Component;
    5. @Data
    6. @Component
    7. public class User {
    8. @Autowired
    9. private Cat cat;
    10. @Autowired
    11. private Dog dog;
    12. private String str;
    13. }

    2、此时配置文件内容

    1. <context:annotation-config/>
    2. <bean id="dog" class="com.yanyu.pojo.Dog"/>
    3. <bean id="cat" class="com.yanyu.pojo.Cat"/>
    4. <bean id="user" class="com.yanyu.pojo.User"/>

    3、测试,成功输出结果!

    @Autowired(required=false)  说明:false,对象可以为null;true,对象必须存对象,不能为null。

    1. //如果允许对象为null,设置required = false,默认为true
    2. @Autowired(required = false)
    3. private Cat cat;

    @Qualifier

    • @Autowired是根据类型自动装配的,加上@Qualifier则可以根据byName的方式自动装配

    • @Qualifier不能单独使用

    测试实验步骤:

    1、配置文件修改内容,保证类型存在对象。且名字不为类的默认名字!

    1. <bean id="dog1" class="com.yanyu.pojo.Dog"/>
    2. <bean id="dog2" class="com.yanyu.pojo.Dog"/>
    3. <bean id="cat1" class="com.yanyu.pojo.Cat"/>
    4. <bean id="cat2" class="com.yanyu.pojo.Cat"/>

    2、没有加Qualifier测试,直接报错

    3、在属性上添加Qualifier注解

    1. @Autowired
    2. @Qualifier(value = "cat2")
    3. private Cat cat;
    4. @Autowired
    5. @Qualifier(value = "dog2")
    6. private Dog dog;

    测试,成功输出!

    @Resource(jdk8取消注解)

    • @Resource如有指定的name属性,先按该属性进行byName方式查找装配;

    • 其次再进行默认的byName方式进行装配;

    • 如果以上都不成功,则按byType的方式自动装配。

    • 都不成功,则报异常。

    三、Spring注解开发

    3.1环境配置

    1. "1.0" encoding="UTF-8"?>
    2. <beans xmlns="http://www.springframework.org/schema/beans"
    3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    4. xmlns:context="http://www.springframework.org/schema/context"
    5. xsi:schemaLocation="http://www.springframework.org/schema/beans
    6. http://www.springframework.org/schema/beans/spring-beans.xsd
    7. http://www.springframework.org/schema/context
    8. http://www.springframework.org/schema/context/spring-context.xsd">
    9. <context:component-scan base-package="com.yanyu.pojo"/>
    10. <context:annotation-config/>
    11. beans>

    3.2Bean的实现

    我们之前都是使用 bean 的标签进行bean注入,但是实际开发中,我们一般都会使用注解!

    1、配置扫描哪些包下的注解

    1. <context:component-scan base-package="com.yanyu.pojo"/>

    2、在指定包下编写类,增加注解

    1. @Component("user")
    2. // 相当于配置文件中 <bean id="user" class="当前注解的类"/>
    3. public class User {
    4.    public String name = "烟雨";
    5. }

    3、测试

    1. @Test
    2. public void test(){
    3.    ApplicationContext applicationContext =
    4.        new ClassPathXmlApplicationContext("beans.xml");
    5.    User user = (User) applicationContext.getBean("user");
    6.    System.out.println(user.name);
    7. }

    3.3属性注入

    使用注解注入属性

    1、可以不用提供set方法,直接在直接名上添加@value("值")

    1. @Component("user")
    2. // 相当于配置文件中
    3. public class User {
    4.    @Value("烟雨")
    5.    // 相当于配置文件中
    6.    public String name;
    7. }

    2、如果提供了set方法,在set方法上添加@value("值");

    1. @Component("user")
    2. public class User {
    3. public String name;
    4. @Value("烟雨")
    5. public void setName(String name) {
    6. this.name = name;
    7. }
    8. }

    3.4衍生注解

    我们这些注解,就是替代了在配置文件当中配置步骤而已!更加的方便快捷!

    @Component三个衍生注解

    为了更好的进行分层,Spring可以使用其它三个注解,功能一样,目前使用哪一个功能都一样。

    • @Controller:web层

    • @Service:service层

    • @Repository:dao层

    写上这些注解,就相当于将这个类交给Spring管理装配了!

    3.5自动装配注解

    在Bean的自动装配已经讲过了,可以回顾!

    作用域

    @scope

    • singleton:默认的,Spring会采用单例模式创建这个对象。关闭工厂 ,所有的对象都会销毁。

    • prototype:多例模式。关闭工厂 ,所有的对象不会销毁。内部的垃圾回收机制会回收

    1. @Controller("user")
    2. @Scope("prototype")
    3. public class User {
    4.    @Value("yanyu")
    5.    public String name;
    6. }

    3.6小结

    XML与注解比较

    • XML可以适用任何场景 ,结构清晰,维护方便

    • 注解不是自己提供的类使用不了,开发简单方便

    xml与注解整合开发 :推荐最佳实践

    • xml管理Bean

    • 注解完成属性注入

    • 使用过程中, 可以不用扫描,扫描是为了类上的注解

      

    作用:

    • 进行注解驱动注册,从而使注解生效

    • 用于激活那些已经在spring容器里注册过的bean上面的注解,也就是显示的向Spring注册

    • 如果不扫描包,就需要手动配置bean

    • 如果不加注解驱动,则注入的值为null!

     3.7基于Java类进行配置

    JavaConfig 原来是 Spring 的一个子项目,它通过 Java 类的方式提供 Bean 的定义信息,在 Spring4 的版本, JavaConfig 已正式成为 Spring4 的核心功能 。

    测试:

    1、编写一个实体类,Dog

    1. package com.yanyu.pojo;
    2. import org.springframework.stereotype.Component;
    3. @Component //将这个类标注为Spring的一个组件,放到容器中!
    4. public class Dog {
    5. public String name = "dog";
    6. }

    2、新建一个config配置包,编写一个MyConfig配置类

    1. package com.yanyu.config;
    2. import com.yanyu.pojo.Dog;
    3. import org.springframework.context.annotation.Bean;
    4. import org.springframework.context.annotation.Configuration;
    5. @Configuration //代表这是一个配置类
    6. public class MyConfig {
    7. @Bean //通过方法注册一个bean,这里的返回值就Bean的类型,方法名就是bean的id!
    8. public Dog dog(){
    9. return new Dog();
    10. }
    11. }

    3、测试

    1. import com.yanyu.config.MyConfig;
    2. import com.yanyu.pojo.Dog;
    3. import org.junit.Test;
    4. import org.springframework.context.ApplicationContext;
    5. import org.springframework.context.annotation.AnnotationConfigApplicationContext;
    6. import org.springframework.context.support.ClassPathXmlApplicationContext;
    7. public class MyTest {
    8. @Test
    9. public void test2(){
    10. ApplicationContext applicationContext =
    11. new AnnotationConfigApplicationContext(MyConfig.class);
    12. Dog dog = (Dog) applicationContext.getBean("dog");
    13. System.out.println(dog.name);
    14. }
    15. }

    导入其他配置如何做呢?

    1、我们再编写一个配置类!

    1. @Configuration  //代表这是一个配置类
    2. public class MyConfig2 {
    3. }

    2、在之前的配置类中我们来选择导入这个配置类

    1. @Configuration
    2. @Import(MyConfig2.class)  //导入合并其他配置类,类似于配置文件中的 inculde 标签
    3. public class MyConfig {
    4.    @Bean
    5.    public Dog dog(){
    6.        return new Dog();
    7.   }
    8. }

    关于这种Java类的配置方式,我们在之后的SpringBoot 和 SpringCloud中还会大量看到,我们需要知道这些注解的作用即可!

  • 相关阅读:
    腾讯mini项目-【指标监控服务重构】2023-08-11
    03--nginx架构实战
    应用系统设计:在线教育平台,B2C平台设计
    17、学习MySQL 事务
    【示波器专题】示波器最基本的电压测量和时间测量
    小程序“超级入口”出现了
    IPD笔记
    ceph性能测试
    【生成对抗网络 论文泛读】……StarganV1 & StarganV2……
    脉冲神经网络原理及应用,脉冲神经网络编码方式
  • 原文地址:https://blog.csdn.net/qq_62377885/article/details/133432967