当前位置: 首页 > news >正文

Spring DI详解--依赖注入的三种方式及优缺点分析

一、什么是DI?

DI(Dependency Injection,依赖注入)是 IoC(控制反转) 思想的最典型实现方式,核心目标只有一个:

让对象不再自己“找”依赖,而是由外部容器“送”依赖进来,从而彻底解耦。

一句话看懂DI:以创建狗这个实例举例

过去:自己 new

//当我们创建狗这个实例时,需要通过 new 关键实现对象的实例化
public class testDiBlog {public static void main(String[] args) {Dog dog=new Dog();}
}

现在:依赖注入(这里是构造注入)

public class testDiBlog {private Dog dog; //只声明,不创建,容器将实例送进来public testDiBlog(Dog dog) { //解耦this.dog = dog;}
}

spring 依赖注入的方式有三种,上述是其中一种(构造方法注入)其他两种分别是:属性注入、Setter方法注入,接下来让我们仔细的来看看这三种注入方式该如何进行书写。

二、依赖注入

 首先创建 Dog 类,创建 run 方法

@Getter @Setter
public class Dog {private String name;public void run(){System.out.println("running....");}
}

将 Dog 类交给Spring 进行管理(PS:@Bean(方法注解) 需要搭配 五大类注解(@Controller、@Service、@Component、@Repository、@Configuration)使用)

@Configuration
public class DogConfig {@Beanpublic Dog dog(){Dog dog=new Dog();dog.setName("旺财");return dog;}}

