IOC 操作 Bean 管理(基于注解方式)

1、什么是注解

(1)注解是代码特殊标记,格式:@注解名称(属性名称=属性值, 属性名称=属性值…)

(2)使用注解,注解作用在类上面,方法上面,属性上面

(3)使用注解目的:简化 xml 配置

2、Spring 针对 Bean 管理中创建对象提供注解

下面四个注解功能是一样的,都可以用来创建 bean 实例

(1)@Component

(2)@Service

(3)@Controller

(4)@Repository

3、基于注解方式实现对象创建
第一步 引入依赖 (引入spring-aop jar包)
第二步 开启组件扫描
1
2
<!--开启组件扫描 1.如果扫描多个包,多个包使用逗号隔开;2.扫描包上层目录-->
<context:component-scan base-package="com.atguigu"/>
第三步 创建类,在类上面添加创建对象注解
1
2
3
4
5
6
7
8
9
10
11
//在注解里面 value 属性值可以省略不写(这个类只有一个bean)
//默认值是类名称,首字母小写
//UserService -- userService
@Component(value = "userService")
//注解等同于XML配置文件:<bean id="userService" class=".."/>
@Scope("prototype")//控制生成单个或多个对象 "singleton "或"prototype" 默认singleton
public class UserService {
public void add() {
System.out.println("service add.......");
}
}

Spring提供@Component 注解的三个衍生注解:

@Controller:用于表现层bean的定义

@Service:用于业务层bean的定义

@Repository:用于数据层bean的定义

4、开启组件扫描细节配置
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
<!--示例 1
use-default-filters="false" 表示现在不使用默认 filter,自己配置 filter
context:include-filter ,设置扫描哪些内容
-->
<context:component-scan base-package="com.atguigu" use-defaultfilters="false">
<context:include-filter type="annotation"

expression="org.springframework.stereotype.Controller"/><!--代表只扫描Controller注解的类-->
</context:component-scan>
<!--示例 2
下面配置扫描包所有内容
context:exclude-filter: 设置哪些内容不进行扫描
-->
<context:component-scan base-package="com.atguigu">
<context:exclude-filter type="annotation"

expression="org.springframework.stereotype.Controller"/><!--表示Controller注解的类之外一切都进行扫描-->
</context:component-scan>
5、完全注解开发
(1)创建配置类,替代 xml 配置文件
1
2
3
4
5
@Configuration //作为配置类,替代 xml 配置文件
@ComponentScan({"com.atguigu.dao","com.atguigu.service"})//多个扫描,数组格式
public class SpringConfig {

}
(2)编写测试类
1
2
3
4
5
6
7
8
@Test
public void testService2() {
//加载配置类 AnnotationConfigApplicationContext 读取配置java类
ApplicationContext context=new AnnotationConfigApplicationContext(SpringConfig.class);
UserService userService = context.getBean(UserService.class);
System.out.println(userService);
userService.add();
}

Bean的生命周期管理(初始化Bean和销毁Bean方法改写)

1
2
3
4
5
6
7
8
@PostConstruct
public void init(){
System.out.println("bookDao初始化....");
}
@PreDestroy
public void destroy(){
System.out.println("bookDao销毁....");
}
1
2
3
4
5
6
7
//ApplicationContext不具备销毁方法,故用AnnotationConfigApplicationContext
public static void main(String[] args) {
AnnotationConfigApplicationContext c=new AnnotationConfigApplicationContext(SpringConfig.class);
BookDaoImpl bean = c.getBean(BookDaoImpl.class);
bean.save();
c.close();//这样就能执行我们对于bean的销毁操作
}
6、基于注解方式实现属性注入
(1)@Autowired:根据属性类型进行自动装配

第一步 把 service 和 dao 对象创建,在 service 和 dao 类添加创建对象注解

第二步 在 service 注入 dao 对象,在 service 类添加 dao 类型属性,在属性上面使用注解

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
@Service
public class UserService {
//定义 dao 类型属性
//不需要添加 set 方法
//添加注入属性注解
@Autowired
private UserDao userDao;
public void add() {
System.out.println("service add.......");
userDao.add();
}
}

//Dao实现类
@Repository
//@Repository(value = "userDaoImpl1")
public class UserDaoImpl implements UserDao {
@Override
public void add() {
System.out.println("dao add.....");
}
}

