Chapter16 std::variant<>

通过std::variant<>,C++标准库提供了一个新的 联合 类型, 它最大的优势是提供了一种新的具有多态性的处理异质集合的方法。 也就是说,它可以帮助我们处理不同类型的数据,并且不需要公共基类和指针。

16.1 std::variant<>的动机

起源于C语言,C++也提供对union的支持,它的作用是持有一个值, 这个值的类型可能是指定的若干类型中的任意 一个 。 然而,这项语言特性有一些缺陷:

  • 对象并不知道它们现在持有的值的类型。
  • 因此,你不能持有非平凡类型,例如std::string(没有进行特殊处理的话)。

  • 你不能从union派生。

通过std::variant<>,C++标准库提供了一种 可辨识的联合(closed discriminated union) (这意味着要指明一个可能的类型列表)

  • 当前值的类型已知
  • 可以持有任何类型的值
  • 可以从它派生

事实上,一个std::variant<>持有的值有若干 候选项(alternative) ,这些选项通常有不同的类型。 然而,两个不同选项的类型也有可能相同,这在多个类型相同的选项分别代表不同含义的时候很有用 (例如,可能有两个选项类型都是字符串,分别代表数据库中不同列的名称, 你可以知道当前的值代表哪一个列)。

variant所占的内存大小等于所有可能的底层类型中最大的再加上一个记录当前选项的固定内存开销。 不会分配堆内存。

一般情况下,除非你指定了一个候选项来表示为空,否则variant不可能为空。然而,在非常罕见的情况下 (例如赋予一个不同类型新值时发生了异常),variant可能会变为没有值的状态。

std::optional<>std::any一样,variant对象是值语义。 也就是说,拷贝被实现为 深拷贝 ,将会创建一个在自己独立的内存空间内存储有当前选项的值的新对象。 然而,拷贝std::variant<>的开销要比拷贝当前选项的开销稍微大一点, 这是因为variant必须找出要拷贝哪个值。另外,variant也支持move语义。

16.2 使用std::variant<>

下面的代码展示了std::variant<>的核心功能:

  1. #include <variant>
  2. #include <iostream>
  3. int main()
  4. {
  5. std::variant<int, std::string> var{"hi"}; // 初始化为string选项
  6. std::cout << var.index() << '\n'; // 打印出1
  7. var = 42; // 现在持有int选项
  8. std::cout << var.index() << '\n'; // 打印出0
  9. ...
  10. try {
  11. int i = std::get<0>(var); // 通过索引访问
  12. std::string s = std::get<std::string>(var); // 通过类型访问(这里会抛出异常)
  13. ...
  14. }
  15. catch (const std::bad_variant_access& e) { // 当索引/类型错误时进行处理
  16. std::cerr << "EXCEPTION: " << e.what() << '\n';
  17. ...
  18. }
  19. }

成员函数index()可以用来指出当前选项的索引(第一个选项的索引是0)。

初始化和赋值操作都会查找最匹配的选项。如果类型不能精确匹配, 可能会发生奇怪的事情。

注意空variant、有引用成员的variant、有C风格数组成员的variant、 有不完全类型(例如void)的variant都是不允许的。

variant没有空的状态。这意味着每一个构造好的variant对象,至少调用了一次构造函数。 默认构造函数会调用第一个选项类型的默认构造函数:

  1. std::variant<std::string, int> var; // => var.index()==0, 值==""

如果第一个类型没有默认构造函数,那么调用variant的默认构造函数将会导致编译期错误:

  1. struct NoDefConstr {
  2. NoDefConstr(int i) {
  3. std::cout << "NoDefConstr::NoDefConstr(int) called\n";
  4. }
  5. };
  6. std::variant<NoDefConstr, int> v1; // ERROR:不能默认构造第一个选项

辅助类型std::monostate提供了处理这种情况的能力,还可以用来模拟空值的状态。

std::monostate

为了支持第一个类型没有默认构造函数的variant,C++标准库提供了一个特殊的辅助类: std::monostatestd::monostate类型的对象总是处于相同的状态。 因此,比较它们的结果总是相等。它的作用是可以作为一个选项,当variant处于这个选项时表示 此variant 没有其他任何类型的值

