我看这个grate code based on boost.Any并且无法帮助,但想知道我们是否可以使用Boost.Variant。我想知道这样的API是否可行:
void voidFunc()
{
std::cout << "void called" << std::endl;
}
int stringFunc(std::string str)
{
std::cout << str << std::endl;
return 0;
}
int main()
{
some_map_like_type<std::string, boost::variant> funcs;
funcs.insert<void , void >("voidFunc", &voidFunc)); // now our variant vould contain something like boost::function<void, void>
funcs.insert<int , std::string>("stringFunc", &stringFunc)); // and now we added to our variant a new type: boost::function<int , std::string>
funcs.insert<void , void >("voidFunc2", &voidFunc)); // and now our variant should not change because it already contains boost::function<void, void> type
// And here when all the fun part is:
funcs["voidFunc"](); // compiles
funcs["stringFunc"]("hello"); // compiles
funcs["stringFunc"](some_not_std_string_class); // does not compile.
return 0;
}
这意味着最后编译器必须编译类似:
void voidFunc()
{
std::cout << "void called" << std::endl;
}
int stringFunc(std::string str)
{
std::cout << str << std::endl;
return 0;
}
int main()
{
some_map_like_type<std::string, boost::variant< boost::function<void , void>, boost::function<int , std::string> > > funcs;
funcs.insert<void , void >("voidFunc", &voidFunc)); // now our variant vould contain something like boost::function<void, void>
funcs.insert<int , std::string>("stringFunc", &stringFunc)); // and now we added to our variant a new type: boost::function<int , std::string>
funcs.insert<void , void >("voidFunc2", &voidFunc)); // and now our variant should not change because it already contains boost::function<void, void> type
// And here when all the fun part is:
funcs["voidFunc"](); // compiles
funcs["stringFunc"]("hello"); // compiles
funcs["stringFunc"](some_not_std_string_class); // here it would give error and would not compile
return 0;
}
我尝试了什么(基于此Variant docs以及此MPL demos和docs):
#include <boost/static_assert.hpp>
#include <boost/mpl/equal.hpp>
#include <boost/mpl/vector_c.hpp>
#include <boost/mpl/transform.hpp>
#include <boost/mpl/multiplies.hpp>
#include <boost/mpl/placeholders.hpp>
#include <boost/variant.hpp>
#include <iostream>
#include <string>
#include <vector>
class sudo_science
{
public:
typedef boost::mpl::vector_c<int> types_vector1;
typedef boost::make_recursive_variant< types_vector1 >::type recursive_variant_t;
std::vector< recursive_variant_t > variant_seq;
template <typename T>
void append(T val)
{
typedef boost::mpl::push_back<types_vector1,T>::type types_vector1;
variant_seq.push_back(val);
return;
}
std::vector< recursive_variant_t > give_me_end_variant()
{
return variant_seq;
}
};
int main()
{
sudo_science a;
a.append<float>(1.0);
a.append<std::string>("Stack and Boost");
//sorry for C++11
auto varint = a.give_me_end_variant();
return 0;
}
但它无法编译,但有两个相同的错误:
Error 1 error C2665: 'boost::detail::variant::make_initializer_node::apply<BaseIndexPair,Iterator>::initializer_node::initialize' : none of the 2 overloads could convert all the argument types c:\program files\boost\include\boost\variant\variant.hpp 1330 1
答案 0 :(得分:-1)
这是不可能的。 operator[]
是一个运行时的东西,而类型是编译时的东西。那么编译器应该编译以下内容吗?
char const* str;
if (some_condition())
str = "voidFunc";
else
str = "stringFunc";
// ... some more code
if (some_condition())
funcs[str]();
else
funcs[str](str);
编译器如何知道对some_condition()
的第二次调用是否给出与以前相同的结果?或者两者之间的代码是否修改了str
的值?
以下内容如何:
void call(some_map_like_type<std::string, boost::variant> const& funcs)
{
funcs["voidFunc"]();
}
编译器如何知道在调用时funcs
是否包含一个条目映射"voidFunc"
到没有参数的函数?如果一次使用一个值调用它,并且一次使用一个没有的值,那会发生什么呢?
根据您实际想要实现的目标,可能有办法通过模板和constexpr
功能获取它。但请注意,在运行时发生的任何事情都不会影响代码是否编译,原因很简单,代码在编译之前无法运行。