基于注解管理bean

1.标记和扫描

1.注解

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

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

2.扫描

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

3.新建Maven Module

<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.创建配置文件

在resources文件夹下面创建applicationContext.xml

5.表示组件的常用注解

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

@Controller、@Service、@Repository这三个注解只是在@Component基础上取得三个新名字。对于Spring使用IOC容器股那里这些组件来说没有区别,这三个组件只是给开发人员看的,让我们能够便于分辨组件的作用。

6.创建组件

创建控制层组件:

@Controller
public class UserController {
}

创建接口UserService

public interface UserService {
}

创建业务层组件UserServiceImpl

@Service
public class UserServiceImpl implements UserService {

}

创建接口UserDao

public class UserDao {
}

创建持久层组件UserDaoImpl

@Repository
public class UserDaoImpl {
}

7.扫描组件

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

     <!--  扫描组件  -->
     <context:component-scan base-package="com.atguigu.spring"></context:component-scan>
    
  • 情况二:指定要排除的组件

    <context:component-scan base-package="com.atguigu">
    <!-- 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);
}

0.组件所对应的bean的id

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

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

自定义的bean的id:可通过标识组件的注解的value属性设置自定义的bean的id。例如:@Service(“userService”)


2.基于注解的自动装配

1.场景模拟

参考基于XML的自动装配

  • 在UserController中声明UserService对象
  • 在UserServiceImpl中声明UserDao对象

2.@Autowired注解

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

@Controller
public class UserController {
    @Autowired
    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("保存成功!");
    }
}

3.@Autowired注解其他细节

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

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

    public void saveUser(){
        userService.saveUser();
    }
}
@Controller
public class UserController {
    
    private UserService userService;

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

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

4.@Autowired注解原理

  • 默认通过byType的方式,在IOC容器中通过类型匹配某个bean为属性赋值。
  • 若有多个类型匹配的bean,此时会自动转换为byName的方式实现自动装配的效果。即将要复制的属性的属性名作为bean的id匹配某个bean为属性赋值。
  • 若byType和byName的方式都无法实现自动装配,即在IOC容器中有多个类型匹配的bean,且这些bean的id和要复制的属性的属性名都不一致,此时抛出异常:NoUniqueBeanDefinitionException
  • 此时可以在要赋值的属性上,添加注解**@Qualifier**,通过该注解的value属性值,指定某个bean的id,将这个bean为属性赋值
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值