因此,std::monostate可以作为第一个选项类型来保证variant能默认构造。例如:

  1. std::variant<std::monostate, NoDefConstr, int> v2; // OK
  2. std::cout << "index: " << v2.index() << '\n'; // 打印出0

某种意义上,你可以把这种状态解释为variant为空的信号。

下面的代码展示了几种检测monostate的方法,也同时展示了variant的其他一些操作:

  1. if (v2.index() == 0) {
  2. std::cout << "has monostate\n";
  3. }
  4. if (!v2.index()) {
  5. std::cout << "has monostate\n";
  6. }
  7. if (std::holds_alternative<std::monostate>(v2)) {
  8. std::cout << "has monostate\n";
  9. }
  10. if (std::get_if<0>(&v2)) {
  11. std::cout << "has monostate\n";
  12. }
  13. if (std::get_if<std::monostate>(&v2)) {
  14. std::cout << "has monostate\n";
  15. }

get_if<>()的参数是一个指针,并在当前选项为T时返回一个指向当前选项的指针, 否则返回nullptr。这和get<T>()不同,后者获取variant的引用作为参数并在 提供的索引或类型正确时以值返回当前选项,否则抛出异常。

和往常一样,你可以赋予variant一个和当前选项类型不同的其他选项的值, 甚至可以赋值为monostate来表示为空:

  1. v2 = 42;
  2. std::cout << "index: " << v2.index() << '\n'; // index:1
  3. v2 = std::monostate{};
  4. std::cout << "index: " << v2.index() << '\n'; // index: 0

从variant派生

你可以从variant派生。例如,你可以定义如下派生自std::variant<>的 聚合体:

  1. class Derived : public std::variant<int, std::string> {
  2. };
  3. Derived d = {{"hello"}};
  4. std::cout << d.index() << '\n'; // 打印出:1
  5. std::cout << std::get<1>(d) << '\n'; // 打印出:hello
  6. d.emplace<0>(77); // 初始化为int,销毁string
  7. std::cout << std::get<0>(d) << '\n'; // 打印出:77

16.3 std::variant<>类型和操作

这一节详细描述了std::variant<>类型和操作。

16.3.1 std::variant<>类型

在头文件variant,C++标准库以如下方式定义了类std::variant<>

  1. namespace std {
  2. template<typename... Types> class variant;
  3. // 译者注:此处原文的定义是
  4. // template<typename Types...> class variant;
  5. // 应是作者笔误
  6. }

也就是说,std::variant<>是一个 可变参数(variadic) 类模板 (C++11引入的处理任意数量参数的特性)。

另外,还定义了下面的类型和对象:

  • 类模板std::variant_size
  • 类模板std::variant_alternative
  • std::variant_npos
  • 类型std::monostate
  • 异常类std::bad_variant_access,派生自std::exception

还有两个为variant定义的变量模板:std::in_place_type<>std::in_place_index<>。 它们的类型分别是std::in_place_type_tstd::in_place_index_t,定义在头文件<utility>中。

16.3.2 std::variant<>的操作

std::variant的操作列出了std::variant<>的所有操作。

操作符 效果
构造函数 创建一个variant对象(可能会调用底层类型的构造函数)
析构函数 销毁一个variant对象
= 赋新值
emplace<T>() 销毁旧值并赋一个T类型选项的新值
emplace<Idx>() 销毁旧值并赋一个索引为Idx的选项的新值
valueless_by_exception() 返回变量是否因为异常而没有值
index() 返回当前选项的索引
swap() 交换两个对象的值
==、!=、<、<=、>、>= 比较variant对象
hash<> 计算哈希值的函数对象类型
holds_alternative<T>() 返回是否持有类型T的值
get<T>() 返回类型为T的选项的值
get<Idx>() 返回索引为Idx的选项的值
get_if<T>() 返回指向类型为T的选项的指针或nullptr
get_if<Idx>() 返回指向索引为Idx的选项的指针或nullptr
visit() 对当前选项进行操作

构造函数

默认情况下,variant的默认构造函数会调用第一个选项的默认构造函数:

  1. std::variant<int, int, std::string> v1; // 第一个int初始化为0,index()==0

选项被默认初始化,意味着基本类型会初始化为0falsenullptr

