回顾一下基于xml配置的Spring对Bean的管理 ,对Bean的完整管理如下所示:
<bean id="" class="" init-method="" destroy-method="" scope="">
<property name="" value=""/>
<property name="" ref=""/>
</bean>
分析可以发现:我们对Bean的管理就四个方面,分别是:
其实对于注解来说,也是包括了这四个方面,换句话说,使用注解的方式管理Bean和使用xml的方式管理Bean作用是完全一样的,区别仅仅在于配置的形式不同而已。
@Component
public class DogService {
}
<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
https://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd">
<!-- 告诉spring在创建容器时要扫描的包 -->
<context:component-scan base-package="cn.bdqn"/>
</beans>
@Test
public void testDogServiceImpl() throws Exception{
// 1、读取主配置文件信息,获取核心容器对象
ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
DogServiceImpl dogService = (DogServiceImpl) ac.getBean("dogServiceImpl");
System.out.println(dogService);
}
作用
? 用于把对当前修饰的类创建出来,并存放到Spring容器中。
属性
? a. 用该注解所创建的对象默认的id名称是当前类名,且首字母改小写
? b. 可以通过value属性手动的指定bean的id。
? 一般用在表现层,例如SpringMVC、Struts2
? 一般用在业务层,例如Service层
? 一般用在持久层7.2.5、总结
作用:
? 自动按照类型注入。
// 用户service接口
public interface UserService {
public void printUserDao();
}
// 用户service接口实现,bean的名称改为:userService
@Service("userService")
public class UserServiceImpl implements UserService {
@Autowired
private UserDao userDao;
// 打印UserDao,看是否可以将值打印出来,如果打印出来说明值真的注入成功了
public void printUserDao(){
System.out.println(userDao);
}
}
// 用户UserDao接口
public interface UserDao {
}
// 用户UserDao接口实现,bean的名称改为:userDao01
@Repository("userDao01")
public class UserDaoImpl01 implements UserDao {
}
<beans>
<!-- 告诉spring在创建容器时要扫描的包 -->
<context:component-scan base-package="cn.bdqn"/>
</beans>
@Test
public void testUserServiceImpl() throws Exception{
// 1、读取主配置文件信息,获取核心容器对象
ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
UserService userService = (UserService) ac.getBean("userService");
userService.printUserDao(); // cn.bdqn.dao.impl.UserDaoImpl@7a52f2a2
}
假设系统中存在两个UserDao的实现,现在再添加一个。现在再添加测试:
@Repository("userDao02")
public class UserDaoImpl02 implements UserDao {
}
再次运行程序,会发现,程序报错:
No qualifying bean of type 'cn.bdqn.dao.UserDao' available: expected single matching bean but found 2: userDao01,userDao02
翻译:没有找到一个可用的UserDao,期望能够匹配一个,但是发现了2个。换句话说,由于是根据类型匹配的,而userDao01,userDao02都是符合注入的类型的,不知道要用哪个了.
该如何解决呢?既然Spring不知道具体要用哪个了,那我们由开发者来去指定其中的一个告诉Spring你就用这个注入就可以了,那么实现方式就是:通过名称告诉即可
解决方案:
@Service("userService")
public class UserServiceImpl implements UserService {
// 将变量的名称修改为userDao01,那么依赖注入的时候就使用UserDaoImpl01
@Autowired
private UserDao userDao01;
}
再次测试,程序正常执行。
? 在7.3.1章节使用Autowired注解的时候,会存在一个问题,就是如果系统中存在多个类型的Bean都匹配的时候,就会找不到到底要使用哪个Bean对象了,会报错,我们采取的解决办法是:修改变量名即可解决 , 但是这种做法实际上是挺菜的,我现在就想使用userDao这个变量名,那么能否有一种更好的解决办法呢?答案是肯定的,即使用Qualifier注解。
? Bean的定义,仍然采用7.3.1章节所定义好的Bean,唯一的区别是UserServiceImpl这个Bean,修改如下:
public class UserServiceImpl implements UserService {
@Autowired
@Qualifier(value = "userDao01") // 通过此注解中的value属性明确指定要用哪个name的bean
private UserDao userDao;
}
? Bean的定义,仍然采用7.3.1章节所定义好的Bean,唯一的区别是UserServiceImpl这个Bean,修改如下:
@Service("userService")
public class UserServiceImpl implements UserService {
@Resource(name = "userDao01")
private UserDao userDao;
}
注意:
使用Autowired、Qualifier以及Resource这三个注解都只能注入其他bean类型的数据,对于基本数据类型和String类型是无法通过使用该3个注解实现。同时,对于集合数据类型的注入只能通过XML来实现。
作用:
? 用于注入基本类型和String类型的数据。
@Component("jdbcUtils")
public class JdbcUtils {
@Value("com.mysql.jdbc.Driver")
private String driverClass;
}
@Test
public void testJdbcUtils() throws Exception{
// 1、读取主配置文件信息,获取核心容器对象
ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
JdbcUtils utils = (JdbcUtils) ac.getBean("jdbcUtils");
System.out.println(utils); // com.mysql.jdbc.Driver
}
如果Value注解用于案例1,那这样太菜了,我们要为driverClass这个变量赋值,那岂不是直接赋值得了,还需要搞一个Value直接赋值吗?显然是没有必要的,所以一般来说,Value注解常用于对配置文件内容的读取。
driverClass=com.mysql.jdbc.Driver
port=3306
<beans>
<!-- 告诉spring在创建容器时要扫描的包 -->
<context:component-scan base-package="cn.bdqn"/>
<!-- 将properties文件引入到Spring框架中-->
<context:property-placeholder location="classpath:db.properties"/>
</beans>
@Component("jdbcUtils")
public class JdbcUtils {
@Value("${driverClass}")
private String driverClass;
@Value("${port}")
private Integer port;
}
@Test
public void testJdbcUtils() throws Exception{
// 1、读取主配置文件信息,获取核心容器对象
ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
JdbcUtils utils = (JdbcUtils) ac.getBean("jdbcUtils");
System.out.println(utils); // com.mysql.jdbc.Driver,3306
}
该注解通过可以实现对基本数据类型和String类型的注入,并且是支持使用Spring的EL表达式。那么对于Spring的EL表达式的语法就是:${表达式}。
以上注解的作用就和在xml配置文件中的bean标签中写一个标签的作用是一样的
方式:
Bean的定义,仍然采用7.3.1章节所定义好的Bean,唯一的区别是UserServiceImpl这个Bean修改作用域,修改如下:
@Service("userService")
@Scope("singleton")
public class UserServiceImpl implements UserService{
}
@Test
public void testUserServiceImpl() throws Exception{
// 1、读取主配置文件信息,获取核心容器对象
ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
UserService userService = (UserService) ac.getBean("userService");
UserService userService2 = (UserService) ac.getBean("userService");
System.out.println(userService == userService2); // true
}
改造一下:如果将上例的UserServiceImpl中的Scope改为prototype,则再次测试的时候会返回false。
Scope该注解的作用就和在bean标签中使用scope属性实现的功能是一样的
? Bean的定义,仍然采用7.3.1章节所定义好的Bean,唯一的区别是UserServiceImpl这个Bean添加了一个初始化方法和销毁方法。
@Service("userService")
public class UserServiceImpl implements UserService {
@PostConstruct
public void init(){
System.out.println("对象初始化了");
}
@PreDestroy
public void destroy(){
System.out.println("对象销毁了");
}
}
@Test
public void testUserServiceImpl() throws Exception{
// 1、读取主配置文件信息,获取核心容器对象
ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
ac.close();
}
// 对象初始化了
// 对象销毁了
? 这个两个注解作用就和在bean标签中使用init-method和destroy-methode的作用是一样的。