惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

SecWiki News
SecWiki News
I
InfoQ
The Cloudflare Blog
人人都是产品经理
人人都是产品经理
博客园 - Franky
T
Tailwind CSS Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
量子位
博客园_首页
罗磊的独立博客
V
V2EX
李成银的技术随笔
大猫的无限游戏
大猫的无限游戏
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
T
True Tiger Recordings
Vercel News
Vercel News
Cyberwarzone
Cyberwarzone
Cisco Talos Blog
Cisco Talos Blog
F
Fox-IT International blog
D
Darknet – Hacking Tools, Hacker News & Cyber Security
M
Microsoft Research Blog - Microsoft Research
Know Your Adversary
Know Your Adversary
爱范儿
爱范儿
The Register - Security
The Register - Security
G
Google Developers Blog
The Hacker News
The Hacker News
Malwarebytes
Malwarebytes
S
Securelist
博客园 - 三生石上(FineUI控件)
Jina AI
Jina AI
T
Threat Research - Cisco Blogs
T
The Exploit Database - CXSecurity.com
S
SegmentFault 最新的问题
博客园 - 叶小钗
F
Fortinet All Blogs
Apple Machine Learning Research
Apple Machine Learning Research
宝玉的分享
宝玉的分享
博客园 - 聂微东
T
Threatpost
博客园 - 【当耐特】
D
Docker
P
Privacy & Cybersecurity Law Blog
www.infosecurity-magazine.com
www.infosecurity-magazine.com
G
GRAHAM CLULEY
V
Visual Studio Blog
C
Cisco Blogs
IT之家
IT之家
S
Security Archives - TechRepublic
Latest news
Latest news
阮一峰的网络日志
阮一峰的网络日志

Jun's Blog

2023 年度总结 OS 学习记 之 XV6 如何编译 GraalVM LLVM 中端优化之 InstCombine C++ 中 inline 关键字的语义 链接与库 C++ 移动语义基础 如何优化矩阵相乘 2022年度总结 CUDA初学笔记 汇编语言之保护模式 汇编语言之实模式 浅析 libc++ 中的 string 实现 C++模板基础 CSAPP第九章笔记之虚拟内存 CSAPP第八章笔记之异常控制流 2021年度总结 编译安装GCC12 如何给LLVM贡献代码 CSAPP第三章笔记Part 2 CSAPP第三章笔记Part 1 GDB基本使用笔记 读Effetive Modern C++ 之类型推导 给计算机新生的一封信 CMake学习笔记 由Redis学习数据结构--字典 C++中lambda表达式基础 由Redis学习数据结构--链表 对Python及爬虫行业的思考 Vim实用技巧 浅谈C++中的类 简要剖析const关键字 正则表达式基础总结 使用Hugo和Firebase部署个人博客 浅谈HTTPS证书 Linux下使用v2ray 娱乐至死读书笔记 自控力读书笔记 少有人走的路力读书笔记 Postfix & Dovecot 自建邮箱服务 Linux用户管理 Linux常用命令总结 Linux开机流程 Git学习笔记 Linux文件与目录 Linux硬盘管理 Linux服务浅谈 About Me
std::expected 基本使用
2022-05-09 · via Jun's Blog

· Jun

最近看cppreference的编译器支持的时候发现libstdc++已经实现<expected>了,可惜的是在网上看了一下发现没什么太多它的材料,只能自己看paper和实现写篇博文记录下学习过程了。

什么是std::expected?

简单点理解它和std::optional差不多,但是std::optional只能表示有正常的值或者为std::nullopt,即空值。而std::expected则可以表示一个期望的值和一个错误的值,相当于两个成员的std::variant,但是在接口上更方便使用。可以把它当作新的一种的错误处理方式。

基本使用

构造

std::expected<T,E>有两个模板参数,第一个表示为期望的值,第二个表示错误的值。

如果是期望的值,则有一个隐式的转换。

1
std::expected<int, std::string> e = 42;

如果是异常值,则要通过std::expected()来初始化。

1
std::expected<int, std::string> e = std::unexpected("Error");

默认构造函数

1
std::expected<S, int> e;

要求S有一个default constructor,否则会报错。

使用

std::optional一样有指针语义,可以解引用。

1
2
3
std::expected<int, std::string> e = 42;
if (e)
  std::cout << *e << "\n"; // prints 42

注意解引用前必须检查,否则为UB!!!

下面和上面效果上等价:

1
2
3
std::expected<int, std::string> e = 42;
if (e.has_value())
  std::cout << e.value() << "\n"; // prints 42

注意std::expected<T,E>::value()在值不存在会抛出一个异常,一定程度上更安全。

表示错误

1
2
3
std::expected<int, std::string> e = std::unexpected("Error");
if (!e.has_value())
  std::cout << e.error() << "\n"; // prints `Error`

如果没有错误使用的话为UB!!!

如果说返回的结果为错误,可以通过value_or()来更优雅地提供一个默认值。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
std::expected<int, std::string> MayHasErr(int i) {
    if (i < 0)
        return std::unexpected("Error");
    else
        return i * 2;
}

int main() {
    std::cout << MayHasErr(3).value() << "\n"; // print 3
    std::cout << MayHasErr(-2).value_or(42) << "\n"; // print 42
}

内部实现

std::expected<T,E>的实现比较简单,不需要依靠任何新的语言特性。它由一个类型为T的期望值和std::unexpected<E>的union和一个表示是否为错误的bool值组成。剩下的就是一些helper函数。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
template<class T, class E>
  class expected {
  using unexpected_type = unexpected<E>;
   private:
    bool has_val;
    union {
      T val; // 表示期望的值
      E unex; // 表示异常的值
    };
  };

参考