[閒聊] 每日C++小秘密(2):任意型別std::any

作者: yam276 ('_')   2024-10-28 19:02:22
為什麼要用 std::any?
void* 是不安全的指標,只存儲記憶體地址而不帶有型別資訊。
你需要手動轉換型別(reinterpret_cast),
這很有可能會導致型別錯誤,而且不容易追蹤。
std::any 則是型別安全的,
可以存儲任意型別的值,並在執行時檢查型別,
避免了錯誤的轉型操作,提升程式的安全性。
void*的情況
Code:
#include <iostream>
void printValue(void* ptr) {
std::cout << *reinterpret_cast<int*>(ptr) << std::endl; // 需要轉型
}
int main() {
int value = 42;
printValue(&value); // 必須傳指標
return 0;
}
std::any的情況
Code:
#include <any>
#include <iostream>
void printValue(const std::any& value) {
std::cout << std::any_cast<int>(value) << std::endl; // 型別安全
}
int main() {
std::any value = 42; // 可以存放任意型別的值
printValue(value); // 無需手動轉型
return 0;
}
並且std::any可以型別檢查:
if (value.type() == typeid(int)) {...}
也可以偵測錯誤轉換:
try {
std::cout << std::any_cast<int>(value) << std::endl; // 嘗試轉換為int
} catch (const std::bad_any_cast& e) {
std::cerr << "Error: " << e.what() << std::endl; // 捕獲轉換錯誤
// Error: bad any_cast
}
表格:
特性 void* std::any
型別安全 否,可能導致未定義行為 是,提供型別檢查與安全轉換
使用簡單性 要手動追蹤型別,不方便 不需要追蹤型別,自動管理
動態型別支持 基本支持,但容易出錯 完整支持,型別檢查更嚴格
效能 較快,但風險更高 較慢,但更安全
適用場景 C 接口、低層次操作 Modern C++ 更推薦
點評:
std::any不是任何情況都能取代void*,
但在能取代的場合更好用。
作者: sustainer123 (caster)   2023-10-28 19:02:00
大師

Links booklink

Contact Us: admin [ a t ] ucptt.com