4、基于注解管理bean

本文详细介绍了Spring中基于注解管理Bean的过程,包括标记与扫描、组件注解、组件创建和扫描配置。此外,还探讨了基于注解的自动装配,通过`@Autowired`注解实现依赖注入,并解析了其工作流程和注意事项。实验部分展示了不同扫描方式和自动装配的使用场景,强调了代码可读性和结构严谨的重要性。
摘要由CSDN通过智能技术生成


【尚硅谷】SSM框架全套教程-讲师:杨博超

但行好事,莫问前程

4、基于注解管理bean

实验一:标记与扫描

1 注解

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

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

2 扫描

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

然后根据注解进行后续操作。

3 新建Maven Module

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

4 创建Spring配置文件

在这里插入图片描述

5 标识组件的常用注解

@Component:将类标识为普通组件
@Controller:将类标识为控制层组件
@Service:将类标识为业务层组件
@Repository:将类标识为持久层组件

在这里插入图片描述

通过查看源码我们得知,@Controller、@Service、@Repository这三个注解只是在@Component注解的基础上起了三个新的名字。

对于Spring使用IOC容器管理这些组件来说没有区别。

所以@Controller、@Service、@Repository这三个注解只是给开发人员看的,让我们能够便于分辨组件的作用。

注意

虽然它们本质上一样,但是为了代码的可读性,为了程序结构严谨我们肯定不能随便胡乱标记。

6 创建组件

创建控制层组件

@Controller
public class UserController {

}

创建接口UserService

public interface UserService{

}

创建业务层组件UserServiceImpl

@Service
public class UserServiceImpl implements UserService {

}

创建接口UserDao

public interface UserService{

}

创建持久层组件UserDaoImpl

@Repository
public class UserDaoImpl implements UserDao {

}

7 扫描组件

在applicationContext.xml文件中添加context约束

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/context
       http://www.springframework.org/schema/context/spring-context.xsd">

</beans>

情况一:最基本的扫描方式

<context:component-scan base-package="pers.tianyu">
</context:component-scan>

情况二:指定要排除的组件

<context:component-scan base-package="pers.tianyu">
	<!-- context:exclude-filter标签:指定排除规则 -->
	<!--
		type:设置排除或包含的依据
		type="annotation",根据注解排除,expression中设置要排除的注解的全类名
		type="assignable",根据类型排除,expression中设置要排除的类型的全类名
	-->
	<context:exclude-filter type="annotation"
	expression="org.springframework.stereotype.Controller"/>
	<!--<context:exclude-filter type="assignable"
	expression="pers.tianyu.controller.UserController"/>-->
</context:component-scan>

情况三:仅扫描指定组件

<context:component-scan base-package="com.atguigu" use-default-filters="false">
	<!-- context:include-filter标签:指定在原有扫描规则的基础上追加的规则 -->
	<!-- use-default-filters属性:取值false表示关闭默认扫描规则 -->
	<!-- 此时必须设置use-default-filters="false",因为默认规则即扫描指定包下所有类 -->
	<!--
		type:设置排除或包含的依据
		type="annotation",根据注解排除,expression中设置要排除的注解的全类名
		type="assignable",根据类型排除,expression中设置要排除的类型的全类名
	-->
	<context:include-filter type="annotation"
	expression="org.springframework.stereotype.Controller"/>
	<!--<context:include-filter type="assignable"
	expression=""pers.tianyu.controller.UserController"/>-->
</context:component-scan>

8 测试

@Test
public void Test01() {
	// 扫描方式,情况一:最基本的扫描方式
    ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
    UserController bean = context.getBean(UserController.class);
    
}

9 组件所对应的bean的id

在我们使用XML方式管理bean的时候,每个bean都有一个唯一标识,便于在其他地方引用。

现在使用注解后,每个组件仍然应该有一个唯一标识。

默认情况

类名首字母小写就是bean的id。

例如:UserController类对应的bean的id就是userController。

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

@Service(“userService”) //默认为userServiceImpl
public class UserServiceImpl implements UserService {}

实验二:基于注解的自动装配

1 场景模拟

参考基于xml的自动装配

在UserController中声明UserService对象

在UserServiceImpl中声明UserDao对象

2 @Autowired注解

在成员变量上直接标记@Autowired注解即可完成自动装配,不需要提供setXxx()方法。

以后我们在项目中的正式用法就是这样。

Controller层

@Controller
public class UserController {

    @Autowired
    private UserService userService;

    public void saveUser() {
        userService.saveUser();
    }

}

Service层

public interface UserService{
    void saveUser();
}
@Service
public class UserServiceImpl implements UserService {

    @Autowired
    private UserDao userDao;

    @Override
    public void saveUser() {
        userDao.saveUser();
    }
}

Dao层

public interface UserDao {
    void saveUser();
}
@Repository
public class UserDaoImpl implements UserDao {

    @Override
    public void saveUser() {
        System.out.println("保存成功");
    }
}

3 @Autowired注解其他细节

在IDEA中,@Autowired注解写在字段上,会出现警告(不建议使用字段注入)。

在这里插入图片描述

字段注入:

注解作用在字段上,就叫字段注入,除了字段注入,还有setter注入构造方法注入

字段注入问题:

1、可以对final修饰的变量赋值,导致final失效
2、隐藏依赖关系,易导致循环依赖而不自知
3、注入时机后置,风险增大,字段注入是业务执行的时候注入,构造器或者setter是启动服务的时候注入

构造方法注入

@Controller
public class UserController {

    private UserService userService;

    @Autowired
    public UserController(UserService userService) {
        this.userService = userService;
    }

    public void saveUser() {
        userService.saveUser();
    }
}

setter注入

@Controller
public class UserController {

    private UserService userService;

    @Autowired
    public void setUserService(UserService userService) {
        this.userService = userService;
    }

    public void saveUser() {
        userService.saveUser();
    }
}

4 @Autowired工作流程

在这里插入图片描述

  • 首先根据所需要的组件类型到IOC容器中查找
    • 能够找到唯一的bean:直接执行装配
    • 如果完全找不到匹配这个类型的bean:装配失败
    • 和所需类型匹配的bean不止一个
      • 没有@Qualifier注解:根据@Autowired标记位置成员变量的变量名作为bean的id进行匹配
        • 能够找到:执行装配
        • 找不到:装配失败
      • 使用@Qualifier注解:根据@Qualifier注解中指定的名称作为bean的id进行匹配
        • 能够找到:执行装配
        • 找不到:装配失败

字段注入

@Controller
public class UserController {
    @Autowired
    @Qualifier("userServiceImpl")
    private UserService userService;

    public void saveUser() {
        userService.saveUser();
    }
}

setter注入

@Controller
public class UserController {
    private UserService userService;

    @Autowired
    @Qualifier("userServiceImpl")
    public void setUserService(UserService userService) {
        this.userService = userService;
    }

    public void saveUser(){
        userService.saveUser();
    }
}

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

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

但是实际开发时,基本上所有需要装配组件的地方都是必须装配的,用不上这个属性。

@Controller
public class UserController {

    @Autowired(required = false)
    private UserService userService;

    public void saveUser() {
        userService.saveUser();
    }

}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值