对上面代码的解释:

  1. Spring 管理的对象 = dog() 方法返回的那只 Dog 实例

  2. 类型 = Dog(返回值类型)

  3. Bean 名称 = dog(默认等于方法名,等价于 @Bean("dog")

2.1 属性注入

属性注入是使用 @Autowired 注解

@SpringBootTest
class SpringPrincipleApplicationTests {@Autowired //属性注入,使用注解Dog dog; //拿到 dog 实例@Testvoid DogTest(){dog.run(); //调用实例方法}
}

当 @Autowired 被注释掉时,此时的执行结果显示空指针异常

2.2 构造方法注入

使用构造方法注入Bean

//注入Bean
@Controller
public class TestDog2 {private Dog dog;public TestDog2(Dog dog) { this.dog = dog;}public void run(){dog.run();}}//启动spring(此段代码与上面的代码在idea中不是位于同一个类中,这里是为了方便观看写在一起)@SpringBootApplication
public class SpringPrincipleApplication {public static void main(String[] args) {ConfigurableApplicationContext context = SpringApplication.run(SpringPrincipleApplication.class, args);TestDog2 bean = context.getBean(TestDog2.class);bean.run();}
}

执行结果如下:

此时 TestDog2 方法中只有一个构造方法,我们知道如果当类中没有构造方法时,编译器会默认调用一个无参的构造方法,既然现在是构造方法注入,如果现在我们将这个默认的无参构造函数加上,程序执行的结果还会是这样吗??

@Controller
public class TestDog2 {private Dog dog;public TestDog2() {System.out.println("无参构造方法...");//这里打印一下日志为了方便观察}public TestDog2(Dog dog) {System.out.println("有参构造方法...");this.dog = dog;}public void run(){dog.run();}}

结果执行如下:

可以看到这里执行了这个无参的构造方法,且报出了空指针异常,报空指针异常是因为没有执行下面的有参构造方法,dog 没有进行赋值,后面调用了 dog 的 run 方法,此时 dog 为null。此时的解决方法是在有参构造方法上添加 @Autowried 注解,当添加注解后会告诉 Spring  默认帮我执行带注解的构造方法。修改如下:

@Controller
public class TestDog2 {private Dog dog;public TestDog2() {System.out.println("无参构造方法...");}@Autowired //添加注解,指定默认的构造方法public TestDog2(Dog dog) {System.out.println("有参构造方法...");this.dog = dog;}public void run(){dog.run();}}

执行结果如下:

不知道有没有小伙伴注意到下图这里的 dog 参数,在代码中我们没有给它传递参数,这个 dog 是从哪儿来的呢?构造函数注入时,Spring 必须能把所有参数都解析成容器里的 Bean,去进行查找,如果找到了就进行相应的赋值,只要有一个参数匹配不到Bean(类型+名称)启动就会失败并抛出:

No qualifying bean of type 'xxx.xxx' available: expected at least 1 bean which qualifies as autowire candidate.

2.3 Setter方法注入

Setter 注⼊和属性的 Setter ⽅法实现类似,只不过在设置 set ⽅法的时候需要加上 @Autowired 注解

@Controller
public class TestDog3 {private Dog dog;@Autowiredpublic void setDog(Dog dog) {this.dog = dog;}public void run(){System.out.println("这是TestDog3....");dog.run();}
}

执行结果如下:

去掉 @Autowired 后,执行结果如下:

三、优缺点分析

再进行优缺点分析之前,主播先提出一个问题,不知道小伙伴们是否还记得 final 关键字修饰得变量有什么特点?我们知道被 final 修饰得变量初始化要么再最初定义变量得时候就初始化,要么就是再构造器中被初始化。当我们回想起这一点后我们再来看这三种注入方式,不难发现,只有构造方法注入可以注入 final 修饰得变量,Setter 和属性注入不可以注入 final 修饰得变量。

3.1 原因分析

首先我们来看看 spring 创建对象的流程:① 分配空白内存 → ② 默认值(0/null) → ③ 构造代码块/构造器(final 唯一合法写入点) → ④ 对象头设置 → ⑤ 返回引用。这里一旦构造器返回,final 修饰的字段就进入了“只读”模式,后续任何赋值都会编译失败。

public class User {private final String name;public User(String name){ this.name = name; } // 合法public void setName(String name){ this.name = name; } // ❌ 编译错误
}

属性注入发生的时间段在返回引用之后,也就是流程⑤之后,spring 属性注入时机(源码级)

//java源码
AbstractAutowireCapableBeanFactory#populateBean
Field field = UserController.class.getDeclaredField("userService");
field.set(controllerInstance, userServiceImpl);   // 反射 putfield
  • JVM 校验:发现对 final 字段 执行 putfield → 直接抛

    java.lang.IllegalAccessError: Update to final field

3.2 spring三种注入的时间轴

注入方式触发时刻对象状态能否再写final
属性注入第⑤步:对象引用返回对象创建完成❌ 拒绝
Setter注入第⑤步:对象引用返回对象创建完成❌ 拒绝
构造器注入第③步:构造器里对象正在创建✅ 允许

3.3 时间轴再次对照

步骤时刻final可否写入spring属性注入是否在此
类加载类装载模板❌    ❌    不参与
new构造器类✅ 唯一机会❌    尚未开始
构造器返回对象已创建❌ 锁死❌    尚未开始
populateBean反射字段赋值❌ 抛错✅ 在这里发生 →失败

3.4 优缺点总结

通过上面的分析我们可以总结出三种注入方式的优缺点

方式优点缺点
属性注入
简洁,使⽤⽅便
只能⽤于 IoC 容器,如果是⾮ IoC 容器不可⽤,并且只有在使⽤的时候才会出现 NPE(空指
针异常)
不能注⼊一个Final修饰的属性
构造方法注入
可以注⼊final修饰的属性
注⼊的对象不会被修改
依赖对象在使⽤前⼀定会被完全初始化,因为依赖是在类的构造⽅法中执⾏的,⽽构造⽅法 是在类加载阶段就会执⾏的⽅法.
通⽤性好, 构造⽅法是JDK⽀持的, 所以更换任何框架,他都是适⽤的
注⼊多个对象时, 代码会⽐较繁琐
Setter注入
⽅便在类实例之后, 重新对该对象进⾏配置或者注⼊
不能注⼊⼀个Final修饰的属性
注⼊对象可能会被改变, 因为setter⽅法可能会被多次调⽤,就有被修改的风险

四、@Autowired存在的问题

当同一个类型的对象有多个的时候,此时又会发生什么状况呢??

@Component
public class TestUser {@Beanpublic User user1(){User user=new User();user.setName("图图");return user;  //对象1}@Beanpublic User user2(){User user=new User();user.setName("小美");return user;  //对象2}
}

错误提示:这里不只有一个User Bean对象,当同⼀类型存在多个bean时, 使⽤@Autowired会存在问题。

如何解决上述问题呢?Spring提供了以下⼏种解决⽅案: • @Primary • @Qualifier • @Resource

4.1 解决方法之 @Primary

使⽤@Primary注解:当存在多个相同类型的Bean注⼊时,加上@Primary注解,来确定默认的实现.

@Component
public class TestUser {@Primary  //指定该 Bean为默认的 Bean@Beanpublic User user1(){User user=new User();user.setName("图图");return user;}@Beanpublic User user2(){User user=new User();user.setName("小美");return user;}
}
@Controller
public class UserController {@Autowiredprivate User user; //注入成功没有报错public void desc(){user.desc();}
}

注意:@Qualifier注解不能单独使⽤,必须配合@Autowired使⽤

4.2 解决方法之 @Qualifier

使⽤ @Qualifier 注解:指定当前要注⼊的 bean 对象。 在 @Qualifier 的 value 属性中,指定注⼊的 bean 的名称。 
@Controller
public class UserController {@Qualifier("user1")  //添加注入指定Bean@Autowiredprivate User user;public void desc(){user.desc();}
}

4.3 解决方法之 @Resource

使⽤@Resource注解:是按照bean的名称进⾏注⼊。通过name属性指定要注⼊的bean的名称。

@Controller
public class UserController {@Resource(name= "user1")private User user;public void desc(){user.desc();}
}

五、@Resource 和 @Autowired 的区别

1. @Autowired 是spring框架提供的注解,⽽@Resource是JDK提供的注解
2. @Autowired 默认是按照类型注⼊,当同类型有多个实例时,也会根据名称去进行匹配,⽽@Resource是按照名称注⼊,按名称肯定也会需要类型是一致的.
3. 相⽐于 @Autowired 来说,@Resource ⽀持更多的参数设置,例如 name 设置,根据名称获取 Bean

六、@Autowired装配顺序



文章转载自:

http://r90LDVyl.hgwsj.cn
http://TGyQrgxi.hgwsj.cn
http://ZWCwq7ca.hgwsj.cn
http://42sOgE4K.hgwsj.cn
http://7ypjLzaV.hgwsj.cn
http://Fh1H5rNU.hgwsj.cn
http://ZKbI0srF.hgwsj.cn
http://pSlXLdYQ.hgwsj.cn
http://1XR17dVW.hgwsj.cn
http://szsLJHP5.hgwsj.cn
http://frR35RzM.hgwsj.cn
http://S2kYTqtt.hgwsj.cn
http://2QDZwJSa.hgwsj.cn
http://BBMsyeGz.hgwsj.cn
http://dGhhwutI.hgwsj.cn
http://NHv4oCnY.hgwsj.cn
http://OHbdSlKR.hgwsj.cn
http://XcWFRWba.hgwsj.cn
http://5biUWqAl.hgwsj.cn
http://0R16WIp2.hgwsj.cn
http://UStouCwr.hgwsj.cn
http://zhrAnxex.hgwsj.cn
http://fzQlOpGy.hgwsj.cn
http://deVq58Jg.hgwsj.cn
http://XyR3k6QK.hgwsj.cn
http://h03LeqYh.hgwsj.cn
http://I1NKkPOg.hgwsj.cn
http://H9ghhzHq.hgwsj.cn
http://caVcqnSS.hgwsj.cn
http://g3Kwf3hT.hgwsj.cn
http://www.dtcms.com/a/368478.html

相关文章:

  • 苹果TF签名全称TestFlight签名,需要怎么做才可以上架呢?
  • 小团队如何高效完成 uni-app iOS 上架,从分工到工具组合的实战经验
  • 华为认证HCIA备考知识点 :IP路由基础(含配置案例)
  • AI测试:自动化测试框架、智能缺陷检测、A/B测试优化
  • 从零到上线:直播美颜SDK中人脸美型功能的技术实现与效果优化
  • 大数据毕业设计选题推荐-基于大数据的高级大豆农业数据分析与可视化系统-Hadoop-Spark-数据可视化-BigData
  • 自演化大语言模型的技术背景
  • 3D目标跟踪重磅突破!TrackAny3D实现「类别无关」统一建模,多项SOTA达成!
  • Ubuntu中使用nginx-rtmp-module实现视频点播
  • 一文教您解决win11运行Ubuntu,wsl相关命令出现系统找不到指定文件的错误提示
  • 从零开始:用uv构建并发布一个Python CLI应用,集成CI/CD自动化发布与Docker容器化部署
  • Ubuntu 文件权限管理
  • [相机成像] 彩色相机成像 “灰蒙蒙” 问题排查与解决记录
  • STM32传感器模块编程实践(十六)DIY人脸识别智能垃圾桶模型
  • vscode连接SSH
  • 在VSCode中更新或安装最新版的npx和uv工具
  • 如何选择文件夹然后用vscode直接打开
  • 命令行中如如何打开目录?vscode中如何打开目录
  • 一阶低通滤波器应用示例(演示)
  • 如何选择适合的实验室铸铁地板和铸铁试验平板?专业人士帮助指南
  • centos sshd:xxx.xxx.xxx.xxx:allow 如何设置
  • 如果要获得Ecovadis认证需要费用是多少?
  • SQL Server全链路安全防护
  • C++_数据结构
  • MySQL数据库精研之旅第十六期:深度拆解事务核心(上)
  • solidity函数篇
  • 数据库(基础操作)
  • Python+DRVT 从外部调用 Revit:批量创建梁
  • 【软考架构】V模型、W模型、增量模型和螺旋模型
  • 华为云昇腾云服务