0

0

如何用反射实现依赖注入 动态创建和组装对象实例

P粉602998670

P粉602998670

发布时间:2025-08-16 18:07:01

|

847人浏览过

|

来源于php中文网

原创

依赖注入的核心思想是将对象创建和依赖管理交由外部容器处理,通过反射机制在运行时动态创建对象并解析构造函数参数,利用类型映射实现接口与实现的绑定,结合递归解析和缓存机制完成实例的自动组装,同时通过维护解析栈防止循环依赖,最终实现一个支持transient和singleton生命周期的简易di容器,尽管功能简单,但完整展现了主流di框架的核心原理。

如何用反射实现依赖注入 动态创建和组装对象实例

依赖注入(DI)的核心思想是将对象的创建和依赖关系的管理从代码中剥离出来,交由外部容器处理。利用反射机制,我们可以在运行时动态创建对象、解析依赖并自动组装实例,从而实现一个简易的依赖注入容器。以下是实现的关键思路和步骤。


一、通过反射动态创建对象

在没有依赖的情况下,创建对象很简单:

var instance = new MyClass();

但使用反射,我们可以在运行时根据类型信息创建实例:

Type type = typeof(MyClass);
object instance = Activator.CreateInstance(type);

更进一步,如果构造函数有参数,反射也能处理:

ConstructorInfo ctor = type.GetConstructor(new[] { typeof(IService) });
object[] args = { /* 已创建的依赖实例 */ };
object instance = ctor.Invoke(args);

二、自动解析构造函数依赖

大多数依赖注入框架优先使用构造函数注入。我们可以用反射获取类型最长的构造函数(通常包含最多依赖),然后递归解析其参数类型。

public object CreateInstance(Type type)
{
    // 查找最长的构造函数(最常用的注入方式)
    ConstructorInfo ctor = type.GetConstructors()
        .OrderByDescending(c => c.GetParameters().Length)
        .First();

    ParameterInfo[] parameters = ctor.GetParameters();

    // 递归解析每个参数(依赖)
    object[] args = parameters.Select(p =>
    {
        return Resolve(p.ParameterType); // 递归获取依赖实例
    }).ToArray();

    return ctor.Invoke(args);
}

Resolve
方法是核心:它检查当前类型是否已注册,若未实例化则调用
CreateInstance
创建。


三、维护一个类型映射和实例缓存

为了支持接口到实现类的映射(如

IService
ServiceA
),我们需要一个注册表:

Bika.ai
Bika.ai

打造您的AI智能体员工团队

下载
Dictionary _registrations = new();
Dictionary _singletons = new();

注册示例:

public void Register() where TypeTo : TypeFrom
{
    _registrations[typeof(TypeFrom)] = typeof(TypeTo);
}

支持生命周期管理:

  • Transient:每次创建新实例
  • Singleton:首次创建后缓存实例
public object Resolve(Type serviceType)
{
    // 如果是单例且已创建,直接返回
    if (_singletons.ContainsKey(serviceType))
        return _singletons[serviceType];

    Type implType = _registrations.ContainsKey(serviceType)
        ? _registrations[serviceType]
        : serviceType;

    object instance = CreateInstance(implType);

    // 若是单例,缓存
    if (/* 是单例注册 */)
        _singletons[serviceType] = instance;

    return instance;
}

四、处理循环依赖

反射 + 递归容易在循环依赖时导致栈溢出,比如 A 依赖 B,B 又依赖 A。

简单防范策略:

  • Resolve
    过程中维护一个“正在解析”的类型栈
  • 每次开始解析前检查是否已存在,若存在则抛出异常
HashSet _resolving = new();

public object Resolve(Type serviceType)
{
    if (_resolving.Contains(serviceType))
        throw new InvalidOperationException($"Circular dependency detected: {serviceType}");

    _resolving.Add(serviceType);

    // ...解析逻辑...

    _resolving.Remove(serviceType); // 完成后移除
    return instance;
}

五、完整简化示例(C#)

public class SimpleDIContainer
{
    private Dictionary _registrations = new();
    private Dictionary _singletons = new();
    private HashSet _resolving = new();

    public void Register() where TTo : TFrom
    {
        _registrations[typeof(TFrom)] = typeof(TTo);
    }

    public void RegisterSingleton() where TTo : TFrom
    {
        Register();
        // 提前标记为单例(或首次创建时缓存)
    }

    public T Resolve()
    {
        return (T)Resolve(typeof(T));
    }

    public object Resolve(Type serviceType)
    {
        if (_singletons.ContainsKey(serviceType))
            return _singletons[serviceType];

        if (_resolving.Contains(serviceType))
            throw new Exception($"Circular dependency: {serviceType}");

        _resolving.Add(serviceType);

        try
        {
            Type implType = _registrations.ContainsKey(serviceType)
                ? _registrations[serviceType]
                : serviceType;

            var ctor = implType.GetConstructors()
                .OrderByDescending(c => c.GetParameters().Length)
                .First();

            var parameters = ctor.GetParameters();
            var args = parameters.Select(p => Resolve(p.ParameterType)).ToArray();

            var instance = ctor.Invoke(args);

            if (_registrations.ContainsValue(implType) && /* 注册为 singleton */)
            {
                _singletons[serviceType] = instance;
            }

            return instance;
        }
        finally
        {
            _resolving.Remove(serviceType);
        }
    }
}

使用方式:

var container = new SimpleDIContainer();
container.Register();
container.Register();

var service = container.Resolve(); // 自动组装依赖

基本上就这些。虽然没有主流框架(如 Autofac、Microsoft.Extensions.DependencyInjection)那样高效和健壮,但它展示了反射 + 递归解析 + 类型映射是如何实现依赖注入的核心机制的。理解这些原理,有助于更好地使用现代 DI 框架,也能在特定场景下实现轻量级容器。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1128

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

213

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1685

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

20

2026.01.19

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

397

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

575

2023.08.10

vsd文件打开方法
vsd文件打开方法

vsd文件打开方法有使用Microsoft Visio软件、使用Microsoft Visio查看器、转换为其他格式等。想了解更多vsd文件相关内容,可以阅读本专题下面的文章。

488

2023.10.30

Golang 网络安全与加密实战
Golang 网络安全与加密实战

本专题系统讲解 Golang 在网络安全与加密技术中的应用,包括对称加密与非对称加密(AES、RSA)、哈希与数字签名、JWT身份认证、SSL/TLS 安全通信、常见网络攻击防范(如SQL注入、XSS、CSRF)及其防护措施。通过实战案例,帮助学习者掌握 如何使用 Go 语言保障网络通信的安全性,保护用户数据与隐私。

2

2026.01.29

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

446

2026.01.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号