c++ - 为什么我的变体将 std::string 转换为 bool?

标签 c++ c++17 variant

我的std::variant可以为空(std::monostate),包含一个int,一个std::字符串bool

当我想用 var = "this is my string" 给出的字符串来输入它时,它会被转换为 bool 而不是字符串。如果我明确声明类型,它会工作 var = std::string("this is my string")。为什么会这样?有什么办法可以避免吗?

#include <string>
#include <variant>
#include <iostream>

int main()
{
    using var = std::variant<std::monostate, int, std::string, bool>;

    var contains_nothing;    
    var contains_int = 5;
    var contains_string = "hello";
    var contains_expl_string = std::string("explicit hello");
    var contains_bool = false;

    auto visitor = [](auto&& arg){
            using T = std::decay_t<decltype(arg)>;
            if constexpr (std::is_same<T, std::monostate>())
                std::cout<<"nothing\n";
            else if constexpr (std::is_same<T, int>())  
                std::cout<<"int: "<<arg<<"\n";
            else if constexpr (std::is_same<T, std::string>())  
                std::cout<<"string: "<<arg<<"\n";
            else if constexpr (std::is_same<T, bool>())  
                std::cout<<"bool: "<<arg<<"\n";
            else 
                std::cout<<"Visitor is not exhaustive\n";
        };

    std::visit(visitor, contains_nothing);       // nothing
    std::visit(visitor, contains_int);           // int: 5
    std::visit(visitor, contains_string);        // bool: 1
    std::visit(visitor, contains_expl_string);   // string: explicit hello
    std::visit(visitor, contains_bool);          // bool: 0    
}

编辑 由于我的代码的用户可能没有明确地制作 strings,我喜欢捕获它。否则,这将是错误的来源。我制作了模板助手函数来检查 char* 是否已通过,如果是,则创建 std::string。效果很好。感谢您帮助简化此过程!

编辑 2 通过将 std::monostate 声明为默认参数/类型,它甚至可以在没有任何参数的情况下调用 make_var

#include <string>
#include <variant>
#include <iostream>

using var = std::variant<std::monostate, int, std::string, bool>;

template<typename T = std::monostate>
var make_var(T value = std::monostate())
{
    if constexpr (std::is_same<typename std::remove_const<typename std::decay<T>::type>::type, const char*>())
        return std::string(value);
    return value;
}

int main()
{
    auto contains_nothing = make_var();    
    auto contains_int = make_var(3);
    auto contains_string = make_var("hello");
    auto contains_expl_string = make_var(std::string("excplicit hello"));
    var contains_bool = make_var(false);

    auto visitor = [](auto&& arg){
            using T = std::decay_t<decltype(arg)>;
            if constexpr (std::is_same<T, std::monostate>())
                std::cout<<"nothing\n";
            else if constexpr (std::is_same<T, int>())  
                std::cout<<"int: "<<arg<<"\n";
            else if constexpr (std::is_same<T, std::string>())  
                std::cout<<"string: "<<arg<<"\n";
            else if constexpr (std::is_same<T, bool>())  
                std::cout<<"bool: "<<arg<<"\n";
            else 
                std::cout<<"Visitor is not exhaustive\n";
        };

    std::visit(visitor, contains_nothing);
    std::visit(visitor, contains_int);
    std::visit(visitor, contains_string);
    std::visit(visitor, contains_expl_string);
    std::visit(visitor, contains_bool);    
}

最佳答案

"hello"的类型是const char [6],衰减为const char *const char *bool 的转换是内置转换,而 const char *std::string是用户自定义的转换,表示是前者。

由于您使用的是 C++ >= 14,因此您可以使用文字后缀 s 来表示 std::string 文字:

using namespace std::string_literals;

var contains_string = "hello"s;

关于c++ - 为什么我的变体将 std::string 转换为 bool?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44086269/

相关文章:

c++ - 无法在 SDL 2 中创建流畅的动画

c++计算编译时间常数,同时防止整数常数溢出

c++ - 基本 C++ 语法 : why is the namespace used before the variable name but not before the class type

c++ - 从 std::apply 中的折叠表达式捕获返回值

c++ - 为什么我们不能在内联命名空间之外声明演绎指南?

c++ - 具有自动转换支持的变体实现,例如 boost::any

arrays - 如何制作 UBound < LBound 的 VB6 变体数组?

delphi - 将 TMemoryStream 转换为变体

c++ - 代码块 : Unit test file requires entire address of a file

C++多菱形继承和纯虚函数