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

Any类(C++17类型擦除,也称上帝类)

Any类(C++17类型擦除,也称上帝类)

在C++中,std::any 是C++17标准引入的一个灵活的类型安全容器,用于存储任意类型的单个值。


1. std::any 的核心特性

  • 类型安全:存储的值必须通过明确的类型转换(any_cast)访问,否则会抛出异常。
  • 值语义any对象管理其内部存储的值的生命周期,拷贝时会深拷贝存储的值。
  • 小对象优化(SOO):大多数实现会在栈上直接存储小对象,避免堆分配的开销。
  • 类型擦除:隐藏具体类型,但保留运行时类型信息(RTTI)。

2. 基本用法

2.1 包含头文件
#include <any>
2.2 存储值
std::any data;data = 42;                     // 存储int
data = std::string("Hello");   // 存储string
data = 3.14;                   // 存储double
2.3 检查是否有值
if (data.has_value()) {std::cout << "Data contains a value.\n";
}
2.4 获取值的类型
const std::type_info& type = data.type();
std::cout << "Type: " << type.name() << "\n";  // 输出类似"i"(int)或"d"(double)
2.5 安全地获取值(any_cast
try {int value = std::any_cast<int>(data);std::cout << "Value: " << value << "\n";
} catch (const std::bad_any_cast& e) {std::cerr << "Wrong type: " << e.what() << "\n";
}
2.6 重置值
data.reset();  // 清空内容

3. 高级用法

3.1 存储复杂对象
struct Point { int x, y; };
std::any pt = Point{10, 20};// 通过指针访问(不抛出异常)
if (auto* p = std::any_cast<Point>(&pt)) {std::cout << "Point: (" << p->x << ", " << p->y << ")\n";
}
3.2 移动语义
std::string str = "Hello";
std::any a = std::move(str);  // 移动构造,str变为空
3.3 自定义类型支持

任何可拷贝构造的类型均可存储:

class MyClass {
public:MyClass(int v) : value(v) {}
private:int value;
};std::any obj = MyClass(42);

4. 与类似技术的对比

特性std::anystd::variantvoid*
类型安全✔️✔️
存储类型数量任意编译时固定任意
类型检查运行时(type()编译时
内存管理自动自动手动
适用场景动态类型需求已知有限类型集合低级操作

5. 实现原理简析

  • 类型擦除:通过内部模板类保存类型信息和值。
  • 小对象优化:栈存储小对象,堆存储大对象。
  • 虚函数分发:使用虚函数处理拷贝、析构和类型查询。

6. 注意事项

  • 性能:频繁的类型检查或大型对象存储可能影响性能。
  • 异常安全:错误的any_cast会抛出std::bad_any_cast
  • 类型一致性:必须精确匹配存储类型(如intlong视为不同)。

7. 实际应用场景

  • 配置系统:存储不同类型的配置参数(如int、string、bool)。
  • 消息传递:在事件总线上传递多种类型的消息。
  • 插件架构:处理来自不同模块的未知数据类型。

8. 示例代码:异构容器

#include <any>
#include <vector>
#include <iostream>int main() {std::vector<std::any> container;container.push_back(42);container.push_back(3.14);container.push_back(std::string("C++17"));for (const auto& elem : container) {if (elem.type() == typeid(int)) {std::cout << "Int: " << std::any_cast<int>(elem) << "\n";} else if (elem.type() == typeid(double)) {std::cout << "Double: " << std::any_cast<double>(elem) << "\n";} else if (elem.type() == typeid(std::string)) {std::cout << "String: " << std::any_cast<std::string>(elem) << "\n";}}return 0;
}

输出

Int: 42
Double: 3.14
String: C++17

9. 总结

std::any为C++提供了灵活的类型安全存储机制,适用于需要运行时处理未知类型的场景。尽管它牺牲了一定的性能(类型检查和存储开销),但在设计动态系统时非常有用。使用时需注意类型匹配和异常处理,避免滥用。

http://www.xdnf.cn/news/597205.html

相关文章:

  • ORA-00313 ORA-00312 ORA-27037 redo被删除后重建
  • 如何顺利地将应用程序从 Android 转移到Android
  • SpringCloud (3) 配置中心
  • vue项目的dist在nginx部署后报错Uncaught SyntaxError
  • 技术篇-2.2.JAVA应用场景及开发工具安装
  • Spring Boot 注解 @ConditionalOnMissingBean是什么
  • 嵌入式开发学习日志(linux系统编程--io文件偏移函数(3)和目录)Day26
  • 文件IO操作、目录操作
  • 【leetcode】3355. 零数组变换Ⅰ
  • HCIP-AI培养计划,成为新时代AI解决方案架构高级工程师
  • Metal入门,使用Metal实现纹理效果
  • SpringBoot的启动原理?
  • 若依代码生成
  • 人工智能时代:从“知识容器”到“知识地图”的认知革命
  • 芯片数据手册下载网站整理
  • 价格行为(PriceAction)复盘 - Google - 250521
  • vector
  • Python训练营---Day33
  • Unity Max SDK接入MRec广告,自定义显示位置
  • Linux spi
  • 投票合约1
  • 二叉树结构与遍历
  • 消费类,小家电产品如何做Type-C PD快充快速充电
  • Hugging Face 设置镜像源 | 下载失败 | 下载的速度很慢
  • 轴承与螺母表面缺陷数据集
  • 全栈开发中主流 AI 编程辅助工具的实践与对比分析20250522
  • 人工智能时代教育信息化领导力的培养与发展研究
  • 【知识图谱-一-综述】
  • 新一代网校培训平台的技术演进:从教育源码开发到AI赋能教学转型
  • debian12安装freeswitch1.10