自动装配基于反射设计创建对象并暴力反射对应属性为私有化属性初始化数据,因此无需提供setter方法

自动装配建议使用无参构造器方法创建对象(默认),如果不提供对应构造方法,请提供唯一的构造方法

(2)@Qualifier:根据名称进行注入,这个@Qualifier 注解的使用,和上面@Autowired 一起使用
1
2
3
4
5
6
7
//定义 dao 类型属性
//不需要添加 set 方法
//添加注入属性注解
@Autowired //根据类型进行注入
@Qualifier(value = "userDaoImpl1") //开启指定名称装配bean
//根据名称进行注入(目的在于区别同一接口下有多个实现类,根据类型就无法选择,从而出错!)
private UserDao userDao;
(3)@Resource:可以根据类型注入,也可以根据名称注入(它属于javax包下的注解,不推荐使用!)
1
2
3
//@Resource //根据类型进行注入
@Resource(name = "userDaoImpl1") //根据名称进行注入
private UserDao userDao;
(4)@Value:注入普通类型属性

情况一:

1
2
@Value(value = "abc")
private String name

情况二:加载properties文件 注入普通类型属性

1
2
3
4
5
6
7
//使用@PropertySource注解加载properties文件
//在Spring配置类中
@Configuration
@ComponentScan("com.kk")
@PropertySource({"jdbc.properties"})
public class SpringConfig {
}

路径只支持单一文件配置,多文件使用数组格式配置,不允许使用通配符*

1
2
@Value(value = "${name}")
private String name
7.注解开发管理第三方Bean
1
2
3
4
5
6
7
8
9
10
11
12
13
@Configuration
public class SpringConfig {
//1.定义一个方法管理相关里的第三方Bean
//2.添加注解@Bean表示该方法返回值是一个bean
@Bean
public DruidDataSource dataSource(){
DruidDataSource druidDataSource=new DruidDataSource();
druidDataSource.setDriverClassName("com.mysql.jdbc.Driver");
druidDataSource.setUrl("jdbc:mysql://localhost:3306/db");
druidDataSource.setUsername("root");
druidDataSource.setPassword("457631");
return druidDataSource;
}

不建议全部都写入配置类里:应使用独立配置类管理第三方bean

将独立的配置类加入核心配置类:

方式一:导入式(推荐使用)

1
2
3
4
5
6
public class JdbcConfig {
@Bean
public DruidDataSource dataSource(){
//相关配置
return druidDataSource;
}

使用@Import注解手动加入配置类到核心配置,此注解只能添加一次,多个数据用数组格式

1
2
3
4
5
6
@Configuration
@ComponentScan("com.kk")
@Import({JdbcConfig.class})
public class SpringConfig {

}

方式二:扫描式

1
2
3
4
5
6
7
@Configuration
public class JdbcConfig {
@Bean
public DruidDataSource dataSource(){
//相关配置
return druidDataSource;
}

使用@ComponentScan注解扫描配置类所在的包,加载对应的配置类信息

1
2
3
4
5
@Configuration
@ComponentScan("com.kk")
public class SpringConfig {

}

第三方Bean简单类型依赖注入:用成员变量

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class JdbcConfig {
@Vlaue("com.mysql.jdbc.Driver")
private String driver;
@Vlaue("jdbc:mysql://localhost:3306/db")
private String url;
@Vlaue("root")
private String user;
@Vlaue("457631")
private String password;
@Bean
public DruidDataSource dataSource(){
DruidDataSource druidDataSource=new DruidDataSource();
druidDataSource.setDriverClassName("com.mysql.jdbc.Driver");
druidDataSource.setUrl("jdbc:mysql://localhost:3306/db");
druidDataSource.setUsername("root");
druidDataSource.setPassword("457631");
return druidDataSource;
}

第三方Bean引用类型注入:用方法形参

1
2
3
4
5
6
7
@Bean
public DruidDataSource dataSource(BookDao bookDao){
System.out.println(bookDao);
DruidDataSource druidDataSource=new DruidDataSource();
//相关配置
return druidDataSource;
}

引用类型注入只需要为bean定义方法设置形参即可,容器会根据类型自动装配对象