C++三剑客之std::variant(二):深入剖析

目录

1.概述

2.辅助类介绍

2.1.std::negation

2.2.std::conjunction

2.3.std::is_destructible

2.4.std::is_object

2.5.is_default_constructible

2.6.std::is_trivially_destructible

2.7.std::in_place_type和std::in_place_index

3.原理分析

3.1.存储分析

3.2.构造函数

3.2.1.默认构造

3.2.2.使用单一值初始化

3.2.3.std::in_place_type

3.2.4.std::in_place_index

3.3.访问值

3.3.1.直接赋值

3.3.2.emplace

3.3.3.get

3.3.4.get_if

4.总结


前一篇关于std::variant的博客详细介绍了std::variant的使用和一些注意事项,熟悉和了解它的使用方法后,我们接着追根溯源,探索其本质,仔细阅读它的实现,分析一下源码。

1.概述

本文我们主要研究问题模板类std::variant如何做到存任意多个类型值的容器?不同类型怎么做到巧妙的构造与转换的?多种构造函数如何实现?内部数据怎么储存?为什么不能保存引用、数组和void类型?

std::variant是在头文件variant中,是C++17引入的,本文以VS2019平台展开讲解variant的原理和深层次用法。

2.辅助类介绍

2.1.std::negation

std::negation 逻辑非元函数,一元函数对象类,其调用将返回对其实参求反的结果(由一元操作符-返回)。如:

#include <iostream>
#include <type_traits>static_assert(std::is_same<std::bool_constant<false>,typename std::negation<std::bool_constant<true>>::type>::value,"");
static_assert(std::is_same<std::bool_constant<true>,typename std::negation<std::bool_constant<false>>::type>::value,"");int main()
{std::cout << std::negation<std::bool_constant<true>>::value << '\n';  //输出:falsestd::cout << std::negation<std::bool_constant<false>>::value << '\n'; //输出:true
}

std::bool_constant<true>即是 true_type,取值为true,逻辑反则为false。std::bool_constant<false> 即是false_type,取值为false,逻辑反则为true。

2.2.std::conjunction

std::conjunction 逻辑与元对象,在头文件type_traits中,一般用在判断可变参数是否满足某种条件上。示例如下:

#include <iostream>
#include <type_traits>// func is enabled if all Ts... have the same type as T
template<typename T, typename... Ts>
std::enable_if_t<std::conjunction_v<std::is_same<T, Ts>...>>
func(T, Ts...) {std::cout << "all types in pack are T\n";
}// otherwise
template<typename T, typename... Ts>
std::enable_if_t<!std::conjunction_v<std::is_same<T, Ts>...>>
func(T, Ts...) {std::cout << "not all types in pack are T\n";
}int main() {func(1, 2, 3);func(1, 2, "hello!");
}

输出:

all types in pack are T
not all types in pack are T

上述代码在func中用std::is_same判断模板函数的参数类型是否都是一样的,所有参数类型一样判定为true,否则为false;同样std::variant的源码也用到了这个,如:

template <class... _Types>
using _Variant_storage = _Variant_storage_<conjunction_v<is_trivially_destructible<_Types>...>, _Types...>;

std::conjunction就是判断可变参数对象是否都为简单销毁对象。

2.3.std::is_destructible

std::is_destructible 类型特征来检查一个类是否有可析构的类型。这有助于我们在编译时发现潜在的问题,例如试图删除非指针类型的对象。但它并不保证这个类型的析构函数是否真正做了正确的清理工作。因此,在定义类的析构函数时,我们需要仔细地考虑它是否真正释放了所有分配的资源。如下示例:

#include <iostream>
#include <fstream>
#include <type_traits>class MyClass {
public:MyClass(int size) : arr(new int[size]), file("example.txt") {}~MyClass() { delete [] arr; }private:int* arr;std::ofstream file;
};int main() {std::cout << std::is_destructible<MyClass>::value << '\n'; //输出:truestd::cout << std::is_destructible<int>::value << '\n';     //输出: truestd::cout << std::is_destructible<int[]>::value << '\n';   //输出:falsestd::cout << std::is_destructible<std::ofstream>::value << '\n'; //输出:true
}

从上面的代码可以看出,我们定义的MyClass类具有可析构的类型。而int类型和std::ofstream类型也是可析构的。但是,int[]类型不是可析构的。这是因为数组类型不支持默认构造函数、拷贝构造函数或移动构造函数,从而导致不能正确地销毁。

2.4.std::is_object

