返回顶部
首页 > 资讯 > 后端开发 > Python >一文详解Spring构造函数推断
  • 881
分享到

一文详解Spring构造函数推断

Spring构造函数推断Spring构造函数 2023-05-14 20:05:32 881人浏览 安东尼

Python 官方文档:入门教程 => 点击学习

摘要

目录正文自动注入底层原理构造函数推断的限制源码解析结论正文 spring 提供了一组基本的功能,例如依赖注入(DI)和面向切面编程(aop)。其中一个非常强大的功能是构造函数自动注

正文

spring 提供了一组基本的功能,例如依赖注入(DI)和面向切面编程aop)。其中一个非常强大的功能是构造函数自动注入,也称为构造函数推断。在本文中,我们将深入探讨Spring构造函数推断的底层原理,并解释Spring是如何实现它的。

自动注入

构造函数自动注入是指 Spring 自动解析 bean 的构造函数参数,并将它们传递给相应的构造函数。这样,我们就不必显式地在XML或Java配置文件中指定每个 bean 的构造函数参数。这是一个非常方便的功能,特别是在有许多 bean 需要注入的情况下。

底层原理

Spring 使用 Java 反射机制来执行构造函数推断。当 Spring 需要实例化一个 bean 时,它将首先使用 Java 反射机制检查该 bean 的构造函数。

然后,它将检查每个构造函数的参数类型,并尝试在 Spring 应用程序上下文中查找与参数类型匹配的 bean。

如果找到匹配的 bean,则 Spring 将使用该 bean 实例化构造函数参数。如果找不到匹配的 bean,则 Spring 将继续检查下一个构造函数。

如果没有任何构造函数可以匹配,则 Spring 将抛出一个异常。但是,可以使用 @Autowired(required=false) 注解来取消必需的依赖关系,这意味着如果找不到与构造函数参数类型匹配的bean,则 Spring 将不会抛出异常。

以下是一个简单的例子,说明如何在 Spring 中使用构造函数推断:

public class MyBean {
    private final MyDependency myDependency;
    public MyBean(MyDependency myDependency) {
        this.myDependency = myDependency;
    }
    public void doSomething() {
        myDependency.doSomething();
    }
}

在这个例子中,MyBean依赖于MyDependency。Spring 将使用构造函数推断自动注入 MyDependency,而不需要在XML或Java配置文件中显式指定。

@Configuration
public class AppConfig {
    @Bean
    public MyDependency myDependency() {
        return new MyDependency();
    }
    @Bean
    public MyBean myBean() {
        return new MyBean(myDependency());
    }
}

在这个例子中,AppConfig 类使用 @Configuration 注解指定一个 Spring 应用程序上下文,并定义两个bean:MyDependency 和 MyBean。在 MyBean 的构造函数中,我们将 MyDependency 作为参数传递,Spring 将使用构造函数推断自动注入 MyDependency。

构造函数推断的限制

构造函数推断有一些限制,例如:

  • 如果有多个构造函数可以匹配,则 Spring 将抛出一个异常。在这种情况下,您需要使用 @Qualifier 注解来指定要注入的 bean。

  • 如果构造函数参数是原始类型(例如int、float、boolean等),则 Spring 无法推断它们。在这种情况下,您需要使用 @Value 注解将值直接注入到构造函数参数中。

  • 如果构造函数参数是数组集合,则 Spring 无法推断它们。在这种情况下,您需要使用 @Qualifier 注解和 @Autowired 的 List 或 Set 类型的字段来手动注入数组或集合。

源码解析

Spring 使用 BeanWrapperImpl 类来执行构造函数推断。BeanWrapperImpl 类是 Spring 的核心类之一,它提供了一种方便的方式来对 Java 对象进行包装和访问。以下是 Spring 中用于执行构造函数推断的 BeanWrapperImpl 类的源代码:

