day124 - 基于注解管理bean

发布时间 2023-08-04 19:55:51作者: 北海之上

基于注解管理bean

注解

和 XML 配置文件一样,注解本身并不能执行,注解本身仅仅只是做一个标记,具体的功能是框架检测到注解标记的位置,然后针对这个位置按照注解标记的功能来执行具体操作。

本质上:所有一切的操作都是Java代码来完成的,XML和注解只是告诉框架中的Java代码如何执行。

Spring 为了知道程序员在哪些地方标记了什么注解,就需要通过扫描的方式,来进行检测。然后根据注解进行后续操作。

依赖

<!-- 基于Maven依赖传递性,导入spring-context依赖即可导入当前所需所有jar包 -->
<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-context</artifactId>
    <version>5.3.1</version>
</dependency>
<!-- junit测试 -->
<dependency>
    <groupId>junit</groupId>
    <artifactId>junit</artifactId>
    <version>4.12</version>
    <scope>test</scope>
</dependency>

 

常用注解

@Component:将类标识为普通组件

@Controller:将类标识为控制层组件

@Service:将类标识为业务层组件

@Repository:将类标识为持久层组件

扫描组件

  1. 基本扫描

<!-- 扫描组件-->
<context:component-scan base-package="com.gu.spring"></context:component-scan>

 

  1. 排除扫描

<context:component-scan base-package="com.gu.spring" use-default-filters="false">
 <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
<context:exclude-filter type="assignable" expression="com.gu.spring.controller.UserController"/>
    </context:component-scan>

 

  1. 仅扫描指定组件

<context:component-scan base-package="com.gu.spring" use-default-filters="false">
<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>

 

@Test
    public void test(){
        ApplicationContext ioc = new ClassPathXmlApplicationContext("spring-ioc-annotation.xml");
        UserController userController = ioc.getBean(UserController.class);
//        System.out.println(userController);
//        UserServiceImpl userService = ioc.getBean(UserServiceImpl.class);
//        System.out.println(userService);
//        UserDao userDao = ioc.getBean(UserDao.class);
//        System.out.println(userDao);
        userController.saveUser();
    }

 

id对应bean

在我们使用XML方式管理bean的时候,每个bean都有一个唯一标识,便于在其他地方引用。现在使用注解后,每个组件仍然应该有一个唯一标识。

默认情况

类名首字母小写就是bean的id。例如:UserController类对应的bean的id就是userController。

自定义bean的id可通过标识组件的注解的value属性设置自定义的bean的id

@Service("userService")

//默认为userServiceImpl

public class UserServiceImpl implements UserService {}

自动装配

@Autowired注解

在成员变量上直接标记@Autowired注解即可完成自动装配,不需要提供setXxx()方法。以后我们在项目中的正式用法就是这样

@Controller
public class UserController {
    @Autowired
//    @Qualifier("service")
    private UserService userService;
​
    public void saveUser(){
        userService.saveUser();
    }
​
}
​
public interface UserService { void saveUser(); }
​
@Service 
public class UserServiceImpl implements UserService 
{ 
    @Autowired 
    private UserDao userDao; 
    @Override 
    public void saveUser() { userDao.saveUser(); } 
}
​
public interface UserDao { void saveUser(); }
​
@Repository
public class UserDaoImpl implements UserDao {
    @Override
    public void saveUser() {
        System.out.println("okokok");
    }
}

 


细节

@Autowired注解可以标记在构造器和set方法上

@Autowired中有属性required,默认值为true,因此在自动装配无法找到相应的bean时,会装配失败

可以将属性required的值设置为true,则表示能装就装,装不上就不装,此时自动装配的属性为默认值

总结

<!--
    context:exclude-filter:排除扫描
        type用来设置排除扫描的方式
        annotation:根据注解的类型进行排除,expression需要设置排除的注解的全类名
        assignable:根据类的类型进行排除,expression需要设置排除的类的全类名
​
    context:include-filter:包含扫描
    注意:需要在context:component-scan中设置use-default-filters为false
    use-default-filters="true":所设置包下的所有类都需要扫描,此时可以用排除扫描
    use-default-filters="false":所设置包下的所有类都不需要扫描,此时可以用包含扫描
        type用来设置排除扫描的方式
        annotation:根据注解的类型进行包含,expression需要设置包含的注解的全类名
        assignable:根据类的类型进行包含,expression需要设置包含的类的全类名
-->

 

/**
 * @Component:将类标识为普通组件
 * @Controller:将类标识为控制层组件
 * @Service:将类标识为业务层组件
 * @Repository:将类标识为持久层组件
 *
 * 通过注解加扫描配置的bean的id为默认值的类的小驼峰,即类名首字母小写的结果
 * 或者在设置注解的时候在后面自定义其id
 *
 *
 * @Autowired:实现自动装配的注解
 * 1. @Autowired能够表示的位置:
 *      1. 标识在成员变量上,此时不需要设置成员变量的set方法
 *      2. 标识在set方法上
 *      3. 标识为当前成员变量赋值的有参构造上
 * 2. @Autowired注解的原理:
 *      1. 默认通过byType的形式,在ioc容器中通过类型匹配某个bean为属性赋值
 *      2. 若有多个类型匹配的bean,此时会自动转换为byName的方式实现自动装配的效果
 *          即将要赋值的属性名作为bean的id匹配某个bean来为当前属性赋值
 *      3. 若byType和btName的方式都无法实现自动装配,即ioc容器中有多个类型的bean,且这些bean的id和要赋值的属性的属性名都不一致
 *          此时抛出异常NoUniqueBeanDefinitionException
 *      4. 此时使用@Qualifier(""),通过该注解的value值指定某个bean的id,将这个bean为属性赋值。
 *
 *      注意:@Autowired中有属性required,默认值为true,因此在自动装配无法找到相应的bean时,会装配失败
 *          可以将属性required的值设置为true,则表示能装就装,装不上就不装,此时自动装配的属性为默认值
 *          但是实际开发时,基本上所有需要装配组件的地方都是必须装配的,用不上这个属性。
 */

 

over