如果传递一个值来初始化,将会使用最佳匹配的类型:

  1. std::variant<long, int> v2{42};
  2. std::cout << v2.index() << '\n'; // 打印出1

然而,如果有两个类型同等匹配会导致歧义:

  1. std::variant<long, long> v3{42}; // ERROR:歧义
  2. std::variant<int, float> v4{42.3}; // ERROR:歧义
  3. std::variant<int, double> v5{42.3}; // OK
  4. std::variant<int, long double> v6{42.3}; // ERROR:歧义
  5. std::variant<std::string, std::string_view> v7{"hello"}; // ERROR:歧义
  6. std::variant<std::string, std::string_view, const char*> v8{"hello"}; // OK
  7. std::cout << v8.index() << '\n'; // 打印出2

为了传递多个值来调用构造函数初始化,你必须使用in_place_type或者 in_place_index标记:

  1. std::variant<std::complex<double>> v9{3.0, 4.0}; // ERROR
  2. std::variant<std::complex<double>> v10{{3.0, 4.0}}; // ERROR
  3. std::variant<std::complex<double>> v11{std::in_place_type<std::complex<double>>, 3.0, 4.0};
  4. std::variant<std::complex<double>> v12{std::in_place_index<0>, 3.0, 4.0};

你也可以使用in_place_index在初始化时解决歧义问题或者打破匹配优先级:

  1. std::variant<int, int> v13{std::in_place_index<1>, 77}; // 初始化第二个int
  2. std::variant<int, long> v14{std::in_place_index<1>, 77}; // 初始化long,而不是int
  3. std::cout << v14.index() << '\n'; // 打印出1

你甚至可以传递一个带有其他参数的初值列:

  1. // 用一个lambda作为排序准则初始化一个set的variant:
  2. auto sc = [] (int x, int y) {
  3. return std::abs(x) < std::abs(y);
  4. };
  5. std::variant<std::vector<int>, std::set<int, decltype(sc)>>
  6. v15{std::in_place_index<1>, {4, 8, -7, -2, 0, 5}, sc};

然而,只有当所有初始值都和容器里元素类型匹配时才可以这么做。 否则你必须显式传递一个std:: initializer_list<>

  1. // 用一个lambda作为排序准则初始化一个set的variant
  2. auto sc = [] (int x, int y) {
  3. return std::abs(x) < std::abs(y);
  4. };
  5. std::variant<std::vector<int>, std::set<int, decltype(sc)>>
  6. v15{std::in_place_index<1>, std::initializer_list<int>{4, 5L}, sc};

std::variant<>不支持类模板参数推导, 也没有make_variant<>()快捷函数(不像std::optional<>std::any)。这样做也没有意义,因为variant的目标是处理多个候选项。

如果所有的候选项都支持拷贝,那么就可以拷贝variant对象:

  1. struct NoCopy {
  2. NoCopy() = default;
  3. NoCopy(const NoCopy&) = delete;
  4. };
  5. std::variant<int, NoCopy> v1;
  6. std::variant<int, NoCopy> v2{v1}; // ERROR

访问值

通常的方法是调用get<>()get_if<>来访问当前选项的值。 你可以传递一个索引、或者传递一个类型(该类型的选项只能有一个)。 使用一个无效的索引和无效/歧义的类型将会导致编译错误。 如果访问的索引或者类型不是当前的选项,将会抛出一个std::bad_variant_access异常。

例如:

  1. std::variant<int, int, std::string> var; // 第一个int设为0,index()==0
  2. auto a = std::get<double>(var); // 编译期ERROR:没有double类型的选项
  3. auto b = std::get<4>(var); // 编译期ERROR:没有第五个选项
  4. auto c = std::get<int>(var); // 编译期ERROR:有两个int类型的选项
  5. try {
  6. auto s = std::get<std::string>(var); // 抛出异常(当前选项是第一个int)
  7. auto i = std::get<0>(var); // OK,i==0
  8. auto j = std::get<1>(var); // 抛出异常(当前选项是另一个int)
  9. }
  10. catch (const std::bad_variant_access& e) {
  11. std::cout << "Exception: " << e.what() << '\n';
  12. }