public class BeanWrapperImpl extends AbstractPropertyAccessor implements BeanWrapper {
    // ...
    @Override
    public Object createInstance() throws BeansException {
        // 获取Bean的构造函数
        Constructor<?> constructorToUse = this.constructorResolver.autowireConstructor;
        if (constructorToUse == null) {
            throw new BeansException("No default constructor found");
        }
        try {
            // 使用构造函数创建Bean实例,并返回
            return constructorToUse.newInstance(getConstructorArgumentValues(constructorToUse), getBeanClassLoader());
        }
        catch (Throwable ex) {
            throw new BeanCreationException("Could not instantiate bean", ex);
        }
    }
    // ...
    private Constructor<?> autowireConstructor;
    // ...
    public BeanWrapperImpl(Class<?> clazz) {
        super();
        this.objectType = clazz;
        this.propertyEditorReGIStry = new SimpleTypeConverter();
        this.constructorResolver = new ConstructorResolver(this.objectType, this);
    }
    // ...
    private class ConstructorResolver {
        // ...
        public ConstructorResolver(Class<?> clazz, BeanWrapperImpl bw) {
            // 查找可以使用的构造函数并将其缓存
            Constructor<?>[] candidates = clazz.getDeclaredConstructors();
            Constructor<?> autowireCandidate = null;
            int numAutowireCandidates = 0;
            for (Constructor<?> candidate : candidates) {
                if (candidate.isAnnotationPresent(Autowired.class)) {
                    autowireCandidate = candidate;
                    numAutowireCandidates++;
                }
            }
            if (numAutowireCandidates == 1) {
                this.autowireConstructor = autowireCandidate;
            }
            else if (numAutowireCandidates > 1) {
                throw new BeansException("Multiple autowire constructors found");
            }
        }
        // ...
    }
    // ...
}

在BeanWrapperImpl 类中,构造函数推断是在 createInstance() 方法中执行的。该方法首先获取与Bean匹配的构造函数(由constructorResolver.autowireConstructor决定),然后使用该构造函数创建 Bean 实例。

ConstructorResolver 内部类中,构造函数推断是通过查找带有 @Autowired 注解的构造函数来实现的。如果找到了一个带有 Autowired注解 的构造函数,则它将被缓存到 autowireConstructor 字段中,并在 createInstance() 方法中使用。

结论

构造函数自动注入是 Spring 的一个非常强大的功能,它可以大大简化 bean 的配置和管理。我们深入探讨了 Spring 构造函数推断的底层原理,并解释了 Spring 是如何实现它的,还提供了一个简单的例子和源代码解析,以帮助大家更好地了解构造函数推断的工作方式。

以上就是一文详解Spring构造函数推断的详细内容,更多关于Spring 构造函数推断的资料请关注编程网其它相关文章!

--结束END--

本文标题: 一文详解Spring构造函数推断

