Java注解之@Autowired,一文掌握@Autowired注解知识(3)

news2025/1/11 22:44:49

在这里插入图片描述

🏆作者简介,普修罗双战士,一直追求不断学习和成长,在技术的道路上持续探索和实践。
🏆多年互联网行业从业经验,历任核心研发工程师,项目技术负责人。
🎉欢迎 👍点赞✍评论⭐收藏

Java注解知识专栏学习

Java注解知识云集访问地址备注
Java注解知识点(1)https://blog.csdn.net/m0_50308467/article/details/135240536Java注解专栏
Java注解知识点(2)https://blog.csdn.net/m0_50308467/article/details/135263693Java注解专栏
Java注解知识点(3)https://blog.csdn.net/m0_50308467/article/details/135278911Java注解专栏

文章目录

  • 🏆 学习Java注解之@Autowired
    • 🔎 Java 注解@Autowired学习(3)
      • 🍁🍁 01、如何解决在使用 @Autowired 注解时可能出现的循环依赖问题?
      • 🍁🍁 02、是否可以在静态字段上使用 @Autowired 注解?
      • 🍁🍁 03、如何处理在使用 @Autowired 注解时出现多个匹配候选对象的情况?
      • 🍁🍁 04、如何使用 @Autowired 注解注入集合类型的依赖?
      • 🍁🍁 05、是否可以在构造函数上同时使用 @Autowired 和 @Value 注解?
      • 🍁🍁 06、如何通过 @Autowired 注解注入通用的 ApplicationContext 对象?
      • 🍁🍁 07、@Autowired 注解的 required 属性的作用是什么?
      • 🍁🍁 08、如何自定义一个注解来替代 @Autowired 注解的功能?

🏆 学习Java注解之@Autowired

🔎 Java 注解@Autowired学习(3)

🍁🍁 01、如何解决在使用 @Autowired 注解时可能出现的循环依赖问题?

在使用 @Autowired 注解时,可能会出现循环依赖的问题。循环依赖指的是两个或多个 Bean 之间相互引用,无法完成依赖注入的情况。

为了解决循环依赖问题,Spring 提供了三种解决方案:

1. 构造器注入:使用构造器注入可以避免循环依赖问题。在每个类的构造器参数中,注入所需的依赖项。这样,在实例化每个类时,所有依赖关系都已解决,不会发生循环依赖的情况。

@Component
public class A {
    private B b;
    
    @Autowired
    public A(B b) {
        this.b = b;
    }
}

@Component
public class B {
    private A a;
    
    @Autowired
    public B(A a) {
        this.a = a;
    }
}

2. @Lazy 注解@Lazy 注解可以延迟解决循环依赖问题。通过将 @Lazy 注解应用到需要延迟注入的 bean 上,可以确保在需要使用时才进行依赖注入。

@Component
@Lazy
public class A {
    @Autowired
    private B b;
}

@Component
@Lazy
public class B {
    @Autowired
    private A a;
}

3. @DependsOn 注解@DependsOn 注解可以指定 bean 之间的依赖顺序,从而解决循环依赖问题。通过在需要指定顺序的 bean 上使用 @DependsOn 注解,可以确保被注解的 bean 在指定的 bean 之后实例化。

@Component
@DependsOn("b")
public class A {
    @Autowired
    private B b;
}

@Component
public class B {
    @Autowired
    private A a;
}

需要注意的是,循环依赖是一种不良的设计模式,尽量要避免出现循环依赖的情况。如果你在设计中出现了循环依赖,可以考虑重构代码,优化依赖关系,以减少循环依赖发生的可能性。

🍁🍁 02、是否可以在静态字段上使用 @Autowired 注解?

不可以在静态字段上使用 @Autowired 注解。@Autowired 注解是用来实现依赖注入的,它将自动将对应类型的 bean 注入到被注解的字段、构造器或方法参数上。但是,静态字段(static fields)属于类级别,而不是对象级别,无法通过依赖注入的方式进行注入。