还有另一个API可以在访问值之前先检查给定的选项是否是当前选项。 你需要给get_if<>()传递一个指针,如果访问成功则返回指向当前选项的指针,否则返回nullptr

  1. if (auto ip = std::get_if<1>(&var); ip != nullptr) {
  2. std::cout << *ip << '\n';
  3. }
  4. else {
  5. std::cout << "alternative with index 1 not set\n";
  6. }

这里还使用了带初始化的if语句,把初始化过程和条件检查分成了两条语句。 你也可以直接把初始化语句用作条件语句:

  1. if (auto ip = std::get_if<1>(&var)) {
  2. std::cout << *ip << '\n';
  3. }
  4. else {
  5. std::cout << "alternative with index 1 not set\n";
  6. }

另一种访问不同选项的值的方法是使用variant访问器。

修改值

赋值操作和emplace()函数可以修改值:

  1. std::variant<int, int, std::string> var; // 设置第一个int为0,index()==0
  2. var == "hello"; // 设置string选项,index()==2
  3. var.emplace<1>(42); // 设置第二个int,index()==1

注意operator=将会直接赋予variant一个新值,只要有和新值类型对应的选项。 emplace()在赋予新值之前会先销毁旧的值。

你也可以使用get<>()或者get_if<>()来给当前选项赋予新值:

  1. std::variant<int, int, std::string> var; // 设置第一个int为0,index()==0
  2. std::get<0>(var) = 77; // OK,但当前选项仍是第一个int
  3. std::get<1>(var) = 99; // 抛出异常(因为当前选项是另一个int)
  4. if (auto p = std::get_if<1>(&var); p) { // 如果第二个int被设置
  5. *p = 42; // 修改值
  6. }

另一个修改不同选项的值的方法是variant访问器。

比较

对两个类型相同的variant(也就是说,它们每个选项的类型和顺序都相同),你可以使用通常的比较运算符。 比较运算将遵循如下规则:

  • 当前选项索引较小的小于当前选项索引较大的。
  • 如果两个variant当前的选项相同,将调用当前选项类型的比较运算符进行比较。 注意所有的std::monostate类型的对象都相等。
  • 两个variant都处于特殊状态(valueless_by_exception()为真的状态)时相等。 否则,valueless_by_ exception()返回ture的variant小于另一个。

例如:

  1. std::variant<std::monostate, int, std::string> v1, v2{"hello"}, v3{42};
  2. std::variant<std::monostate, std::string, int> v4;
  3. v1 == v4 // 编译期错误
  4. v1 == v2 // 返回false
  5. v1 < v2 // 返回true
  6. v1 < v3 // 返回true
  7. v2 < v3 // 返回false
  8. v1 = "hello";
  9. v1 == v2 // 返回true
  10. v2 = 41;
  11. v2 < v3 // 返回true

move语义

只要所有的选项都支持move语义,那么std::variant<>也支持move语义。

如果你move了variant对象,那么当前状态会被拷贝,而当前选项的值会被move。 因此,被move的variant对象仍然保持之前的选项,但值会变为未定义。

你也可以把值移进或移出variant对象。

哈希

如果所有的选项类型都能计算哈希值,那么variant对象也能计算哈希值。 注意variant对象的哈希值 保证是当前选项的哈希值。在某些平台上它是,有些平台上不是。

16.3.3 访问器

另一个处理variant对象的值的方法是使用访问器(visitor)。访问器是为每一个可能的类型都提供一个 函数调用运算符的对象。当这些对象“访问”一个variant时,它们会调用和当前选项类型最匹配的函数。

使用函数对象作为访问器

例如:

  1. #include <variant>
  2. #include <string>
  3. #include <iostream>
  4. struct MyVisitor
  5. {
  6. void operator() (int i) const {
  7. std::cout << "int: " << i << '\n';
  8. }
  9. void operator() (std::string s) const {
  10. std::cout << "string: " << s << '\n';
  11. }
  12. void operator() (long double d) const {
  13. std::cotu << "double: " << d << '\n';
  14. }
  15. };
  16. int main()
  17. {
  18. std::variant<int, std::string, double> var(42);
  19. std::visit(MyVisitor(), var); // 调用int的operator()
  20. var = "hello";
  21. std::visit(MyVisitor(), var); // 调用string的operator()
  22. var = 42.7;
  23. std::visit(MyVisitor(), var); // 调用long double的operator()
  24. }

