4. Spring基于注解的IOC配置
1、准备
image.png image.png利用前一个工程作为模板修改
image.png
我们针对service层操作
package com.neuedu.service.impl;
import com.neuedu.dao.IAccountDao;
import com.neuedu.service.IAccountService;
import org.springframework.stereotype.Component;
/**
* 账户的业务层实现类
*/
@Component
public class AccountServiceImpl implements IAccountService {
private IAccountDao accountDao;
public void saveAccount(){
accountDao.saveAccount();
}
}
曾经的xml配置
<bean id="accountService" class="com.neuedu.service.impl.AccountServiceImpl"
scope="" init-method="" destroy-method="">
<property name="" value=""或者ref=""></property>
</bean>
2、用于创建对象的
作用:和在XML配置文件中编写一个<bean>标签实现的功能是一样的
@Component
@Component:用于把当前类对象存入spring容器中
属性:value,用于指定bean的id。当我们不写时,它的默认值是当前类名,且首字母改小写。如果指定value,则获取bean对象时使用该value作为id。
加入约束
image.png
<?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="com.neuedu"></context:component-scan>
测试
//1、获取核心容器对象
ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
//2、根据id获取Bean对象
IAccountService as = (IAccountService)ac.getBean("accountServiceImpl");
System.out.println(as);
image.png
由Component衍生的注解
Controller:一般用在表现层
Service:一般用在业务层
Repository:一般用在持久层
以上三个注解他们的作用和属性与Component是一模一样。
他们三个是spring框架为我们提供明确的三层使用的注解,使我们的三层对象更加清晰
例如:
package com.neuedu.dao.impl;
import com.neuedu.dao.IAccountDao;
import org.springframework.stereotype.Repository;
/**
* 账户的持久层实现类
*/
@Repository("accountDao")
public class AccountDaoImpl implements IAccountDao {
public void saveAccount(){
System.out.println("保存了账户");
}
}
IAccountDao adao = ac.getBean("accountDao", IAccountDao.class);
System.out.println(adao);
image.png
3、用于注入数据的
作用:和在xml配置文件中的bean标签中写一个<property>标签的作用是一样的
@Autowired:
作用:自动按照类型注入。只要容器中有唯一的一个bean对象类型和要注入的变量类型匹配,就可以注入成功
如果ioc容器中没有任何bean的类型和要注入的变量类型匹配,则报错。
如果Ioc容器中有多个类型匹配时:
先利用类型查找,同类型,再用变量名查找
出现位置:
可以是变量上,也可以是方法上
细节:
在使用注解注入时,set方法就不是必须的了。
/**
* 账户的业务层实现类
*/
@Service("accountService")
public class AccountServiceImpl implements IAccountService {
@Autowired
private IAccountDao accountDao;
public void saveAccount(){
accountDao.saveAccount();
}
}
//1、获取核心容器对象
ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
//2、根据id获取Bean对象
IAccountService as = (IAccountService)ac.getBean("accountService");
as.saveAccount();
image.png
@Qualifier:
作用:在按照类中注入的基础之上再按照名称注入。它在给类成员注入时不能单独使用。但是在给方法参数注入时可以(稍后说明)
属性:
value:用于指定注入bean的id。
@Resource
作用:直接按照bean的id注入。它可以独立使用
属性:
name:用于指定bean的id。
以上三个注入都只能注入其他bean类型的数据,而基本类型和String类型无法使用上述注解实现。另外,集合类型的注入只能通过XML来实现。
@Value
作用:用于注入基本类型和String类型的数据
属性:
value:用于指定数据的值。它可以使用spring中SpEL(也就是spring的el表达式)
SpEL的写法:${表达式}
4、用于改变作用范围的
作用:和在bean标签中使用scope属性实现的功能是一样的
@Scope
作用:用于指定bean的作用范围
属性:
value:指定范围的取值。常用取值:singleton prototype
//1.获取核心容器对象
ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
//2.根据id获取Bean对象
IAccountService as = (IAccountService)ac.getBean("accountService");
System.out.println(as == as2);
image.png
修改属性为:prototype
image.png image.png
5、和生命周期相关(了解)
作用:和在bean标签中使用init-method和destroy-methode的作用是一样的
@PreDestroy
作用:用于指定销毁方法
@PostConstruct
作用:用于指定初始化方法
@Service("accountService")
public class AccountServiceImpl implements IAccountService {
@Resource(name = "accountDao2")
private IAccountDao accountDao = null;
@PostConstruct
public void init(){
System.out.println("初始化方法执行了");
}
@PreDestroy
public void destroy(){
System.out.println("销毁方法执行了");
}
public void saveAccount(){
accountDao.saveAccount();
}
}
测试:
//1.获取核心容器对象
ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
//2.根据id获取Bean对象
IAccountService as = (IAccountService)ac.getBean("accountService");
as.saveAccount();
ac.close();
image.png
注意:多例模式不负责销毁,所以@Scope("prototype") 可以注释掉。