本文链接: https://lsjlt.com/news/204661.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 一文详解Spring构造函数推断
    目录正文自动注入底层原理构造函数推断的限制源码解析结论正文 Spring 提供了一组基本的功能,例如依赖注入(DI)和面向切面编程(AOP)。其中一个非常强大的功能是构造函数自动注...
    99+
    2023-05-14
    Spring 构造函数推断 Spring 构造函数
  • Spring构造函数推断的原理是什么
    这篇文章主要介绍了Spring构造函数推断的原理是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Spring构造函数推断的原理是什么文章都会有所收获,下面我们一起来看看吧。正文Spring 提供了一组基本的...
    99+
    2023-07-05
  • Spring源码解析之推断构造方法
    Spring推断构造方法 贴个测试代码直接开干,这只是个样例,其他情况自行分析 @Component public class OrderService { public ...
    99+
    2024-04-02
  • C++:构造函数,析构函数详解
    目录前言一、面向对象二、构造函数1.基本概念2.构造函数重载1.构造函数分类2.有参构造函数:3.有参构造函数3个调用规则:4.拷贝构造函数5.析构函数总结前言 上期了解C++类中有...
    99+
    2024-04-02
  • C#构造函数详解
    一、简介 构造函数,基本用法是在类对象声明的时候完成初始化工作。 二、实例构造函数 1、构造函数的名字与类名相同。 2、使用 new 表达式创建类的对象或者结构(例如int)时,会调...
    99+
    2024-04-02
  • 详解C++构造函数
    目录1.作用2.代码举例2.1 示例1:2.2 示例2:3. 使用3.1 使用构造函数初始化3.2 有参数的构造函数3.3 默认的构造函数4. 成员初始化列表例1:正常初始化例2:成...
    99+
    2024-04-02
  • C++构造函数详解
    文章转自公众号:Coder梁(ID:Coder_LT) 上一篇文章我们介绍了定义了类,在使用之前,往往还需要对类进行初始化。这篇介绍的就是对类进行初始化的方法。 像是结构体,我们可以...
    99+
    2024-04-02
  • C++中构造函数详解
    构造函数按参数为为:有参构造函数和无参构造函数 按类型分为:普通构造函数和拷贝构造函数 构造函数的三种调用方法:括号法,显示法,隐式转换法; //括号法 Person p1; ...
    99+
    2024-04-02
  • 无法推断函数构造函数的泛型参数中的类型
    php小编柚子将为大家解答一个常见的问题:在构造函数的泛型参数中,无法推断出类型的原因。在PHP中,泛型参数的类型推断是通过参数的默认值来实现的。然而,在构造函数中,由于参数是在对象创...
    99+
    2024-02-09
    编译错误
  • Spring源码解析之推断构造方法的示例分析
    小编给大家分享一下Spring源码解析之推断构造方法的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!Spring推断构造方法贴个测试代码直接开干,这只是个...
    99+
    2023-06-15
  • C++中的构造函数详解
    目录普通变量的初始化构造函数一定会生成默认构造函数吗?防止隐式类型转换赋值与初始化的区别对象的计数成员初始化的顺序类的引用成员构造函数使用注意事项参考总结普通变量的初始化 当我们在定...
    99+
    2024-04-02
  • JavaScript构造函数举例详解
    典型的面向对象编程语言(比如C++和Java),存在“类”(class)这个概念。所谓“类”就是对象的模板,对象就是“类&...
    99+
    2023-05-14
    js构造函数的作用 js构造函数定义方法 js构造函数实例
  • c++特殊构造函数详解
    目录前言拷贝构造函数一、什么是拷贝构造函数二、调用时机注意浅拷贝和深拷贝总结前言 众所周知,构造函数的作用是类在创建对象时的初始化,而拷贝构造函数则是构造函数里的一种特殊构造。 拷贝...
    99+
    2024-04-02
  • C++超详细讲解构造函数
    目录类的6个默认成员函数构造函数特性编译器生成的默认构造函数成员变量的命名风格类的6个默认成员函数 如果我们写了一个类,这个类我们只写了成员变量没有定义成员函数,那么这个类中就没有函...
    99+
    2024-04-02
  • Python 含参构造函数实例详解
    本篇博文主要介绍在Python3中如何构造含参构造函数 样例如下: class MyOdlHttp: username = '' password = '' def __init__(se...
    99+
    2022-06-04
    详解 函数 实例
  • Android自定义View构造函数详解
    初始Custom View的构造函数 之前写过一篇实现圆形进度条的博客(自定义圆形进度条),通常我们在实现Custom View的时候,都会先继承View并实现View的三个构...
    99+
    2022-06-06
    view 函数 Android
  • C++中的复制构造函数详解
    目录复制构造函数复制构造函数的三种调用复制构造函数的禁用深拷贝与浅拷贝一定会生成默认复制构造函数吗?参考 总结普通变量的复制 有时我们会在定义一个变量的同时使用另一个变量来初始化它。...
    99+
    2024-04-02
  • C++中的拷贝构造函数详解
    目录C++拷贝构造函数(复制构造函数)详解1) 为什么必须是当前类的引用呢?2) 为什么是 const 引用呢?默认拷贝构造函数总结C++拷贝构造函数(复制构造函数)详解 拷贝和复制...
    99+
    2024-04-02
  • JAVA构造函数(方法)示例详解
    目录 一、什么是构造函数二、构造函数的特点三、示例四、默认构造函数五、构造函数的重载六、构造函数的使用构造函数的特点总结 一、什么是构造函数 Java构造函数,也叫构造方法,是JAV...
    99+
    2024-04-02
  • C++类与对象及构造函数析构函数基础详解
    目录C++类与对象类的定义对象的创建构造函数和析构函数访问修饰符继承多态成员变量与成员方法总结C++类与对象 C++是一门面向对象的编程语言。在C++中,我们可以利用类来创建对象,...
    99+
    2023-05-16
    C++类对象函数 c++ 构造析构函数
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作