如果访问器没有某一个可能的类型的operator()重载,那么调用visit()将会导致 编译期错误,如果调用有歧义的话也会导致编译期错误。这里的示例能正常工作是因为long doubleint更匹配double

你也可以使用访问器来修改当前选项的值(但不能赋予一个其他选项的新值)。 例如:

  1. struct Twice
  2. {
  3. void operator()(double& d) const {
  4. d *= 2;
  5. }
  6. void operator()(int& i) const {
  7. i *= 2;
  8. }
  9. void operator()(std::string& s) const {
  10. s = s + s;
  11. }
  12. };
  13. std::visit(Twice(), var); // 调用匹配类型的operator()

访问器调用时只根据类型判断,你不能对类型相同的不同选项做不同的处理。

注意上面例子中的函数调用运算符都应该标记为const,因为它们 是 无状态的(stateless) (它们的行为只受参数的影响)。

使用泛型lambda作为访问器

最简单的使用访问器的方式是使用泛型lambda,它是一个可以处理任意类型的函数对象:

  1. auot printvariant = [] (const auto& val) {
  2. std::cout << val << '\n';
  3. };
  4. ...
  5. std::visit(printvariant, var);

这里,泛型lambda生成的闭包类型中会将函数调用运算符定义为模板:

  1. class ComplierSpecificClosureTypeName {
  2. public:
  3. template<typename T>
  4. auto operator() (const T& val) const {
  5. std::cout << val << '\n';
  6. }
  7. };

因此,只要调用时生成的函数内的语句有效(这个例子中就是输出运算符要有效), 那么把lambda传递给std::visit()就可以正常编译。

你也可以使用lambda来修改当前选项的值:

  1. // 将当前选项的值变为两倍:
  2. std::visit([] (auto& val) {
  3. val = val + val;
  4. }, var);

或者:

  1. // 将当前选项的值设为默认值:
  2. std::visit([] (auto& val) {
  3. val = std::remove_reference_t<decltype(val)>{};
  4. }, var);

你甚至可以使用编译期if语句来对不同的选项类型进行不同的处理。例如:

  1. auto dblvar = [](auto& val) {
  2. if constexpr(std::is_convertible_v<decltype(val), std::string>) {
  3. val = val + val;
  4. }
  5. else {
  6. val *= 2;
  7. }
  8. };
  9. ...
  10. std::visit(dblvar, var);

这里,对于std::string类型的选项,泛型lambda会把函数调用模板实例化为计算:

  1. val = val + val;

而对于其他类型的选项,例如intdouble,lambda函数调用模板会实例化为计算:

  1. val *= 2;