@Autowired 注解通常用于实例化的对象,它将在对象创建时注入依赖的 bean。而静态字段不依赖于对象的创建和生命周期,它是类加载时就分配的静态资源。这意味着无法使用 @Autowired 注解将 bean 自动注入到静态字段上。

如果你需要在静态字段上引用依赖的 bean,可以考虑通过静态方法或其他方式手动获取该 bean,或者将依赖的 bean 作为参数传递给静态方法,以实现所需的功能。

🍁🍁 03、如何处理在使用 @Autowired 注解时出现多个匹配候选对象的情况?

在使用 @Autowired 注解时,如果存在多个匹配候选对象(multiple matching candidates),将会导致注入失败,Spring 框架会抛出 NoUniqueBeanDefinitionException 异常。

为了解决多个匹配候选对象的问题,Spring 框架提供了多种解决方案,具体如下:

1. 使用 @Qualifier 注解@Qualifier 注解可以和 @Autowired 注解一起使用,消除多个匹配候选项的歧义。@Qualifier 注解指定了要注入的 bean 的名称或 ID。

示例如下:

@Component("myComponentA")
public class MyComponentA implements MyComponentInterface {
    // ...
}

@Component("myComponentB")
public class MyComponentB implements MyComponentInterface {
    // ...
}

@Component
public class MyComponentConsumer {
    @Autowired
    @Qualifier("myComponentA")
    private MyComponentInterface myComponentA;

    @Autowired
    @Qualifier("myComponentB")
    private MyComponentInterface myComponentB;
}

2. 使用 @Primary 注解@Primary 注解用于标识 bean 的首选项,当一个类型的 bean 有多个候选对象时,Spring 框架将优先选择带有 @Primary 注解的 bean。

示例如下:

@Component
@Primary
public class MyPrimaryComponent implements MyComponentInterface {
    // ...
}

@Component
public class MyComponentConsumer {
    @Autowired
    private MyComponentInterface myComponent;
}

3. 使用 ListMap 类型:如果存在多个匹配候选项,可以使用 ListMap 类型来注入所有匹配的 bean。如果存在多个匹配项,Spring 框架会自动将它们注入一个 ListMap 实例中。可以使用 @Autowired 注解将 ListMap 注入到目标 bean 中。

示例如下:

@Component
public class MyComponentA implements MyComponentInterface {
    // ...
}

@Component
public class MyComponentB implements MyComponentInterface {
    // ...
}

@Component
public class MyComponentConsumer {
    @Autowired
    private List<MyComponentInterface> myComponentList;

    @Autowired
    private Map<String, MyComponentInterface> myComponentMap;
}

在使用上述解决方案时,需要根据实际情况选择最适合的方法,以确保正确地注入 bean。需要注意的是,选择不同的解决方案可能会影响代码的可读性和复杂度,因此需要在考虑各种因素的情况下选择最适合的方案。

🍁🍁 04、如何使用 @Autowired 注解注入集合类型的依赖?

要使用 @Autowired 注解注入集合类型的依赖,可以依赖注入一个集合(如 ListSetMap)来容纳匹配的 bean。

以下是几种常见的方式:

1. 使用 List 注入:

@Component
public class MyComponentConsumer {
    @Autowired
    private List<MyComponentInterface> myComponentList;
}

在上面的示例中,所有实现了 MyComponentInterface 接口的 bean 都会被自动注入到 myComponentList 中。

2. 使用 Set 注入:

@Component
public class MyComponentConsumer {
    @Autowired
    private Set<MyComponentInterface> myComponentSet;
}

List 类似,所有实现了 MyComponentInterface 接口的 bean 都会被自动注入到 myComponentSet 中,但是 Set 不允许重复的元素。

3. 使用 Map 注入:

@Component
public class MyComponentConsumer {
    @Autowired
    private Map<String, MyComponentInterface> myComponentMap;
}

在上面的示例中,myComponentMap 将以 bean 的名称作为键,MyComponentInterface 实例作为值。可以通过键来访问和调用相应的 bean。

需要注意的是,确保集合的泛型参数正确,并且匹配的 bean 类型与集合的泛型参数相符。另外,如果没有匹配的 bean,集合将是空的而不是 null。