std::is_object是一个用于元编程的C++类型特性,用于判断一个类型是否是对象类型,而不是类类型或枚举类型。这个在我之前的博客也讲的很清楚,如果还不是特别明白,可以再去翻翻博客C++之std::is_object-CSDN博客;在这里我就不多赘述了。

2.5.is_default_constructible

std::is_default_constructible模板,用于判断一个类型是否有默认构造函数。因为在某些情况下,需要在编译期间确定一个类型是否有默认构造函数。在使用该模板时需要包含头文件type_traits。示例代码:

#include <iostream>
#include <type_traits>class X {
public:X(int x): m_x(x) { }
private:int m_x;
};class Y {
public:Y() = default;
private:double m_y;
};int main() {std::cout << std::is_default_constructible<X>::value << '\n'; //输出:falsestd::cout << std::is_default_constructible<Y>::value << '\n'; //输出:truestd::cout << std::is_default_constructible<int>::value << '\n'; //输出:truestd::cout << std::is_default_constructible<int[]>::value << '\n'; //输出:falsereturn 0;
}

在上述示例代码中,我们定义了两个类X和Y,分别设置了构造函数和默认构造函数。然后分别使用is_default_constructible模板来判断是否有默认构造函数,最后还演示了一些基本类型和数组类型的情况。

2.6.std::is_trivially_destructible

判断一个类型T是否是一个平凡的可销毁类型(trivivally destructible)。主要用于检查这个类型的析构函数。一个trivivally destructible类(由class,struct/union)需要满足下面的条件:使用默认的析构函数、析构函数不能为虚的、它的基类和静态成员类型也必须是一个trivivally destructible类。如下示例:

// is_trivially_destructible example
#include <iostream>
#include <type_traits>struct A { }; /* 符合trivivally destructible类型定义 */
struct B { ~B(){} }; /* 没有使用隐式应答的析构函数, 即编译器合成的默认析构函数, 因此不是trivivally destructible类型 */int main() {std::cout << std::boolalpha; /* 将输出流bool解析为true/false, 而不是1/0 */std::cout << "is_trivially_destructible:" << std::endl;std::cout << "int: " << std::is_trivially_destructible<int>::value << std::endl; /* 基本类型是trivivally destructible类型 */std::cout << "A: " << std::is_trivially_destructible<A>::value << std::endl; /* A是trivivally destructible类型 */std::cout << "B: " << std::is_trivially_destructible<B>::value << std::endl; /* B不是trivivally destructible类型 */return 0;
}

输出:

is_trivially_destructible:
int: true
A: true
B: false

2.7.std::in_place_type和std::in_place_index

std::in_place_inde实际就是一个占位符,它的定义如下:

template <size_t _Idx>
inline constexpr in_place_index_t<_Idx> in_place_index{};

   in_place_index_t 定义如下:

template <size_t>
struct in_place_index_t { // tag that selects the index of a type to construct in placeexplicit in_place_index_t() = default;
};

从上面的代码可以看出 std::in_place_inde<_Idx> 是用来标识参数位置的数据类型,不过它是根据参数位置序号来判断的;同理也可以分析出std::in_place_type<_Ty>也是用来标识参数位置的数据类型,不过它是根据参数的类型来判断的,从std::in_place_type的定义可以看出来:

struct in_place_t { // tag used to select a constructor which initializes a contained object in placeexplicit in_place_t() = default;
};
inline constexpr in_place_t in_place{};template <class>
struct in_place_type_t { // tag that selects a type to construct in placeexplicit in_place_type_t() = default;
};
template <class _Ty>
inline constexpr in_place_type_t<_Ty> in_place_type{};

3.原理分析

3.1.存储分析

std::variant的内部用了union递归存储各种类型的数据,在头文件variant中按码索骥找到了存储std::variant的类_Variant_storage,内部定义了一个union:

template <bool _TrivialDestruction, class... _Types>
class _Variant_storage_ {}; // empty storage (empty "_Types" case)// ALIAS TEMPLATE _Variant_storage
template <class... _Types>
using _Variant_storage = _Variant_storage_<conjunction_v<is_trivially_destructible<_Types>...>, _Types...>;template <class _First, class... _Rest>
class _Variant_storage_<true, _First, _Rest...> { // Storage for variant alternatives (trivially destructible case)
public:static constexpr size_t _Size = 1 + sizeof...(_Rest);union {remove_const_t<_First> _Head;_Variant_storage<_Rest...> _Tail;};_Variant_storage_() noexcept {} // no initialization (no active member)...
};

