关于 IoC 的含义,推荐看IoC含义介绍(Spring的核心思想)
喜欢 Java 的推荐点一个免费的关注,主页有更多 Java 内容
通过上述的博客我们知道了 IoC 的含义,既然 Spring 是⼀个 IoC(控制反转)容器,作为容器, 那么它就具备两个最基础的功能:‘存’和‘取’
Spring容器管理的主要是对象, 这些对象, 我们称之为"Bean". 我们把这些对象交由Spring管理,我们的程序只需要告诉Spring,哪些对象需要存,以及在需要时从Spring中取出 对象
要想将对象交给 Spring 进行管理,Spring 提供了丰富的注解来实现这一功能
共有两种注解类型可以实现:
1. 类注解:@Controller、@Service、@Repository、@Component、@Configuration.(五大注解)
2. ⽅法注解:@Bean.
对应 Spring Web MVC 开发,三层架构中的 Controller (控制层),案例:
如下的代码,对 UserController 这个控制类加上 @Controller 注解就表明将 UserController 类的对象交给 Spring 进行管理
@Controller // 将对象存储到 Spring 中
public class UserController {
public void sayHi(){
System.out.println("hi,UserController");
}
}
通过下面的代码来验证,我们是否成功将 UserController 类的对象交给 Spring 进行管理,SpringIocDemoApplication 是当前 Spring Web MVC 项目中的启动类,SpringApplication 是 Spring Boot框架中用于启动应用程序的类,SpringApplication 调用 run 方法启动了当前的项目,并且返回了 Spring 的 IoC 容器(也叫做 Spring 上下文),用 ApplicationContext 类型的对象 context 来接收,获得了 Spring 的 IoC 容器后,调用 getBean 方法,传入 UserController 的类对象 我们便获得了 Spring IoC 容器中管理的 UserController 类型的 对象(Bean)
@SpringBootApplication
public class SpringIocDemoApplication {
public static void main(String[] args) {
//获取Spring上下⽂对象
ApplicationContext context = SpringApplication.run(SpringIocDemoApplicatio
//从Spring上下⽂中获取对象
UserController userController = context.getBean(UserController.class);
//使⽤对象
userController.sayHi();
}
}
获得 userController 对象后调用 sayHi 方法,我们便在控制台发现 sayHi 方法正确执行,代表我们成功将 UserController 类型的对象交给了 Spring 进行管理(成功将创建 UserController 对象的控制权交给了 Spring ,控制反转),并成功从 Spring 获取到对象
对应 Spring Web MVC 开发,三层架构中的 Service(业务逻辑层),案例:
如下的代码,对 UserService 这个业务逻辑类加上 @Service 注解就表明将 UserService 类的对象交给 Spring 进行管理
@Service
public class UserService {
public void sayHi(String name) {
System.out.println("Hi," + name);
}
}
通过下面的代码来验证,我们是否成功将 UserService 类的对象交给 Spring 进行管理
@SpringBootApplication
public class SpringIocDemoApplication {
public static void main(String[] args) {
//获取Spring上下⽂对象
ApplicationContext context = SpringApplication.run(SpringIocDemoApplicatio
//从Spring中获取UserService对象
UserService userService = context.getBean(UserService.class);
//使⽤对象
userService.sayHi();
}
获得 UserService 对象后调用 sayHi 方法,我们便在控制台发现 sayHi 方法正确执行,代表我们成功将 UserService 类型的对象交给了 Spring 进行管理(成功将创建 UserService 对象的控制权交给了 Spring ,控制反转)
通过上述 @Controller 和 @Service 注解的使用以及检验,我们会发现有很多的相似之处,实际上 @Controller、@Service、@Repository、@Component、@Configuration 这五大注解的除了名称不同以外,作用都是将 对象 (Bean)交给 Spring 进行管理(将创建对象的控制权交给 Spring )(此处我们不谈这些注解在其他地方的作用,比如 @Controlle 还代表该类是一个控制类,在接收到 Http 请求后 Spring 要遍历 带有 @Controlle 注解的类)
既然 @Controller、@Service、@Repository、@Component、@Configuration 这五大注解在这里的作用都是一样的,那 Spring 为什么要提供这么多注解呢?实际上这和 Spring Web MVC 开发的三层架构对应,@Controller 这个注解对应三层架构的 Controller 控制层,@Service 对应三层架构的 Service 业务逻辑层,@Repository 对应三层架构的 Dao 数据层,不处于这三层中的类就用 @Component(组件) 注解,@Configuration 表示配置类的注解
提供 @Controller、@Service、@Repository、@Component、@Configuration 这五大注解主要是为了让程序猿针对不同层次的类用对应的注解,让程序员看到类注解之后,就能直接了解当前类的⽤途.
注意:使用五大注解,Spring 只会管理类的一个对象,无论获取这个类的对象多少次,都是同一个对象,是单例模式,那我们想要 Spring 管理一个类的多个对象的话就要看下面的方法注解@Bean
五大注解存在一些无法解决的问题,此时我们就需要使用方法注解,比如下面的问题:
1. 使⽤外部包⾥的类,没办法添加类注解
2. ⼀个类,需要多个对象,⽐如多个数据源
这种场景,我们就需要使⽤⽅法注解 @Bean
方法注解的使用方式如下:
根据下面的代码,我们便创建了一个类型为 User ,名称为 user 的对象交给了 Spring 管理(方法名就是 Spring 管理的对象名)
@Component
public class BeanConfig {
@Bean
public User user(){
User user = new User();
user.setName("zhangsan");
user.setAge(18);
return user;
}
}
注意:方法注解 @Bean 要搭配五大注解一起使用,因为 Spring 在项目运行时扫描的是加上相关注解,需要 Spring 处理的类,所以要是不加五大注解,Spring 压根就不会扫描到 @Bean 注解,也就不会管理方法创建的对象
对于同⼀个类,如何定义多个对象呢? 代码如下
我们通过下面的代码便创建了两个 User 类型的对象 user1 和 user2 交给 Spring 进行管理
@Component
public class BeanConfig {
@Bean
public User user1(){
User user = new User();
user.setName("zhangsan");
user.setAge(18);
return user;
}
@Bean
public User user2(){
User user = new User();
user.setName("lisi");
user.setAge(19);
return user;
}
}
注意:当 Spring 管理的一个类中有多个对象的时候,我们获取就不能通过类对象来获取了,要不然会报错,我们可以通过对象(Bean)的名称或者对象(Bean) 的名称加类型来从 Spring 那里获取指定的对象
@SpringBootApplication2
public class SpringIocDemoApplication {
public static void main(String[] args) {
//获取Spring上下⽂对象
ApplicationContext context = SpringApplication.run(SpringIocDemoApplicatio
//从Spring上下⽂中获取对象
//根据bean类型, 从Spring上下⽂中获取对象
UserController userController1 = context.getBean(UserController.class);
//根据bean名称, 从Spring上下⽂中获取对象
UserController userController2 = (UserController)context.getBean("userController");
//根据bean名称+类型, 从Spring上下⽂中获取对象
UserController userController3 = context.getBean("userController",UserController.class);
}
}
根据上面的介绍我们知道,使用 @Bean 注解让 Spring 管理方法创建的对象时,对象的名称就是方法名称,那我们可不可以修改 Spring 管理的对象名称呢?那必然是可以的
通过在 @Bean 注解中设置属性 name 来重命名对象名称,如下的方式为该对象设置了两个名称,所以我们查找名称 u1 和 user1 都是这个对象
@Bean(name = {"u1","user1"})
public User user1(){
User user = new User();
user.setName("zhangsan");
user.setAge(18);
return user;
}
而如果一个对象就一个名称的话,重命名就更加的简单,直接在 @Bean 注解中加上要修改的名称即可
@Bean("u1")
public User user1(){
User user = new User();
user.setName("zhangsan");
user.setAge(18);
return user;
}
Bean的获取就是获取 Spring 管理的对象,也可以叫做依赖注⼊DI,简单来说,就是把对象取出来放到某个类的属性中
关于依赖注⼊,Spring也给我们提供了三种⽅式:
1. 属性注⼊(Field Injection)
2. 构造⽅法注⼊(Constructor Injection)
3. Setter 注⼊(Setter Injection)
属性注⼊是使⽤注解 @Autowired 实现的
如下代码,UserController 类中有 UserService 类型的属性需要赋值,在 UserService 类型的属性前加上 @Autowired 注解,表明要从 Spring 那里获取到 UserService 类型的对象赋值给 userService 变量
public class UserController {
//注⼊⽅法1: 属性注⼊
@Autowired
private UserService userService;
public void sayHi(){
System.out.println("hi,UserController...");
userService.sayHi();
}
}
构造⽅法注⼊是在类的构造⽅法中实现注⼊,如下代码所⽰:
在构造方法前加上 @Autowired 注解,表明要从 Spring 那里获取到构造方法的参数,此时构造方法需要 UserService 类型的对象,便会从 Spring 那里获取到,完成构造方法
public class UserController2 {
//注⼊⽅法2: 构造⽅法
private UserService userService;
@Autowired
public UserController2(UserService userService) {
this.userService = userService;
}
public void sayHi(){
System.out.println("hi,UserController2...");
userService.sayHi();
}
}
注意:如果类只有⼀个构造⽅法,那么 @Autowired 注解可以省略;如果类中有多个构造⽅法, 那么需要添加上 @Autowired 来明确指定到底使⽤哪个构造⽅法。
Setter 注⼊和属性的 Setter ⽅法实现类似,只不过在设置 set ⽅法的时候需要加 @Autowired 注 解 ,如下代码所⽰:
在 set 方法前加上 @Autowired 注解就表示要从 Spring 那里获取到 set 方法的参数
@Controller
public class UserController3 {
//注⼊⽅法3: Setter⽅法注⼊
private UserService userService;
@Autowired
public void setUserService(UserService userService) {
this.userService = userService;
}
}
◦ 优点: 简洁,使⽤⽅便;
◦ 缺点: ▪ 1.只能⽤于 IoC 容器,如果是⾮ IoC 容器不可⽤,并且只有在使⽤的时候才会出现 NPE(空 指针异常)
▪ 不能注⼊⼀个Final修饰的属性
◦ 优点: ▪ 1.可以注⼊final修饰的属性
▪ 2.注⼊的对象不会被修改
▪ 3.依赖对象在使⽤前⼀定会被完全初始化,因为依赖是在类的构造⽅法中执⾏的,⽽构造⽅ 法是在类加载阶段就会执⾏的⽅法。
▪ 4.通⽤性好,构造⽅法是JDK⽀持的,所以更换任何框架,他都是适⽤的
◦ 缺点: ▪ 注⼊多个对象时,代码会⽐较繁琐
◦ 优点 ⽅便在类实例之后,重新对该对象进⾏配置或者注⼊ ◦ 缺点:
▪ 1.不能注⼊⼀个Final修饰的属性
▪ 2.注⼊对象可能会被改变,因为setter⽅法可能会被多次调⽤,就有被修改的⻛险.
当同⼀类型存在多个 bean(对象) 时, 使⽤ @Autowired 会存在问题,因为 Spring 不知道你要获取该类型的哪个对象
假设 Spring 中管理了 UserService 类型的多个对象,此时在一个类中有 UserService 类型的属性,并且加上了 @Autowired 注解表明要从 Spring 那里获取属性的值,此时 Spring 会先进行名称匹配,要是属性的名称和 Spring 那里管理的对象名称相同,那 Spring 就知道此时要获取哪个对象了,但要是名称都不同,就会报错
如何解决上述问题呢?
Spring提供了以下⼏种解决⽅案:
• @Primary
• @Qualifier
• @Resource
如下代码,要将 User 类型的 user1 和 user2 对象交给 Spring 进行管理,在 user1 方法前加上 @Primary 注解,就表明了 user1 是主要的对象,这样在不明确程序需要获取 User 类型的哪个对象的时候,Spring 就会提供 user1 对象
@Component
public class BeanConfig {
@Primary //指定该bean为默认bean的实现
@Bean("u1")
public User user1(){
User user = new User();
user.setName("zhangsan");
user.setAge(18);
return user;
}
@Bean
public User user2() {
User user = new User();
user.setName("lisi");
user.setAge(19);
return user;
}
}
注意:但是实际上这个方法并不常用,因为要是设置了主要的对象,那么大多数情况下使用的都只会是主要的那个对象,该类型其他对象的存在就没有必要了
指定当前要注⼊的bean对象。在 @Qualifier 的value属性中,指定注⼊的 bean 的名称。
如下代码,加上注解 @Qualifier("user2") 就代表要获取 Spring 管理的名称为 user2 的对象注入到属性 user 中
public class UserController {
@Qualifier("user2") //指定bean名称
@Autowired
private User user;
public void sayHi(){
System.out.println("hi,UserController...");
System.out.println(user);
}
}
注意: @Qualifier 注解不能单独使⽤,必须配合 @Autowired 使⽤
是按照bean的名称进⾏注⼊。通过name属性指定要注⼊的bean的名称。
如下代码,加上注解 @Resource(name = "user2") 就代表要获取 Spring 管理的名称为 user2 的对象注入到属性 user 中
public class UserController {
@Resource(name = "user2")
private User user;
public void sayHi(){
System.out.println("hi,UserController...");
System.out.println(user);
}
}
@Autowired 是spring框架提供的注解,⽽@Resource是JDK提供的注解
@Autowired 默认是按照类型注⼊,⽽ @Resource 是按照名称注⼊( @Autowired 搭配@Qualifier 也可以按照名称注入)