0

0

为什么说现代c++应该避免使用void*? (类型安全替代方案)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-15 14:10:03

|

666人浏览过

|

来源于php中文网

原创

void*在现代C++中是危险信号,因它放弃编译期类型检查,易导致未定义行为;应优先用std::any(运行时类型安全)、std::variant(编译期已知类型)或模板(零开销泛型)。

为什么说现代c++应该避免使用void*? (类型安全替代方案)

void\* 为什么在现代 C++ 中被视为危险信号

因为 void\* 完全放弃编译期类型检查,把类型转换责任推给程序员,而现代 C++ 的核心设计哲学之一就是「让错误在编译时暴露」。一旦你用 void\* 传参、存储或返回,static_castreinterpret_cast 的时机、目标类型是否匹配、对象生命周期是否有效,全靠人工保证——这正是内存越界、未定义行为(UB)和难以调试崩溃的温床。

std::any 是最直接的类型安全替代方案

当你需要一个能存任意可复制类型的容器(比如配置项、插件参数、反射字段值),std::any 提供运行时类型信息 + 安全取值机制,比 void\* + 手动 reinterpret_cast 可靠得多:

std::any data = 42;
data = std::string("hello");

if (data.type() == typeid(std::string)) { std::string& s = std::any_cast(data); // 安全:失败抛 std::bad_any_cast }

  • std::any 自动管理所含对象的构造/析构和内存
  • 取值必须显式指定目标类型,不匹配时抛异常而非静默 UB
  • 不能隐式转换,杜绝「我以为是 int,其实是 double」这类误用
  • 注意:它有小对象优化(SOO),但大对象仍涉及堆分配,性能敏感路径需权衡

std::variant 更适合已知有限类型的场景

如果你实际只会在几个固定类型间切换(例如解析 JSON 值:int / double / bool / std::string / nullptr_t),std::variant 是零开销、类型安全、且支持 std::visit 的首选:

using json_value = std::variant;
json_value v = 3.14;

std::visit([](const auto& x) { using T = std::decay_t; if constexpr (std::is_same_v) { std::cout << "double: " << x << "\n"; } }, v);

  • 编译期确定所有可能类型,无运行时类型擦除开销
  • 访问必须覆盖全部分支(否则编译警告),强制处理每种情况
  • 不能存非 nothrow_copy_constructible 类型(如某些自定义类),这点比 std::any 严格
  • 别用 std::get(v) 直接强取——先用 std::holds_alternative(v) 检查,否则抛 std::bad_variant_access

模板和概念(concepts)才是真正的“泛型”解法

绝大多数本想用 void\* 实现的通用逻辑(比如容器、算法、回调),其实应该用模板。C++20 的 concepts 还能让约束更清晰:

魔珐星云
魔珐星云

无需昂贵GPU,一键解锁超写实/二次元等多风格3D数字人,跨端适配千万级并发的具身智能平台。

下载

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

template
requires std::copyable && std::equality_comparable
bool contains(const std::vector& v, const T& x) {
    return std::find(v.begin(), v.end(), x) != v.end();
}
  • 编译器生成特化版本,无运行时开销,类型安全拉满
  • 错误信息明确指向哪个约束不满足(比如传了不可拷贝的类型),而不是 void\* 那种「段错误后才猜」
  • 函数指针回调场景?用 std::function 或模板参数,别用 void\* user_data
  • 唯一合理用 void\* 的地方:对接 C API(如 qsort、OpenGL 回调)。此时务必用 static_cast 且确保生命周期严格对齐

真正难的不是选 std::any 还是 std::variant,而是识别出「这个接口真需要类型擦除吗」——多数时候,答案是否定的。强行用 void\* 省事,最后花十倍时间 debug,得不偿失。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

411

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

533

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

309

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

74

2025.09.10

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

537

2024.08.29

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

8

2026.01.15

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.3万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.2万人学习

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

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