首页 > 代码库 > boost::tie()和boost::variant()讲解
boost::tie()和boost::variant()讲解
#include<iostream>
#include<boost/tuple/tuple.hpp>
#include<boost/variant.hpp>
#include<boost/tuple/tuple_io.hpp>
#include<boost/any.hpp>
#include<vector>
#include<iterator>
#include<string>
using namespace std;
boost::tuple<std::string, int> func()
{
return boost::make_tuple("Error message", 2009);
}
std::vector<boost::any> vector1;
struct output : public boost::static_visitor<>
{
template <typename T>
void operator()(T &t) const
{
vector1.push_back(t);
}
};
int main(){
//Boost.Tuple 还提供了一种叫做 Tier 的特殊元组。 Tier 的特殊之处在于它包含的所有元素都是引用类型的。 它可以通过构造函数 boost::tie() 来创建。
int i; char c; double d;
boost::tie(i,c,d) = boost::tuple<int,char,double>(1,‘A‘,0.618);
d = 0.718;
cout<<d<<endl;
//用ignore忽略元素
char c2;
boost::tie(boost::tuples::ignore,c2) = std::make_pair(1,‘a‘);
cout<<c2<<endl;
//boost::tie() 在一定程度上简化了语法, 同时, 也可以用作“拆箱”元组。 在接下来的这个例子里, 元组中的各个元素就被很方便的“拆箱”并直接赋给了其他变量。
std::string errmsg;
int errcode;
boost::tie(errmsg, errcode) = func();
std::cout << errmsg << ": " << errcode << std::endl;
//Boost.Variant 为我们提供了一个定义在 boost/variant.hpp 中的类: boost::variant 。 既然 boost::variant 是一个模板, 你必须要指定至少一个参数。 Variant 所存储的数据类型就由这些参数来指定。 上面的例子就给 v 指定了 double 类型和 char 类型。 注意, 一旦你将一个 int 值赋给了 v, 你的代码将不会编译通过。
//当然, 上面的例子也可以用一个 union 类型来实现, 但是与 union 不同的是: boost::variant 可以储存像 std::string 这样的 class 类型的数据。
//boost::apply_visitor() 第一个参数需要传入一个继承自 boost::static_visitor 类型的对象。 这个类必须要重载 operator()() 运算符来处理 boost::variant 每个可能的类型。 相应的, 例子中的 v 就重载了三次 operator() 来处理三种可能的类型: double, char 和 std::string。
//再仔细看代码, 不难发现 boost::static_visitor 是一个模板。 那么,当 operator()() 有返回值的时候, 就必须返回一个模板才行。 如果 operator() 像例子那样没有返回值时, 你就不需要模板了。
//boost::apply_visitor() 的第二个参数是一个 boost::variant 类型的值。
//在使用时, boost::apply_visitor() 会自动调用跟第二个参数匹配的 operator()() 。 示例程序中的 boost::apply_visitor() 就自动调用了三个不同的 operator 第一个是 double 类型的, 第二个是 char 最后一个是 std::string。
//boost::apply_visitor() 的优点不只是“自动调用匹配的函数”这一点。 更有用的是, boost::apply_visitor() 会确认是否 boost::variant 中的每个可能值都定义了相应的函数。 如果你忘记重载了任何一个函数, 代码都不会编译通过。
boost::variant<double, char, std::string> v;
v = 3.14;
boost::apply_visitor(output(), v);
v = ‘A‘;
boost::apply_visitor(output(), v);
v = "Hello, world!";
boost::apply_visitor(output(), v);
}
#include<boost/tuple/tuple.hpp>
#include<boost/variant.hpp>
#include<boost/tuple/tuple_io.hpp>
#include<boost/any.hpp>
#include<vector>
#include<iterator>
#include<string>
using namespace std;
boost::tuple<std::string, int> func()
{
return boost::make_tuple("Error message", 2009);
}
std::vector<boost::any> vector1;
struct output : public boost::static_visitor<>
{
template <typename T>
void operator()(T &t) const
{
vector1.push_back(t);
}
};
int main(){
//Boost.Tuple 还提供了一种叫做 Tier 的特殊元组。 Tier 的特殊之处在于它包含的所有元素都是引用类型的。 它可以通过构造函数 boost::tie() 来创建。
int i; char c; double d;
boost::tie(i,c,d) = boost::tuple<int,char,double>(1,‘A‘,0.618);
d = 0.718;
cout<<d<<endl;
//用ignore忽略元素
char c2;
boost::tie(boost::tuples::ignore,c2) = std::make_pair(1,‘a‘);
cout<<c2<<endl;
//boost::tie() 在一定程度上简化了语法, 同时, 也可以用作“拆箱”元组。 在接下来的这个例子里, 元组中的各个元素就被很方便的“拆箱”并直接赋给了其他变量。
std::string errmsg;
int errcode;
boost::tie(errmsg, errcode) = func();
std::cout << errmsg << ": " << errcode << std::endl;
//Boost.Variant 为我们提供了一个定义在 boost/variant.hpp 中的类: boost::variant 。 既然 boost::variant 是一个模板, 你必须要指定至少一个参数。 Variant 所存储的数据类型就由这些参数来指定。 上面的例子就给 v 指定了 double 类型和 char 类型。 注意, 一旦你将一个 int 值赋给了 v, 你的代码将不会编译通过。
//当然, 上面的例子也可以用一个 union 类型来实现, 但是与 union 不同的是: boost::variant 可以储存像 std::string 这样的 class 类型的数据。
//boost::apply_visitor() 第一个参数需要传入一个继承自 boost::static_visitor 类型的对象。 这个类必须要重载 operator()() 运算符来处理 boost::variant 每个可能的类型。 相应的, 例子中的 v 就重载了三次 operator() 来处理三种可能的类型: double, char 和 std::string。
//再仔细看代码, 不难发现 boost::static_visitor 是一个模板。 那么,当 operator()() 有返回值的时候, 就必须返回一个模板才行。 如果 operator() 像例子那样没有返回值时, 你就不需要模板了。
//boost::apply_visitor() 的第二个参数是一个 boost::variant 类型的值。
//在使用时, boost::apply_visitor() 会自动调用跟第二个参数匹配的 operator()() 。 示例程序中的 boost::apply_visitor() 就自动调用了三个不同的 operator 第一个是 double 类型的, 第二个是 char 最后一个是 std::string。
//boost::apply_visitor() 的优点不只是“自动调用匹配的函数”这一点。 更有用的是, boost::apply_visitor() 会确认是否 boost::variant 中的每个可能值都定义了相应的函数。 如果你忘记重载了任何一个函数, 代码都不会编译通过。
boost::variant<double, char, std::string> v;
v = 3.14;
boost::apply_visitor(output(), v);
v = ‘A‘;
boost::apply_visitor(output(), v);
v = "Hello, world!";
boost::apply_visitor(output(), v);
}
编译后输出:
0.718
a
Error message: 2009
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。