union自动按最大数据类型对齐的。std::variant的内存布局为:

第N个的_Tail为 _Variant_storage_<true> 或  _Variant_storage_<false>,举个例子,如定义

std::variant<int, double, bool, float> y; 那么y的内存布局如下所示:

_Variant_storage_根据对象是否为"简单销毁对象"划分为:

_Variant_storage_<true, _Types...> 和 _Variant_storage_<false, _Types...>,_Variant_storage_<true, _Types...>的实现为:

template <class... _Types>
using _Variant_storage = _Variant_storage_<conjunction_v<is_trivially_destructible<_Types>...>, _Types...>;template <class _First, class... _Rest>
class _Variant_storage_<true, _First, _Rest...> { // Storage for variant alternatives (trivially destructible case)
public:static constexpr size_t _Size = 1 + sizeof...(_Rest);union {remove_const_t<_First> _Head;_Variant_storage<_Rest...> _Tail;};_Variant_storage_() noexcept {} // no initialization (no active member)template <class... _Types>constexpr explicit _Variant_storage_(integral_constant<size_t, 0>, _Types&&... _Args) noexcept(is_nothrow_constructible_v<_First, _Types...>): _Head(static_cast<_Types&&>(_Args)...) {} // initialize _Head with _Args...template <size_t _Idx, class... _Types, enable_if_t<(_Idx > 0), int> = 0>constexpr explicit _Variant_storage_(integral_constant<size_t, _Idx>, _Types&&... _Args) noexcept(is_nothrow_constructible_v<_Variant_storage<_Rest...>, integral_constant<size_t, _Idx - 1>, _Types...>): _Tail(integral_constant<size_t, _Idx - 1>{}, static_cast<_Types&&>(_Args)...) {} // initialize _Tail (recurse)_NODISCARD constexpr _First& _Get() & noexcept {return _Head;}_NODISCARD constexpr const _First& _Get() const& noexcept {return _Head;}_NODISCARD constexpr _First&& _Get() && noexcept {return _STD move(_Head);}_NODISCARD constexpr const _First&& _Get() const&& noexcept {return _STD move(_Head);}
};

_Variant_storage_<false, _Types...>的实现为:

