Spring复习——day06_Spring 基于注解管理Bean

发布于:2022-12-26 ⋅ 阅读:(231) ⋅ 点赞:(0)

 

1.标记与扫描

(1)注解

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

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

(2)扫描

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

(3)新建Maven工程

<dependencies>
    <!-- 基于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>
</dependencies>

(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实现UserService接口

@Service
public class UserServiceImpl implements UserService {

}

创建接口UserDao

public interface UserDao {

}

创建持久层组件UserDaoImpl实现UserDao接口

@Repository
public class UserDaoImpl implements UserDao {

}

(7)扫描组件

情况一:最基本的扫描方式(扫描com.itwpf下的所有包)

<context:component-scan base-package="com.itwpf">
</context:component-scan>

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

<context:component-scan base-package="com.itwpf">
    <!-- 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="com.atguigu.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="com.atguigu.controller.UserController"/>-->
</context:component-scan>

(8)测试

@Test
public void testAutowireByAnnotation(){
    ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
    UserController userController = ac.getBean(UserController.class);
    System.out.println(userController);
    UserService userService = ac.getBean(UserService.class);
    System.out.println(userService);
    UserDao userDao = ac.getBean(UserDao.class);
    System.out.println(userDao);
}

(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 {}

2.基于注解的自动装配

(1)场景模拟

参考基于xml的自动装配

在UserController中声明UserService对象

在UserServiceImpl中声明UserDao对象

(2)@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
//    @Qualifier("dao")
    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("保存信息");
    }
}

(3)@Autowired注解的其他细节

标识位置

(1)标识在成员变量上,此时不需要设置成员变量的set方法
(2)标识在set方法上
(3)标识在为当前成员变量赋值的有参构造上

注解的原理

(1)默认通过byType的方式,在Ioc容器中通过类型匹配某个bean为属性赋值
(2)若有多个类型匹配的bean,此时会自动转换为byName的方式实现自动装配的效果,即将要赋值的属性的属性名作为bean的id匹配某个bean为属性赋值
(3)若byType和byName的方式都无法实现自动装配,即ioc容器中有多个类型匹配的bean且这些bean的id和要赋值的属性的属性名都不一致,此时抛出异常NoUniqueBeanDefinitionException此时可以在要赋值的属性上,添加一个注解@Qualifier通过该注解的value属性值,指定某个bean的id,将这个bean为属性赋值。

注意:

若ioc容器中没有任何一个类型匹配的bean此时会抛出异常NoSuchBeanDefinitionException
(出现原因:(1)类上没有加注解@Component或者@Service或者@Controller@或者@Repository
          (2)扫描问题,我们在扫描时,没有扫描到当前的类)
在@Autowire注解中有个属性required,默认值为true,要求必须完成自动装配可以将required设置为false,此时能装配则装配,无法装配则使用属性的默认值

(4)@Autowired工作流程

首先根据所需要的组件类型到IOC容器中查找

        能够找到唯一的bean:直接执行装配

        如果完全找不到匹配这个类型的bean:装配失败

        和所需类型匹配的bean不止一个

              没有@Qualifier注解:根据@Autowired标记位置成员变量的变量名作为bean的id进行匹配

                        能够找到:执行装配

                        找不到:装配失败

               使用@Qualifier注解:根据@Qualifier注解中指定的名称作为bean的id进行匹配

                        能够找到:执行装配

                        找不到:装配失败


网站公告

今日签到

点亮在社区的每一天
去签到