这三种方式都可以用于注入集合类型的依赖。它们之间的主要区别在于它们所使用的容器类型以及容器的特性。

  1. List 类型:使用 List 类型可以容纳多个元素,并且元素的顺序是按照它们被添加到集合中的顺序。如果需要按照固定的顺序处理多个实例,就可以使用 List 类型。

  2. Set 类型:使用 Set 类型可以容纳多个元素,但是集合中的元素是无序的并且不能重复。如果需要在一个集合中保证唯一性,可以使用 Set 类型。

  3. Map 类型:使用 Map 类型可以容纳多个键值对,并且键值对是唯一的。使用 Map 可以根据键值对应的键(例如 bean 的名称)来访问相应的实例,这种方式比较灵活。

在实际开发中,应根据具体的需求选择最适合的类型,以便更好地处理依赖注入的集合类型。

下面是一个表格,说明了使用 ListSetMap 类型注入集合类型依赖的主要区别:

区别ListSetMap
容器类型ListSetMap
元素顺序有序无序无序
元素唯一性元素可以重复元素不重复键值对的键不重复
访问方式按照索引访问无法用索引访问通过键访问
注入示例List<MyComponent>Set<MyComponent>Map<String, MyComponent>
示例代码`````````
@Autowiredprivate List<MyComponent> myComponentList;private Set<MyComponent> myComponentSet;private Map<String, MyComponent> myComponentMap;
````````````

希望以上的表格能够清晰地展示 ListSetMap 在注入集合类型依赖方面的区别。根据具体的需求,选择最合适的容器类型进行注入。

🍁🍁 05、是否可以在构造函数上同时使用 @Autowired 和 @Value 注解?

可以同时在构造函数上使用 @Autowired@Value 注解,但是需要注意一些细节。

首先,@Autowired 用于注入类的实例,而 @Value 用于注入配置属性值。因此,@Autowired@Value 同时使用时,需要保证它们分别注入不同类型的依赖。

另外需要注意的是,@Value 注解需要指定一个属性名称来获取对应的配置值,例如:

public class MyComponent {
    private String name;

    public MyComponent(@Value("${my.component.name}") String name) {
        this.name = name;
    }
    // ...
}

在这个例子中,@Value 注解通过 ${my.component.name} 从配置文件中获取 name 属性值并注入到构造函数中。

如果需要在构造函数中同时使用 @Autowired@Value 注解,可以在构造函数参数列表中按顺序声明 @Autowired@Value 注解,例如:

public class MyComponent {
    private AnotherComponent anotherComponent;
    private String name;

    public MyComponent(@Autowired AnotherComponent anotherComponent, @Value("${my.component.name}") String name) {
        this.anotherComponent = anotherComponent;
        this.name = name;
    }
    // ...
}

在这个例子中,@Autowired 注解注入了 AnotherComponent 的实例,@Value 注解注入了 name 属性值。

需要注意的是,使用 @Autowired@Value 同时注入多个属性时,建议按照属性类型或者名称进行排序,提高代码的可读性。使用 @Autowired@Value 注解时,最好注入不同类型的依赖,以确保注入顺序正确性。

🍁🍁 06、如何通过 @Autowired 注解注入通用的 ApplicationContext 对象?

要通过 @Autowired 注解注入通用的 ApplicationContext 对象,可以按照以下步骤进行操作:

1. 在类中添加 @Autowired 注解,并将 ApplicationContext 类型的属性声明为需要注入的字段。

@Autowired
private ApplicationContext applicationContext;

2. 确保你的项目中已经正确地配置了 Spring 上下文,并且已经将 ApplicationContext 注册为一个 Bean。

@Configuration
public class AppConfig {

    @Bean
    public ApplicationContext applicationContext() {
        // 返回适当的 ApplicationContext 实例
        // ...
    }
    
    // 其他配置
    // ...
}

3. 在目标类中使用注入的 ApplicationContext 对象进行需要的操作。

public class MyComponent {

    @Autowired
    private ApplicationContext applicationContext;

    public void doSomething() {
        // 使用 applicationContext 进行操作
        // ...
    }

    // ...
}

现在,当 MyComponent 类的实例被创建时,Spring 容器会自动注入 ApplicationContext 对象,并使其可用。

请确保你的 Spring 配置正确,并且 ApplicationContext 已经正确注册为一个 Bean。同时,也要确保目标类已被 Spring 所管理(例如通过 @Component 注解或 XML 配置等方式进行管理)。

记住,虽然可以通过 ApplicationContext 进行各种操作,但应首先仔细考虑依赖注入的最佳实践,并尽量避免直接访问 Spring 上下文对象。依赖注入是一种更好的设计模式,可以提高代码的可测试性和可维护性。

🍁🍁 07、@Autowired 注解的 required 属性的作用是什么?

@Autowired 注解的 required 属性用于指定所注入的依赖是否为必需的。默认情况下,@Autowired 注解的 required 属性为 true,即所注入的依赖是必需的。当这个依赖不能被注入时,Spring 将会抛出一个异常。

例如,如果没有正确配置所依赖的对象或者没有匹配的候选对象,则会抛出 org.springframework.beans.factory.NoSuchBeanDefinitionException 异常。

@Autowired(required = true)
private MyDependency myDependency;

在上面的示例中,MyDependency 对象是必需的。如果没有找到匹配的对象,则会抛出异常。

如果将 required 属性设置为 false,则所注入的对象不是必需的。当没有找到匹配的对象时,Spring 只会不进行依赖注入,而不会抛出异常。

@Autowired(required = false)
private MyDependency myDependency;

这在编写可选依赖关系的代码时非常有用,例如,可以使用 @Autowired(required = false) 将 ORM 框架或存储库作为可选依赖注入到 Spring 组件中,而不必强制要求具有相关的依赖项。

需要注意的是,在使用 required 属性时,应当牢记依赖的正确性,确保不能发生未明确处理的空指针异常等问题。

总之,通过控制 required 属性,可以实现对所注入依赖的精细处理,避免在依赖项未正确配置时出现不愉快的运行时异常。

🍁🍁 08、如何自定义一个注解来替代 @Autowired 注解的功能?

要自定义一个注解来替代 @Autowired 注解的功能,可以按照以下步骤进行操作:

1. 创建一个自定义注解, 使用 @Retention(RetentionPolicy.RUNTIME) 保留策略和 @Target(ElementType.FIELD) 目标元素类型。

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface MyCustomAutowired {
}

2. 在需要进行依赖注入的字段上使用自定义注解。

public class MyClass {
    @MyCustomAutowired
    private MyDependency myDependency;

    // ...
}

3. 创建一个后置处理器类, 实现 BeanPostProcessor 接口,用于处理标记有自定义注解的字段。

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;
import org.springframework.util.ReflectionUtils;

import java.lang.reflect.Field;

public class MyCustomAutowiredProcessor implements BeanPostProcessor {

    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        Field[] fields = bean.getClass().getDeclaredFields();
        for (Field field : fields) {
            if (field.isAnnotationPresent(MyCustomAutowired.class)) {
                Object dependency = // 根据自定义注解的逻辑获取依赖对象

                // 设置字段可访问性以允许注入
                ReflectionUtils.makeAccessible(field);
                // 为字段注入依赖对象
                ReflectionUtils.setField(field, bean, dependency);
            }
        }
        return bean;
    }

    // 可选择实现 postProcessAfterInitialization() 方法进行其他处理

}

4. 将后置处理器类注册到 Spring 上下文中。

可以通过 XML 配置或者在配置类中使用 @Bean 注解进行注册。

@Configuration
public class AppConfig {

    @Bean
    public MyCustomAutowiredProcessor myCustomAutowiredProcessor() {
        return new MyCustomAutowiredProcessor();
    }

    // ...

}

完成以上步骤后,当 Spring 容器初始化时,后置处理器将会检查标记有 @MyCustomAutowired 注解的字段,并根据自定义的逻辑进行依赖注入操作。

请注意,这只是一个简化的示例,实际情况下你可能需要根据自己的需求对后置处理器进行修改和扩展。

同时,还需确保在 Spring 配置中正确设置了 MyClassMyDependency 的相关定义,让它们被 Spring 所管理。

自定义注解能够帮助我们更好地扩展和控制依赖注入的行为,但在使用自定义注解时,同样要考虑代码的可读性和可维护性,以及与团队的共识和约定一致。

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1342887.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

CSS 缩减顶部

<template><!-- mouseenter"startAnimation" 表示在鼠标进入元素时触发 startAnimation 方法。mouseleave"stopAnimation" 表示在鼠标离开元素时触发 stopAnimation 方法。 --><!-- 容器元素 --><div class"container" mou…

Java InputStream OutputStream

OutputStream 轉 InputStream Case 1 toByteArray ByteArrayOutputStream out new ByteArrayOutputStream(); new ByteArrayInputStream(out.toByteArray()) Case2 從Output字節讀取Input ByteArrayOutputStream out new ByteArrayOutputStream();byte[] bs new byte[…

[足式机器人]Part2 Dr. CAN学习笔记-自动控制原理Ch1-6根轨迹Root locus

本文仅供学习使用 本文参考&#xff1a; B站&#xff1a;DR_CAN Dr. CAN学习笔记-自动控制原理Ch1-6根轨迹Root locus 1. 根的作用2. 手绘技巧3. 分离点/汇合点&根轨迹的几何性质 1. 根的作用 G ( s ) s 3 s 2 2 s 4 G\left( s \right) \frac{s3}{s^22s4} G(s)s22s4s3​…

Mysql使用Mybatis进行时间操作

MySQL中支持以下日期时间类型&#xff1a; DATE&#xff1a;存储年月日&#xff08;例如&#xff1a;2023-04-05&#xff09;。范围从"1000-01-01"到"9999-12-31" TIME&#xff1a;存储时分秒&#xff08;例如&#xff1a;11:22:22&#xff09;。范围从&q…

如何使用Spoofy检测目标域名是否存在欺骗攻击风险

关于Spoofy Spoofy是一款功能强大的域名安全检测工具&#xff0c;在该工具的帮助下&#xff0c;广大研究人员可以轻松检测单个目标域名或域名列表中的域名是否存在遭受欺诈攻击的风险。 该工具基于纯Python开发&#xff0c;可以根据SPF和DMARC记录来检测和判断目标域名是否可…

DS18B20温度检测及其液晶显示

#include<reg51.h> //包含单片机寄存器的头文件 #include<intrins.h> //包含_nop_()函数定义的头文件 unsigned char code digit[10]{"0123456789"}; //定义字符数组显示数字 unsigned char code Str[]{"Test by DS18B20"}; //说明…

【C++干货铺】STL中set和map的介绍和使用

个人主页点击直达&#xff1a;小白不是程序媛 C系列专栏&#xff1a;C干货铺 代码仓库&#xff1a;Gitee 目录 序列式容器 关联式容器 键值对 树形结构的关联式容器 set set的介绍 set的使用 set的模板参数列表 set的构造 ​编辑 set的容量 set的删除和查找 mult…

安装ps提示msvcr71.dll丢失的解决方法,全面解析dll问题

当您在安装PS软件时遇到msvcr71.dll丢失的问题&#xff0c;这是因为该文件是某些程序运行必需的。msvcr71.dll主要包含了C运行时库的函数&#xff0c;这些函数主要用于处理字符串、数学运算、内存管理等基本操作。例如&#xff0c;我们在编写程序时&#xff0c;需要对字符串进行…

轻松实现iphone截图传电脑

目录 摘要 引言 用户登录工具和连接设备 生成截图 摘要 本篇博文介绍了克魔助手这款工具&#xff0c;解决了iPhone与Windows系统下图片传输的烦恼。通过连接同一Wi-Fi&#xff0c;使用克魔助手轻松实现了iPhone截图传输到电脑上的便捷操作。用户只需简单地下载并安装克魔助…

跨境外贸获客该怎么做?请掌握这些技巧!

在当今全球化的商业环境中&#xff0c;跨境外贸已经成为许多企业寻求增长和拓展市场的关键途径&#xff0c;然而&#xff0c;与国内市场不同&#xff0c;跨境外贸需要面对语言、文化、法规等多方面的挑战。 如何有效地获取海外客户&#xff0c;成为每个涉足跨境外贸的企业必须…

LabVIEW在电机噪声与振动探测的应用

LabVIEW在电机噪声与振动探测的应用 硬件部分是电机噪声和振动测试分析系统的基础&#xff0c;主要由三大核心组件构成&#xff1a;高灵敏度振动传感器、先进的信号调理电路和高性能数据采集卡。这些设备协同工作&#xff0c;确保了从电机捕获的噪声和振动信号的准确性和可靠性…

分布式锁常见问题及其解决方案

一、为什么要使用分布式锁&#xff1f; 因为在集群下&#xff0c;相当于多个JVM&#xff0c;就相当于多个锁&#xff0c;集群之间锁是没有关联的&#xff0c;会照成锁失效从而导致线程安全问题 分布式锁可以分别通过MySQL、Redis、Zookeeper来进行实现 二、redis分布式锁的实…

PC9094可调电流限制OVP过压过流保护IC超小体积封装

概述&#xff1a; PC9094过电压和过电流保护该器件具有低80mΩ&#xff08;TYP&#xff09;导通电阻集成MOSFET&#xff0c;主动保护低电压 系统的电压供应故障高达29V直流电。输入电压超过过电压阈值将导致内部MOSFET关闭&#xff0c;防止 损坏下游的过大电压设备。过电压保…

鼠标悬浮时光标变成手势

鼠标悬浮变成光标是因为该组件没有添加 style"cursor: pointer"

旋转矩形问题

问题&#xff1a;判断两个旋转矩形是否重叠&#xff08;相交和包含&#xff09; 矩形的坐标是旋转前的坐标&#xff1a; 矩形A(left1,top1,width1,height1,angle1) 矩形B(left2,top2,width2,height2,angle2) 方法1&#xff1a;碰撞检测判断相交 点在多边形内部判断包含 遍历…

单片机开发从小工到专家

有道无术&#xff0c;术尚可求&#xff1b;有术无道&#xff0c;止于术 背景 向单片机嵌入式开发小伙伴推荐了几本书&#xff0c;阅读量破10 1. 适用范围 2. 书籍推荐 书籍推荐 3. 大师介绍 大师介绍 4. 大师书籍编写逻辑 25年大师出版的关于&#xff1a;嵌入式单片…

JAVA中的回调函数

回调函数的基本概念&#xff1a; 回调函数是一种常见的编程模式&#xff0c;也称为回调机制。回调函数是一种特殊的函数&#xff0c;它允许将一段代码作为参数传递给另一个方法&#xff0c;并在需要时调用。回调函数通常用于异步编程或事件处理&#xff0c;可以将程序的控制权…

NFC物联网智能购物车设计方案

智能购物车是综合利用计算机网络、射频识别技术、数据库技术、单片机于一体的设备具有先进性、便于管理性、经济性、普适性。基于NFC (Near Field Communication&#xff0c;近场通信)技术的智能购物车&#xff0c;能够大幅缩短结账排队时间&#xff0c;实现“无感支付”。NFC是…

对SPI总线上挂接多个X5045的读写操作

#include<reg51.h> //包含单片机寄存器的头文件 #include<intrins.h> //包含_nop_()函数定义的头文件 sbit SCKP3^4; //将SCK位定义为P3.4引脚 sbit SIP3^5; //将SI位定义为P3.5引脚 sbit SOP3^6; //将SO位定义为P3.6引脚 sbit CS1P3^7; …

【js控制页面的模糊程度】【lenis禁止页面滚动】

文章目录 前言一、效果图二、使用步骤1.下载studio-freight/lenis2.使用studio-freight/lenis 三、下载 gsap在编写页面动画1. 下载gsap2.引入gsap3.调用gsap的方法&#xff0c;让页面模糊 总结 前言 在项目中&#xff0c;我们经常会遇到弹窗功能&#xff0c;当弹框弹出时&…