注意检查val的类型时必须小心。 这里,我们检查了val的类型是否能转换为std::string。 如下检查:

  1. if constexpr(std::is_same_v<decltype(val), std::string>) {

将不能正确工作,因为val的类型只可能是int&、std::string&、 long double&这样的引用类型。

在访问器中返回值

访问器中的函数调用可以返回值,但所有返回值类型必须相同。例如:

  1. using IntOrDouble = std::variant<int, double>;
  2. std::vector<IntOrDouble> coll { 42, 7.7, 0, -0.7 };
  3. double sum{0};
  4. for (const auto& elem : coll) {
  5. sum += std::visit([] (const auto& val) -> double {
  6. return val;
  7. }, elem);
  8. }

上面的代码会把所有选项的值加到sum上。 如果lambda中没有显式指明返回类型将不能通过编译,因为自动推导的话返回类型会不同。

使用重载的lambda作为访问器

通过使用函数对象和lambda的 重载器(overloader) ,可以定义一系列lambda, 其中最佳的匹配将会被用作访问器。

假设有一个如下定义的overload重载器:

  1. // 继承所有基类里的函数调用运算符
  2. template<typename... Ts>
  3. struct overload : Ts...
  4. {
  5. using Ts::operator()...;
  6. };
  7. // 基类的类型从传入的参数中推导
  8. template<typename... Ts>
  9. overload(Ts...) -> overload<Ts...>;

你可以为每个可能的选项提供一个lambda,之后使用overload来访问variant:

  1. std::variant<int, std::string> var(42);
  2. ...
  3. std::visit(overload { // 为当前选项调用最佳匹配的lambda
  4. [](int i) { std::cout << "int: " << i << '\n'; },
  5. [] (const std::string& s) {
  6. std::cout << "string: " << s << '\n';
  7. },
  8. }, var);

你也可以使用泛型lambda,它可以匹配所有情况。例如,要想修改一个variant当前选项的值, 你可以使用重载实现字符串和其他类型值“翻倍”:

  1. auto twice = overload {
  2. [] (std::string& s) { s += s; },
  3. [] (auto& i) { i *= 2; },
  4. };

使用这个重载,对于字符串类型的选项,值将变为原本的字符串再重复一遍; 而对于其他类型,将会把值乘以2。下面展示了怎么应用于variant:

  1. std::variant<int, std::string> var(42);
  2. std::visit(twice, var); // 值42变为84
  3. ...
  4. var = "hi";
  5. std::visit(twice, var); // 值"hi"变为"hihi"

16.3.4 异常造成的无值

如果你赋给一个variant新值时发生了异常,那么这个variant可能会进入一个非常特殊的状态: 它已经失去了旧的值但还没有获得新的值。例如:

  1. struct S {
  2. operator int() { throw "EXCEPTION"; } // 转换为int时会抛出异常
  3. };
  4. std::variant<double, int> var{12.2}; // 初始化为double
  5. var.emplace<1>(S{}); // OOPS:当设为int时抛出异常

如果这种情况发生了,那么:

  • var.valueless_by_exception()会返回true
  • var.index()会返回std::variant_npos

这些都标志该variant当前没有值。

这种情况下有如下保证:

  • 如果emplace()抛出异常,那么valueless_by_exception()可能会返回true
  • 如果operator=()抛出异常且该修改不会改变选项,那么index()valueless_by_exception()的状态将保持不变。值的状态依赖于值类型的异常保证。
  • 如果operator=()抛出异常且新值是新的选项,那么variant 可能 会没有值 (valueless_by_exception() 可能 会返回true)。具体情况依赖于异常抛出的时机。 如果发生在实际修改值之前的类型转换期间,那么variant将依然持有旧值。

通常情况下,如果你不再使用这种情况下的variant,那么这些保证就足够了。 如果你仍然想使用抛出了异常的variant,你需要检查它的状态。例如:

  1. std::variant<double, int> var{12.2}; // 初始化为double
  2. try {
  3. var.emplace<1>(S{}); // OOPS:设置为int时抛出异常
  4. }
  5. catch (...) {
  6. if (!var.valueless_by_exception()) {
  7. ...
  8. }
  9. }

16.4 使用std::variant实现多态的异质集合

std::variant允许一种新式的多态性,可以用来实现异质集合。 这是一种带有闭类型集合的运行时多态性。

关键在于variant<>可以持有多种选项类型的值。 可以将元素类型定义为variant来实现异质的集合,这样的集合可以持有不同类型的值。 因为每一个variant知道当前的选项,并且有了访问器接口,我们可以定义在运行时根据不同类型 进行不同操作的函数/方法。同时因为variant有值语义,所以我们不需要指针(和相应的内存管理)或者虚函数。

16.4.1 使用std::variant实现几何对象

例如,假设我们要负责编写表示几何对象的库:

  1. #include <iostream>
  2. #include <variant>
  3. #include <vector>
  4. #include "coord.hpp"
  5. #include "line.hpp"
  6. #include "circle.hpp"
  7. #include "rectangle.hpp"
  8. // 所有几何类型的公共类型
  9. using GeoObj = std::variant<Line, Circle, Rectangle>;
  10. // 创建并初始化一个几何体对象的集合
  11. std::vector<GeoObj> createFigure()
  12. {
  13. std::vector<GeoObj> f;
  14. f.push_back(Line{Coord{1, 2}, Coord{3, 4}});
  15. f.push_back(Circle{Coord{5, 5}, 2});
  16. f.push_back(Rectangle{Coord{3, 3}, Coord{6, 4}});
  17. return f;
  18. }
  19. int main()
  20. {
  21. std::vector<GeoObj> figure = createFigure();
  22. for (const GeoObj& geoobj : figure) {
  23. std::visit([] (const auto& obj) {
  24. obj.draw(); // 多态性调用draw()
  25. }, geoobj);
  26. }
  27. }

首先,我们为所有可能的类型定义了一个公共类型:

  1. using GeoObj = std::variant<Line, Circle, Rectangle>;

这三个类型不需要有任何特殊的关系。事实上它们甚至没有一个公共的基类、没有任何虚函数、 接口也可能不同。例如:

  1. #ifndef CIRCLE_HPP
  2. #define CIRCLE_HPP
  3. #include "coord.hpp"
  4. #include <iostream>
  5. class Circle {
  6. private:
  7. Coord center;
  8. int rad;
  9. public:
  10. Circle (Coord c, int r) : center{c}, rad{r} {
  11. }
  12. void move(const Coord& c) {
  13. center += c;
  14. }
  15. void draw() const {
  16. std::cout << "circle at " << center << " with radius " << rad << '\n';
  17. }
  18. };
  19. #endif

我们现在可以创建相应的对象并把它们以值传递给容器,最后可以得到这些类型的元素的集合:

  1. std::vector<GeoObj> createFigure()
  2. {
  3. std::vector<GeoObj> f;
  4. f.push_back(Line{Coord{1, 2}, Coord{3, 4}});
  5. f.push_back(Circle{Coord{5, 5}, 2});
  6. f.push_back(Rectangle{Coord{3, 3}, Coord{6, 4}});
  7. return f;
  8. }

以前如果没有使用继承和多态的话是不可能写出这样的代码的。 以前要想实现这样的异构集合,所有的类型都必须继承自GeoObj, 并且最后将得到一个元素类型为GeoObj的指针的vector。 为了使用指针,必须用new创建新对象,这导致最后还要追踪什么时候调用delete, 或者要使用智能指针来完成(unique_ptr或者shared_ptr)。

现在,通过使用访问器,我们可以迭代每一个元素,并依据元素的类型“做正确的事情”:

  1. std::vector<GeoObj> figure = createFigure();
  2. for (const GeoObj& geoobj : figure) {
  3. std::visit([] (const auto& obj) {
  4. obj.draw(); // 多态调用draw()
  5. }, geoobj);
  6. }

这里,visit()使用了泛型lambda来为每一个可能的GeoObj类型实例化。 也就是说,当编译visit()调用时,lambda将会被实例化并编译为3个函数:

  • 为类型Line编译代码:
    1. [] (const Line& obj) {
    2. obj.draw(); // 调用Line::draw()
    3. }
  • 为类型Circle编译代码:
    1. [] (const Circle& obj) {
    2. obj.draw(); // 调用Circle::draw()
    3. }
  • 为类型Rectangle编译代码:
    1. [] (const Rectangle& obj) {
    2. obj.draw(); // 调用Rectangle::draw()
    3. }

如果这些实例中有一个不能编译,那么对visit()的调用也不能编译。 如果所有实例都能编译,那么将保证会对所有元素类型调用相应的函数。 注意生成的代码并不是 if-else 链。 C++标准保证这些调用的性能不会依赖于variant选项的数量。

也就是说,从效率上讲,这种方式和虚函数表的方式的行为相同 (通过类似于为所有visit()创建局部虚函数表的方式)。 注意,draw()函数不需要是虚函数。

如果对不同类型的操作不同,我们可以使用编译期if语句或 者重载访问器来处理不同的情况(见上边的第二个例子)。

16.4.2 使用std::variant实现其他异质集合

考虑如下另一个使用std::variant<>实现异质集合的例子:

  1. #include <iostream>
  2. #include <string>
  3. #include <variant>
  4. #include <vector>
  5. #include <type_traits>
  6. int main()
  7. {
  8. using Var = std::variant<int, double, std::string>;
  9. std::vector<Var> values {42, 0.19, "hello world", 0.815};
  10. for (const Var& val : values) {
  11. std::visit([] (const auto& v) {
  12. if constexpr(std::is_same_v<decltype(v), const std::string&>) {
  13. std::cout << '"' << v << "\" ";
  14. }
  15. else {
  16. std::cout << v << ' ';
  17. }
  18. }, val);
  19. }
  20. }

我们又一次定义了自己的类型来表示若干可能类型中的一个:

  1. using Var = std::variant<int, double, std::string>;

我们可以用它创建并初始化一个异质的集合:

  1. std::vector<Var> values {42, 0.19, "hello world", 0.815};

注意我们可以用若干异质的元素来实例化vector,因为它们都能自动转换为variant类型。 然而,如果我们还传递了一个long类型的初值,上面的初始化将不能编译, 因为编译器不能决定将它转换为int还是double

当我们迭代元素时,我们使用了访问器来调用相应的函数。这里使用了一个泛型lambda。 lambda为3种可能的类型分别实例化了一个函数调用。 为了对字符串进行特殊的处理(在输出值时用双引号包括起来),我们使用了编译期if语句:

  1. for (const Var& val : values) {
  2. std::visit([] (const auto& v) {
  3. if constexpr(std::is_same_v<decltype(v), const std::string&>) {
  4. std::cout << '"' << v << "\" ";
  5. }
  6. else {
  7. std::cout << v << ' ';
  8. }
  9. }, val);
  10. }

这意味着输出将是:

  1. 42 0.19 "hello world" 0.815

通过使用重载的访问器,我们可以像下面这样实现:

  1. for (const auto& val : values) {
  2. std::visit(overload {
  3. [] (const auto& v) {
  4. std::cout << v << ' ';
  5. },
  6. [] (const std::string& v) {
  7. std::cout << '"' << v "\" ";
  8. }
  9. }, val);
  10. }

然而,注意这样可能会陷入重载匹配的问题。有的情况下泛型lambda(即函数模板) 匹配度比隐式类型更高,这意味着可能会调用错误的类型。

16.4.3 比较多态的variant

让我们来总结一下使用std::variant实现多态的异构集合的优点和缺点:

优点有:

  • 你可以使用任意类型并且这些类型不需要有公共的基类(这种方法是非侵入性的)
  • 你不需要使用指针来实现异质集合
  • 不需要virtual成员函数
  • 值语义(不会出现访问已释放内存或内存泄露等问题)
  • vector中的元素是连续存放在一起的(原本指针的方式所有元素是散乱分布在堆内存中的)

缺点有:

  • 闭类型集合(你必须在编译期指定所有可能的类型)
  • 每个元素的大小都是所有可能的类型中最大的(当不同类型大小差距很大时这是个问题)
  • 拷贝元素的开销可能会更大

一般来说,我并不确定是否要推荐默认使用std::variant<>来实现多态。 一方面这种方法很安全(没有指针,意味着没有newdelete), 也不需要虚函数。然而另一方面,使用访问器有一些笨拙,有时你可能会需要引用语义 (在多个地方使用同一个对象),还有在某些情形下并不能在编译期确定所有的类型。

性能开销也有很大不同。没有了newdelete可能会减少很大开销。 但另一方面,以值传递对象又可能会增大很多开销。 在实践中,你必须自己测试对你的代码来说哪种方法效率更高。 在不同的平台上,我已经观测到性能上的显著差异了。

16.5 std::variant<>的特殊情况

特定类型的variant可能导致特殊或者出乎意料的行为。

16.5.1 同时有boolstd::string选项

如果一个std::variant<>同时有boolstd::string选项, 赋予一个字符串字面量可能会导致令人惊奇的事,因为字符串字面量会优先转换为bool, 而不是std::string。例如:

  1. std::variant<bool, std::string> v;
  2. v = "hi"; // OOPS:设置bool选项
  3. std::cout << "index: " << v.index() << '\n';
  4. std::visit([] (const auto& val) {
  5. std::cout << "value: " << val << '\n';
  6. }, v);

这段代码片段将会有如下输出:

  1. index: 0
  2. value: true

可以看出,字符串字面量会被解释为把variant的bool选项初始化为true (因为指针不是0所以是true)。

这里有一些修正这个赋值问题的方法:

  1. v.emplace<1>("hello"); // 显式赋值给第二个选项
  2. v.emplace<std::string>("hello"); // 显式赋值给string选项
  3. v = std::string{"hello"}; // 确保用string赋值
  4. using namespace std::literals; // 确保用string赋值
  5. v = "hello"s;

参见https://wg21.link/p0608进一步了解关于这个问题的讨论。