Spring 的两大核心思想 : IoC 和 AOP
我们要将对象的控制权交给Spring ,我们就需要告诉 Spring 哪些对象是需要帮我们进行创建的,这里有两类注解可以实现 :
类注解(@Controller @Service @Repository @Component @Configuration)和方法注解(@Bean)
这五大注解都表示把这个对象交给 Spring 进行管理和创建
五大注解和实际项目开发的关系 :
@Controller @Service @Repository 和三层架构有关系
Controller : 接收请求并返回响应
Service : 接收完请求具体要做些什么样的事情,真正业务的逻辑处理是什么都是 Service 做的
Dao : 对数据进行处理
@Component 存储组件相关,@Configuration 存储配置相关
我们查看五个类注解的原码 : @Controller @Service @Repository @Component @Configuration
这四个注解都是基于 @Component 实现的,可以认为是 @Component 的衍生类
但是是不是说我们所有程序都使用 @Component 就行呢,那是当然不行的,那搞这么多注解就没意义了
我们用代码尝试一下能否使用 @Service 代替 @Controller 呢?@Controller 是接收请求返回响应,看看换成 @Service 是否还能接收请求返回响应
- package com.example.ioc.controller;
-
- import org.springframework.stereotype.Controller;
- import org.springframework.web.bind.annotation.RequestBody;
- import org.springframework.web.bind.annotation.RequestMapping;
- import org.springframework.web.bind.annotation.ResponseBody;
-
- @Controller
- @ResponseBody
- public class TestController {
- @RequestMapping("/test")
- public String test(){
- return "测试Controller 和其他注解的区别";
- }
- }
现在我们把 @Controller 换成 @Service ,我们发现报错了
这是因为 Spring 对 Controller 进行了一些处理,我们想接收请求就只能使用 Controller ,就是说Controller必须作为我们程序的第一关,这是规范,不能使用其他注解
那么 @Service 这些可以使用 @Component 来代替呢? 这是可以的,但是并不建议
但是明明 @Service 和 @Controller 的原码明明差不多,为什么应用上有区别呢?这是因为原码只是注解的声明,只是声明了一个注解,但是 Spring 对于这个注解还会赋予一些别的功能,这些功能在 Spring 的原码里面
五大注解只能加在类上,并且只能加在自己的代码上,如果我引入了一个 jar包,也希望交给Spring 管理,这是没有办法加五大注解的,这时候就要使用 @Bean
@Bean 是方法注解 , @Bean 需要搭配五大注解来使用
使用 @Bean 注解时,bean 的名称是方法名
五大注解是,bean 的名称是类名的首字母转为小写,如果前两个字母都为大写,bean 的名称就是类名
还有另一个场景可以使用 @Bean ,对于一个类,我们需要定义多个对象时,比如数据库操作,定义多个数据源
因为当我们用别的注解的时候,拿到的是同一个数据,地址都一样,我们拿 @Configuration 举例
- package com.example.ioc.config;
- import org.springframework.context.annotation.Configuration;
-
- @Configuration
- public class UserConfig {
- public void doConfig(){
- System.out.println("do configurarion...");
- }
- }
- UserConfig userConfig = context.getBean(UserConfig.class);
- userConfig.doConfig();
- System.out.println("userConfig:"+userConfig);
-
- UserConfig userConfig1 = context.getBean(UserConfig.class);
- System.out.println("userConfig:"+userConfig1);
- System.out.println("userConfig == userConfig1"+(userConfig==userConfig1));
我们发现,两个数据的地址是完全相同的,它们是否相等的判定返回也是 true
接下来我们用 @Bean 试试看
- package com.example.ioc.config;
- import lombok.Data;
-
- @Data
- //假如这是第三方包下的一个类
- //现在我在我的项目中用到了 UserInfo,并且我需要用它定义多个对象
- public class UserInfo {
- private Integer id;
- private String name;
- private Integer age;
- }
- package com.example.ioc.config;
-
- import org.springframework.context.annotation.Bean;
- import org.springframework.context.annotation.Configuration;
-
- @Configuration
- public class BeanConfig {
- @Bean
- public UserInfo userInfo(){
- UserInfo userInfo = new UserInfo();
- userInfo.setId(1);
- userInfo.setName("zhangsan");
- userInfo.setAge(12);
- return userInfo;
- }
- @Bean
- public UserInfo userInfo2(){
- UserInfo userInfo = new UserInfo();
- userInfo.setId(2);
- userInfo.setName("wangwu");
- userInfo.setAge(13);
- return userInfo;
- }
- }
- package com.example.ioc;
-
- import com.example.ioc.component.UserComponent;
- import com.example.ioc.config.UserConfig;
- import com.example.ioc.config.UserInfo;
- import com.example.ioc.controller.UserController;
- import com.example.ioc.repo.UserRepository;
- import com.example.ioc.service.UserService;
- import org.springframework.boot.SpringApplication;
- import org.springframework.boot.autoconfigure.SpringBootApplication;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.annotation.Bean;
-
- @SpringBootApplication
- public class DemoApplication {
-
- public static void main(String[] args) {
- //Spring 上下文
- ApplicationContext context = SpringApplication.run(DemoApplication.class, args);
- //返回的就是 Spring 的运行环境
-
-
- //@Bean演示
- UserInfo userInfo = (UserInfo) context.getBean("userInfo");
- System.out.println(userInfo);
-
- UserInfo userInfo2 = (UserInfo) context.getBean("userInfo2");
- System.out.println(userInfo2);
-
- }
-
- }
这样就能返回两个不一样的内容了
当一个类型存在多个 bean 时,我们就不能使用类型来获取对象了,就要使用名称进行获取了,或者是名称+类型进行获取
- UserInfo userInfo1 = context.getBean("userInfo",UserInfo.class);
- System.out.println(userInfo1);
这样依然可以拿到
当我们想进行传参,代码改为 :
- package com.example.ioc.config;
-
- import org.springframework.context.annotation.Bean;
- import org.springframework.context.annotation.Configuration;
-
- @Configuration
- public class BeanConfig {
- @Bean
- public String name(){
- return "wangwu";
- }
- @Bean
- public UserInfo userInfo(String name){
- UserInfo userInfo = new UserInfo();
- userInfo.setId(1);
- userInfo.setName(name);
- userInfo.setAge(12);
- return userInfo;
- }
-
- }
运行结果如下
上面代码我把 public String name 改为 public String name2 依然可以被拿到,但是如果name和name2都存在,那么就根据名称去匹配,需要哪个就把名称放进传参的括号里
DI 依赖注入也可以叫做"属性装配"或者"依赖装配"
依赖可以简单认为是属性,比如下面代码中,userService 可以看做是 UserController 的属性
我们该如何完成依赖注入呢?
1.属性注入
我们用 @Autowired,这样就能把 Service 给引进来
属性注入以类型进行匹配,与注入的属性名称无关,但是如果一个类型存在多个对象时,优先进行名称匹配,如果名称都匹配不上,那就报错
@Autowired 无法注入一个被 final 修饰的属性(很少遇到这种情况)
- @Controller
- //用 @Controller 告诉 Spring 帮我们管理这个对象
- public class UserController {
- @Autowired
- private UserService userService;
- public void doController(){
- userService.doService();
- System.out.println("do Controller...");
- }
- }
-
-
-
- @Service
- public class UserService {
- public void doService(){
- System.out.println("do service....");
- }
- }
-
-
- @SpringBootApplication
- public class DemoApplication {
-
- public static void main(String[] args) {
- //Spring 上下文
- ApplicationContext context = SpringApplication.run(DemoApplication.class, args);
- //返回的就是 Spring 的运行环境
-
- UserController bean = context.getBean(UserController.class);
- bean.doController();
- }
- }
运行结果如下
2.构造方法注入
如果存只有一个构造函数,@Autowired 可以省略,当有多个构造函数的时候,默认使用无参的构造函数,如果没有无参的构造函数,我们要告诉 Spring 我们用哪个构造函数,在那个构造函数上面添加 @Autowired 就行
- package com.example.ioc.controller;
-
- import com.example.ioc.config.UserInfo;
- import com.example.ioc.service.UserService;
- import org.springframework.beans.factory.annotation.Autowired;
- import org.springframework.stereotype.Controller;
-
- @Controller
- //用 @Controller 告诉 Spring 帮我们管理这个对象
- public class UserController {
- //属性注入
- //@Autowired
- //private UserService userService;
-
- //构造方法注入
- private UserService userService;
- private UserInfo userInfo;
-
- // public UserController() {
- //
- // }
-
- public UserController(UserService userService) {
- this.userService = userService;
- }
-
- @Autowired
- public UserController(UserService userService, UserInfo userInfo) {
- this.userService = userService;
- this.userInfo = userInfo;
- }
-
- public void doController(){
- userService.doService();
- System.out.println("do Controller...");
- }
- }
3.Setter 方法注入
- public class UserController {
-
- private UserService userService;
-
- @Autowired
- public void setUserService(UserService userService) {
-
- this.userService = userService;
- }
-
- public void doController(){
- userService.doService();
- System.out.println("do Controller...");
- }
- }
当程序中同一个类型有多个对象时,使用@Autowired 会报错(一些情况下)
解决方法如下:
1,属性名和你需要使用的对象名保持一致
2.使用@Primary 注解标识默认的对象,优先用它
3.使用@Qualifier
4.使用@Resource