IoC & DI

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值