推荐用隐式布尔转换判空(if (opt)),最简洁安全;其次可用has_value();避免与nullopt比较。取值前必须判空,否则UB;默认值用value_or();构造赋值需注意初始化状态。
std::optional 判空最直接、推荐的方式是用 隐式布尔转换(即 if (opt)),而不是手动比较 has_value() 或与 nullopt 比较——它更简洁、安全、符合惯用法。
✅ 推荐:隐式转换(最自然、最常用)
if (opt) { /* 有值 */ }if (!opt) { /* 为空 */ }has_value(),但语义清晰,无额外开销⚠️ 可用但稍冗余:显式调用 has_value()
if (opt.has_value()) { /* 有值 */ }❌ 不推荐:与 std::nullopt 比较
if (opt != std::nullopt) —— 可读性差,且对自定义类型可能触发不必要的构造/比较opt.value() 和 *opt 都要求 optional 非空;若为空,前者抛 std::bad_optional_access,后者触发未定义行为(UB)。
if (opt) use(*opt); 或 if (opt) use(opt.value());
opt.value_or(default_val),它内部已自动判空int x = *opt;(没检查)→ 危险!✅ 正确初始化
std::optional opt1{42}; —— 有值std::optional opt2{}; 或 std::optional opt
2 = std::nullopt; —— 为空std::optional<:string> opt3{"hello"}; —— 支持直接构造含值对象✅ 安全赋值
opt = 100; → 自动就地构造(若已有值则析构旧值再构造新值)opt = std::nullopt; → 清空为无效状态opt = std::optional{200}; → 移动赋值,高效把可能失败的计算封装成返回 std::optional 的函数,调用方自然处理“无结果”情况:
std::optional find_value(const std::map<:string int>& m, const std::string& k) { auto it = m.find(k); return (it != m.end()) ? std::optional(it->second) : std::nullopt; } if (auto val = find_value(m, "key")) { std::cout (利用 if 初始化 + 隐式判空)