template <class _First, class... _Rest>
class _Variant_storage_<false, _First, _Rest...> { // Storage for variant alternatives (non-trivially destructible case)
public:static constexpr size_t _Size = 1 + sizeof...(_Rest);union {remove_const_t<_First> _Head;_Variant_storage<_Rest...> _Tail;};~_Variant_storage_() noexcept { // explicitly non-trivial destructor (which would otherwise be defined as deleted// since the class has a variant member with a non-trivial destructor)}_Variant_storage_() noexcept {} // no initialization (no active member)template <class... _Types>constexpr explicit _Variant_storage_(integral_constant<size_t, 0>, _Types&&... _Args) noexcept(is_nothrow_constructible_v<_First, _Types...>): _Head(static_cast<_Types&&>(_Args)...) {} // initialize _Head with _Args...template <size_t _Idx, class... _Types, enable_if_t<(_Idx > 0), int> = 0>constexpr explicit _Variant_storage_(integral_constant<size_t, _Idx>, _Types&&... _Args) noexcept(is_nothrow_constructible_v<_Variant_storage<_Rest...>, integral_constant<size_t, _Idx - 1>, _Types...>): _Tail(integral_constant<size_t, _Idx - 1>{}, static_cast<_Types&&>(_Args)...) {} // initialize _Tail (recurse)_Variant_storage_(_Variant_storage_&&)      = default;_Variant_storage_(const _Variant_storage_&) = default;_Variant_storage_& operator=(_Variant_storage_&&) = default;_Variant_storage_& operator=(const _Variant_storage_&) = default;_NODISCARD constexpr _First& _Get() & noexcept {return _Head;}_NODISCARD constexpr const _First& _Get() const& noexcept {return _Head;}_NODISCARD constexpr _First&& _Get() && noexcept {return _STD move(_Head);}_NODISCARD constexpr const _First&& _Get() const&& noexcept {return _STD move(_Head);}
};

_Variant_storage_类中提供了对外访问对象的接口 _Get(),包括左值引用和右值引用。由于上层的类_Variant_base是private继承_Variant_storage_的,从下面的代码可以看出:

template <class... _Types>
class _Variant_base: private _Variant_storage<_Types...> { // Associate an integral discriminator with a _Variant_storage
public:using _Index_t                       = _Variant_index_t<sizeof...(_Types)>;static constexpr auto _Invalid_index = static_cast<_Index_t>(-1);_Index_t _Which;using _Storage_t = _Variant_storage<_Types...>;_NODISCARD constexpr _Storage_t& _Storage() & noexcept { // access this variant's storagereturn *this;}_NODISCARD constexpr const _Storage_t& _Storage() const& noexcept { // access this variant's storagereturn *this;}_NODISCARD constexpr _Storage_t&& _Storage() && noexcept { // access this variant's storagereturn _STD move(*this);}_NODISCARD constexpr const _Storage_t&& _Storage() const&& noexcept { // access this variant's storagereturn _STD move(*this);}_Variant_base() noexcept : _Storage_t{}, _Which{_Invalid_index} {} // initialize to the value-less state...
};

由于上层的类不能访问_Variant_storage_的成员变量和函数,所以提供了专门的访问数据接口_Variant_raw_get,代码如下:

template <size_t _Idx, class _Storage>
_NODISCARD constexpr decltype(auto) _Variant_raw_get(_Storage&& _Obj) noexcept { // access the _Idx-th element of a _Variant_storageif constexpr (_Idx == 0) {return static_cast<_Storage&&>(_Obj)._Get();} else if constexpr (_Idx == 1) {return static_cast<_Storage&&>(_Obj)._Tail._Get();} else if constexpr (_Idx == 2) {return static_cast<_Storage&&>(_Obj)._Tail._Tail._Get();} else if constexpr (_Idx == 3) {return static_cast<_Storage&&>(_Obj)._Tail._Tail._Tail._Get();} else if constexpr (_Idx == 4) {return static_cast<_Storage&&>(_Obj)._Tail._Tail._Tail._Tail._Get();} else if constexpr (_Idx == 5) {return static_cast<_Storage&&>(_Obj)._Tail._Tail._Tail._Tail._Tail._Get();} else if constexpr (_Idx == 6) {return static_cast<_Storage&&>(_Obj)._Tail._Tail._Tail._Tail._Tail._Tail._Get();} else if constexpr (_Idx == 7) {return static_cast<_Storage&&>(_Obj)._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Get();} else if constexpr (_Idx < 16) {return _Variant_raw_get<_Idx - 8>(static_cast<_Storage&&>(_Obj)._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail);} else if constexpr (_Idx < 32) {return _Variant_raw_get<_Idx - 16>(static_cast<_Storage&&>(_Obj)._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail);} else if constexpr (_Idx < 64) {return _Variant_raw_get<_Idx - 32>(static_cast<_Storage&&>(_Obj)._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail);} else { // _Idx >= 64return _Variant_raw_get<_Idx - 64>(static_cast<_Storage&&>(_Obj)._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail._Tail);}
}

函数中参数序号低于8的直接访问值,大于8的递归调用自身来访问值。

3.2.构造函数

3.2.1.默认构造

默认构造函数如下:

template <class _First = _Meta_front<variant>, enable_if_t<is_default_constructible_v<_First>, int> = 0>constexpr variant() noexcept(is_nothrow_default_constructible_v<_First>): _Mybase(in_place_index<0>) {} // value-initialize alternative 0

取出第一个参数_Meta_front,调用基类的构造函数,生成对象。象如下定义std::variant就会调用此构造函数:

std::variant<int, double, bool, float> y;

3.2.2.使用单一值初始化

