0

0

如何在c++中实现一个简单的依赖注入容器? (IoC设计模式)

冰火之心

冰火之心

发布时间:2026-01-11 11:28:02

|

227人浏览过

|

来源于php中文网

原创

最小可用C++依赖注入容器需管理类型生命周期并解析构造依赖链,通过模板+类型擦除(std::function/std::any)绕过无RTTI限制,支持单例/瞬态模式及循环依赖检测。

如何在c++中实现一个简单的依赖注入容器? (ioc设计模式)

依赖注入容器的核心要做什么

一个最小可用的 C++ 依赖注入容器,本质是管理类型生命周期 + 解决构造依赖链。它不一定要支持 XML 配置或反射,关键是能用代码注册类型、按需创建实例,并自动把依赖(比如 ServiceA 依赖 Repository)传进去。

难点不在“注入”本身,而在如何绕过 C++ 没有运行时类型信息(RTTI)和原生反射的限制——所以得靠模板 + 类型擦除 + 工厂函数组合实现。

std::function + std::any 存储工厂

不能直接存裸指针或 std::unique_ptr,因为类型 T 在注册时未知。需要用类型擦除手段统一接口:

  • 注册时传入一个 std::function<:unique_ptr>()> 工厂,返回 void* 级别指针,靠调用方自己 static_cast
  • 或者更安全一点:用 std::any 包裹工厂,再用模板封装注册接口,把类型擦除细节藏起来
  • 避免用 dynamic_cast 或 RTTI 判断类型,性能差且不可靠;std::anytype() 可用于运行时校验,但不是主要逻辑分支
template
void register_type() {
    factories_[typeid(T)] = []() -> std::unique_ptr {
        return std::make_unique();
    };
}

解决构造参数依赖:递归解析模板参数

如果 ServiceB 构造函数是 ServiceB(std::shared_ptr, std::shared_ptr),容器就得在创建时自动提供这两个依赖。C++ 没法在运行时读取构造函数签名,所以必须靠编译期推导:

Seele AI
Seele AI

3D虚拟游戏生成平台

下载

立即学习C++免费学习笔记(深入)”;

  • 用可变参数模板 + std::make_shared 组合,逐个解析 T 的构造函数参数类型
  • 每个参数类型都调用一次 resolve(),递归获取实例
  • 注意循环依赖:两个类互相 std::shared_ptr 构造对方 → 必须检测 resolving_ 集合,抛异常或返回 nullptr
  • 不支持原始指针或值类型参数(如 int port),除非显式绑定常量
template
std::shared_ptr resolve() {
    if (resolving_.count(typeid(T))) {
        throw std::runtime_error("circular dependency for " + std::string(typeid(T).name()));
    }
    resolving_.insert(typeid(T));
    auto ptr = std::shared_ptr(static_cast(create_instance().release()));
    resolving_.erase(typeid(T));
    return ptr;
}

生命周期管理:singleton vs transient

默认每次 resolve() 都新建实例(transient),但多数服务需要单例(singleton)。关键不是“全局唯一”,而是“容器内唯一”:

  • std::unordered_map<:type_info const std::shared_ptr>> 缓存已创建的 singleton 实例
  • 注册时加标记:register_singleton(),内部走缓存分支;register_transient() 每次 new
  • 注意 std::shared_ptr 无法直接转型,得用 std::static_pointer_cast,所以缓存前要先转成具体类型再存为 void 指针
  • 析构顺序难控制:singleton 实例随容器销毁,但若其他静态对象依赖它,可能触发 use-after-free —— 所以建议所有依赖都通过 resolve() 获取,不要存裸指针

真正麻烦的不是写出来,而是当 resolve() 报错时,堆里全是模板展开层,错误信息里连哪个参数没注册都看不清。调试时得靠日志打点 + 类型名字符串化辅助定位。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1463

2023.10.24

pdf怎么转换成xml格式
pdf怎么转换成xml格式

将 pdf 转换为 xml 的方法:1. 使用在线转换器;2. 使用桌面软件(如 adobe acrobat、itext);3. 使用命令行工具(如 pdftoxml)。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1870

2024.04.01

xml怎么变成word
xml怎么变成word

步骤:1. 导入 xml 文件;2. 选择 xml 结构;3. 映射 xml 元素到 word 元素;4. 生成 word 文档。提示:确保 xml 文件结构良好,并预览 word 文档以验证转换是否成功。想了解更多xml的相关内容,可以阅读本专题下面的文章。

2084

2024.08.01

xml是什么格式的文件
xml是什么格式的文件

xml是一种纯文本格式的文件。xml指的是可扩展标记语言,标准通用标记语言的子集,是一种用于标记电子文件使其具有结构性的标记语言。想了解更多相关的内容,可阅读本专题下面的相关文章。

973

2024.11.28

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

520

2023.09.20

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

253

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

206

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1463

2023.10.24

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

78

2026.01.09

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

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

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