  示例如下:

std::variant<bool, int, std::string> v(25);

如果这样编码,就会直接调用std::variant的单一赋值的构造函数,源码如下:

template <class _Ty,enable_if_t<sizeof...(_Types) != 0 //&& !is_same_v<_Remove_cvref_t<_Ty>, variant> //&& !_Is_specialization_v<_Remove_cvref_t<_Ty>, in_place_type_t> //&& !_Is_in_place_index_specialization<_Remove_cvref_t<_Ty>> //&& is_constructible_v<_Variant_init_type<_Ty, _Types...>, _Ty>, //int> = 0>
constexpr variant(_Ty&& _Obj) noexcept(is_nothrow_constructible_v<_Variant_init_type<_Ty, _Types...>, _Ty>): _Mybase(in_place_index<_Variant_init_index<_Ty, _Types...>::value>, static_cast<_Ty&&>(_Obj)) {// initialize to the type selected by passing _Obj to the overload set f(Types)...}

通过_Variant_init_index找到_Ty在_Types...的位置,然后再调用_Variant_base的构造函数:

template <size_t _Idx, class... _UTypes,enable_if_t<is_constructible_v<_Meta_at_c<variant<_Types...>, _Idx>, _UTypes...>, int> = 0>constexpr explicit _Variant_base(in_place_index_t<_Idx>, _UTypes&&... _Args) noexcept(is_nothrow_constructible_v<_Meta_at_c<variant<_Types...>, _Idx>, _UTypes...>): _Storage_t(integral_constant<size_t, _Idx>{}, static_cast<_UTypes&&>(_Args)...),_Which{static_cast<_Index_t>(_Idx)} { // initialize alternative _Idx from _Args...}

初始化_Which和_Variant_storage,如果有多个可能的类型匹配,可能导致歧义。

3.2.3.std::in_place_type

示例如下:

std::variant<int, double, std::string> v(std::in_place_type<double>, 34.66);

如果这样编码,就会直接调用std::variant的std::in_place_type构造函数,源码如下:

template <class _Ty, class... _UTypes, class _Idx = _Meta_find_unique_index<variant, _Ty>,enable_if_t<_Idx::value != _Meta_npos && is_constructible_v<_Ty, _UTypes...>, int> = 0>constexpr explicit variant(in_place_type_t<_Ty>, _UTypes&&... _Args) noexcept(is_nothrow_constructible_v<_Ty, _UTypes...>) // strengthened: _Mybase(in_place_index<_Idx::value>, static_cast<_UTypes&&>(_Args)...) {// initialize alternative _Ty from _Args...}

通过_Meta_find_unique_index找到_Ty在_Types...的位置,_Meta_find_unique_index循环递归查找_Ty的详细实现:

template <class _List, class _Ty>
struct _Meta_find_unique_index_ {using type = integral_constant<size_t, _Meta_npos>;
};
template <class _List, class _Ty>
using _Meta_find_unique_index =// The index of _Ty in _List if it occurs exactly once, otherwise _Meta_npostypename _Meta_find_unique_index_<_List, _Ty>::type;constexpr size_t _Meta_find_unique_index_i_2(const bool* const _Ptr, const size_t _Count,const size_t_First) { // return _First if there is no _First < j < _Count such that _Ptr[j] is true, otherwise _Meta_nposreturn _First != _Meta_npos && _Meta_find_index_i_(_Ptr, _Count, _First + 1) == _Meta_npos ? _First : _Meta_npos;
}constexpr size_t _Meta_find_unique_index_i_(const bool* const _Ptr,const size_t _Count) { // Pass the smallest i such that _Ptr[i] is true to _Meta_find_unique_index_i_2return _Meta_find_unique_index_i_2(_Ptr, _Count, _Meta_find_index_i_(_Ptr, _Count));
}template <template <class...> class _List, class _First, class... _Rest, class _Ty>
struct _Meta_find_unique_index_<_List<_First, _Rest...>, _Ty> {using type = integral_constant<size_t,_Meta_find_unique_index_i_(_Meta_find_index_<_List<_First, _Rest...>, _Ty>::_Bools, 1 + sizeof...(_Rest))>;
};

最后调用_Variant_base的构造函数,生成_Variant_storage,存储数据。

3.2.4.std::in_place_index

示例如下:

std::variant<bool, std::string> v(std::in_place_index<1>, "14256435");

如果这样编码,就会直接调用std::variant的std::in_place_index构造函数,源码如下:

template <size_t _Idx, class... _UTypes,enable_if_t<is_constructible_v<_Meta_at_c<variant, _Idx>, _UTypes...>, int> = 0>constexpr explicit variant(in_place_index_t<_Idx>, _UTypes&&... _Args) noexcept(is_nothrow_constructible_v<_Meta_at_c<variant, _Idx>, _UTypes...>) // strengthened: _Mybase(in_place_index<_Idx>, static_cast<_UTypes&&>(_Args)...) {// initialize alternative _Idx from _Args...}

直接调用_Variant_base的构造函数,这种生成std::vaiant流程会简单一些,也比较好理解一些。

3.3.访问值

3.3.1.直接赋值

如:

std::variant<bool, int,std::string> v;
v = "hello world"; 

如果这样编码,就会直接调用std::variant的operator=,源码如下:

// assignment [variant.assign]template <class _Ty, enable_if_t<!is_same_v<_Remove_cvref_t<_Ty>, variant> //&& is_constructible_v<_Variant_init_type<_Ty, _Types...>, _Ty> //&& is_assignable_v<_Variant_init_type<_Ty, _Types...>&, _Ty>, //int> = 0>variant& operator=(_Ty&& _Obj) noexcept(is_nothrow_assignable_v<_Variant_init_type<_Ty, _Types...>&, _Ty>&&is_nothrow_constructible_v<_Variant_init_type<_Ty, _Types...>, _Ty>) {// assign/emplace the alternative chosen by overload resolution of _Obj with f(_Types)...constexpr size_t _TargetIdx = _Variant_init_index<_Ty, _Types...>::value;if (index() == _TargetIdx) {auto& _Target = _Variant_raw_get<_TargetIdx>(_Storage());_Target       = static_cast<_Ty&&>(_Obj);} else {using _TargetTy = _Variant_init_type<_Ty, _Types...>;if constexpr (_Variant_should_directly_construct_v<_TargetTy, _Ty>) {this->_Reset();_Emplace_valueless<_TargetIdx>(static_cast<_Ty&&>(_Obj));} else {_TargetTy _Temp(static_cast<_Ty&&>(_Obj));this->_Reset();_Emplace_valueless<_TargetIdx>(_STD move(_Temp));}}return *this;}

它的流程如下:

关键步骤:1)比较当前的index()和_TargetIdx是否相同,相同,直接赋值。

2)不相同,这需要析构原来的对象,重新构造新的对象,赋值等操作,流程会比较复杂一些,这些实现是在_Emplace_valueless里面,代码如下:

template <size_t _Idx, class... _ArgTypes>_Meta_at_c<variant, _Idx>& _Emplace_valueless(_ArgTypes&&... _Args) noexcept(is_nothrow_constructible_v<_Meta_at_c<variant, _Idx>, _ArgTypes...>) {// initialize alternative _Idx from _Args...// pre: valueless_by_exception()auto& _Obj = _Variant_raw_get<_Idx>(_Storage());_Construct_in_place(_Obj, static_cast<_ArgTypes&&>(_Args)...);this->_Set_index(_Idx);return _Obj;}

3.3.2.emplace

从emplace的源代码

template <class _Ty, class... _ArgTypes, size_t _Idx = _Meta_find_unique_index<variant, _Ty>::value,enable_if_t<_Idx != _Meta_npos && is_constructible_v<_Ty, _ArgTypes...>, int> = 0>_Ty& emplace(_ArgTypes&&... _Args) noexcept(is_nothrow_constructible_v<_Ty, _ArgTypes...>) /* strengthened */ {// emplace alternative _Ty from _Args...this->_Reset();return _Emplace_valueless<_Idx>(static_cast<_ArgTypes&&>(_Args)...);}

可以看出跟3.3.1的流程差不多,这里就不多赘述了。

3.3.3.get

1) 通过序号 index 来get值

template <size_t _Idx, class... _Types>
_NODISCARD constexpr decltype(auto) get(variant<_Types...>& _Var) { // access the contained value of _Var if its _Idx-th alternative is activestatic_assert(_Idx < sizeof...(_Types), "variant index out of bounds");if (_Var.index() == _Idx) {return _Variant_raw_get<_Idx>(_Var._Storage());}_Throw_bad_variant_access();
}

 通过当前的index()和_Idx比对,来获取std::variant的值,_Variant_raw_get函数在3.1章节讲过;

如果当前的index()不是_Idx,这会抛出异常。

2)通过 类型_Ty 来get值

template <class _Ty, class... _Types>
_NODISCARD constexpr decltype(auto) get(variant<_Types...>& _Var) { // access the contained value of _Var if its alternative _Ty is activeconstexpr size_t _Idx = _Meta_find_unique_index<variant<_Types...>, _Ty>::value;static_assert(_Idx < sizeof...(_Types),"get<T>(variant<Types...>&) requires T to occur exactly once in Types. (N4835 [variant.get]/5)");return _STD get<_Idx>(_Var);
}

通过_Meta_find_unique_index获取到类型_Ty的_Idx, 然后调用序号index版本的get来获取值。

3.3.4.get_if

get_if也是有两种,通过序号index和类型_Ty来获取值,从源码的

template <size_t _Idx, class... _Types>
_NODISCARD constexpr auto get_if(variant<_Types...>* _Ptr) noexcept { // get the address of *_Ptr's contained value if it holds alternative _Idxstatic_assert(_Idx < sizeof...(_Types), "variant index out of bounds");return _Ptr && _Ptr->index() == _Idx ? _STD addressof(_Variant_raw_get<_Idx>(_Ptr->_Storage())) : nullptr;
}

template <class _Ty, class... _Types>
_NODISCARD constexpr add_pointer_t<_Ty> get_if(variant<_Types...>* _Ptr) noexcept { // get the address of *_Ptr's contained value if it holds alternative _Tyconstexpr size_t _Idx = _Meta_find_unique_index<variant<_Types...>, _Ty>::value;static_assert(_Idx != _Meta_npos,"get_if<T>(variant<Types...> *) requires T to occur exactly once in Types. (N4835 [variant.get]/9)");return _STD get_if<_Idx>(_Ptr);
}

可以看到,基本上可以3.3.3的get原理差不多,在这里就不多赘述了。

4.总结

到此我们已经全部分析完毕,细节也谈及了,喜欢的给个赞并收藏,谢谢。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/630903.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Jira 宣布Data Center版涨价5%-15%,6年内第8次提价

近日&#xff0c;Atlassian官方面向合作伙伴发布2024年涨价通知&#xff1a; 自2024年2月15日起&#xff0c;旗下核心产品Jira Software、Confluence、Jira Service Management的DC版本&#xff08;Data Center版本&#xff09;价格提高5%-15%&#xff08;涨幅与坐席数阶梯相关…

Django中使用MySQL的视图View

文章目录 一、MySQL创建视图二、Django模型定义 一、MySQL创建视图 首先确定好自己需要链接的两张表及其对应的字段&#xff0c;视图的意义是将两张表数据联合一起变成一张新的表方便后续查询数据。 源Django两张表模型 class ProjectTaskRange(BaseModel, TimeModel):class …

IOS-高德地图SDK接入-Swift

申请key 这个要前往高德开发平台注册成为个人开发者然后在控制台创建一个应用&#xff1a; 高德开发平台 注册步骤就不写了&#xff0c;写一下创建应用的步骤&#xff1a; 1、点击应用管理——>我的应用 2、点击右上角的创建新应用 3、输入内容&#xff1a; 4、点击添加ke…

新定义51单片机(RD8G37)实现测距测速仪

本文描述用新定义51单片机&#xff08;RD8G37&#xff09;超声波一体测距传感器实现简单的测距测速仪。 测距仪演示效果 新定义RD8G37Q48RJ开发板 超声波测距模块&#xff1a; 8位并口屏 1、main.c unsigned short timeConsuming0; unsigned int oldDistance;void rectClearS…

阿里云国外服务器价格表

阿里云国外服务器优惠活动「全球云服务器精选特惠」&#xff0c;国外服务器租用价格24元一个月起&#xff0c;免备案适合搭建网站&#xff0c;部署独立站等业务场景&#xff0c;阿里云服务器网aliyunfuwuqi.com分享阿里云国外服务器优惠活动&#xff1a; 全球云服务器精选特惠…

web terminal - 如何在mac os上运行gotty

gotty可以让你使用web terminal的方式与环境进行交互&#xff0c;实现终端效果 假设你已经配置好了go环境&#xff0c;首先使用go get github.com/yudai/gotty命令获取可执行文件&#xff0c;默认会安装在$GOPATH/bin这个目录下&#xff0c;注意如果你的go版本比较高&#xff…

大数据深度学习ResNet深度残差网络详解:网络结构解读与PyTorch实现教程

文章目录 大数据深度学习ResNet深度残差网络详解&#xff1a;网络结构解读与PyTorch实现教程一、深度残差网络&#xff08;Deep Residual Networks&#xff09;简介深度学习与网络深度的挑战残差学习的提出为什么ResNet有效&#xff1f; 二、深度学习与梯度消失问题梯度消失问题…

2024年腾讯云轻量服务器和CVM云服务器性能如何?

腾讯云轻量服务器和云服务器有什么区别&#xff1f;为什么轻量应用服务器价格便宜&#xff1f;是因为轻量服务器CPU内存性能比云服务器CVM性能差吗&#xff1f;轻量应用服务器适合中小企业或个人开发者搭建企业官网、博客论坛、微信小程序或开发测试环境&#xff0c;云服务器CV…

RaspberryPi(树莓派)配置 VNC

RaspberryPi&#xff08;树莓派&#xff09;是可以通过 VNC 来连接到机器上进行图形化操作的。 什么 VNC VNC&#xff08;Virtual Network Computing&#xff09;&#xff0c;为一种使用RFB协议的屏幕画面分享及远程操作软件。此软件借由网络&#xff0c;可发送键盘与鼠标的动…

node.js(express.js)+mysql实现注册功能

文章目录 实现步骤一、获取客户端提交到服务器的用户信息&#xff0c;对表单中的数据&#xff0c;进行合法性的效验 代码如下:二、检测用户名是否被占用三、对密码进行加密四、插入新用户&#xff08;完整代码&#xff09;总结 实现步骤 一、获取客户端提交到服务器的用户信息…

plc红绿灯程序

引言&#xff1a; PLC&#xff08;Programmable Logic Controller&#xff0c;可编程逻辑控制器&#xff09;是一种用于工业自动化控制的电子设备。西门子的SIMATIC S7-200是这类设备的一个流行系列&#xff0c;广泛应用于小型至中等规模的自动化项目中。它具有以下特点&#…

常见的设计模式(模板与方法,观察者模式,策略模式)

前言 随着时间的推移&#xff0c;软件代码越来越庞大&#xff0c;随着而来的就是如何维护日趋庞大的软件系统。在面向对象开发出现之前&#xff0c;使用的是面向过程开发来设计大型的软件程序&#xff0c;面向过程开发将软件分成一个个单独的模块&#xff0c;模块之间使用函数…

[足式机器人]Part2 Dr. CAN学习笔记-Advanced控制理论 Ch04-17 串讲

本文仅供学习使用 本文参考&#xff1a; B站&#xff1a;DR_CAN Dr. CAN学习笔记-Advanced控制理论 Ch04-17 串讲

7.6 MySQL基本函数的使用(❤❤❤)

7.6 MySQL基本函数的使用 1. 提要2. 数字函数3. 字符函数3.1 替换字符3.2 左填充字符及截取字符串 4. 日期函数4.1 日期函数4.2 表达式占位符4.3 日期偏移计算4.4 日期间隔 5. 条件函数5.1 IF语句5.2 case...when语句 1. 提要 2. 数字函数 3. 字符函数 3.1 替换字符 -- INSERT…

[剪藏] - 任泽平年终演讲精华:点燃希望——2024中国经济十大预测

任泽平年终演讲精华&#xff1a;点燃希望——2024中国经济十大预测 泽平宏观 2023-12-23 08:01 发表于上海 12月22日22:30&#xff0c;任泽平年终秀“点燃希望乐观者前行——2024中国经济十大预测”圆满收官。 泽平宏观、北京广播电视台、上海高净值研究院、北京时间等携手打…

Gorm 应用开发时区问题与unique唯一索引字段数据冲突问题

文章目录 一、定义表模型时区问题1.1 time.Time 与int641.2 优势 二、unique唯一索引字段数据冲突问题 一、定义表模型时区问题 1.1 time.Time 与int64 一般情况下&#xff0c;我们在定义表模型的时候&#xff0c;会使用time.Time&#xff0c;但是会根据当前时间存储。返回给…

现代密码学 考点汇总(下)

现代密码学 考点汇总 写在最前面考试范围一、给一个简单的方案&#xff0c;判断是否cca安全二、随机预言机模型之下的简单应用 2. MAC概念回顾MAC的定义适应性CMA&#xff08;Chosen Message Attack&#xff09;PPT攻击者不可忽略的概率&#xff08;negl(n)&#xff09;总结 案…

three.js 点按钮,相机飞行靠近观察设备

效果&#xff1a; 代码&#xff1a; <template><div><el-container><el-main><div class"box-card-left"><div id"threejs" style"border: 1px solid red"></div><div class"box-right&quo…

vue2+webpack升级vue3+vite,报错Cannot read properties of null (reading ‘isCE‘)

同学们可以私信我加入学习群&#xff01; 正文开始 前言问题分析解决总结 前言 系列文章&#xff1a;vue2webpack升级vue3vite&#xff0c;修改插件兼容性bug 前面的文章主要是介绍&#xff0c;在升级初始阶段遇到的一些显而易见的兼容性问题和bug。随着项目迭代的不断深入&a…

视频美颜SDK技术解析与技术对比

当下&#xff0c;各类应用和服务纷纷采用视频美颜SDK&#xff0c;以提供更加令人满意的视觉效果。本文将深入探讨视频美颜SDK的技术原理&#xff0c;同时对比不同SDK的特性&#xff0c;为开发者和决策者提供全面的技术参考。 一、技术原理解析 1.图像处理基础 视频美颜SDK基…