c++ - 为 boost::property_maps 中的枚举专门化 boost::lexical_cast

标签 c++ enums boost-graph lexical-cast boost-property-map

我正在尝试使用 boost::graph,它在其捆绑的顶点属性中有一个 enum。当我尝试将捆绑属性用于 boost::dynamic_property 时,问题就开始了。看起来我无法获得 boost::lexical_cast 的正确模板特化来识别类型。

#include <string>
#include <iostream>
#include <boost/lexical_cast.hpp>
#include <boost/graph/adjacency_list.hpp>
#include <boost/property_map/dynamic_property_map.hpp>

enum MyEnum { A, B, C };

namespace boost{
    template<>
    std::string lexical_cast(const MyEnum & m)
    {
        return std::to_string(int(m));
    }
} /* boost */

struct EnumWrapper
{
    MyEnum e;

    EnumWrapper() = default;
    EnumWrapper(MyEnum e) : e{e}
    {}
};

int main()
{
    using namespace boost;

    // This works
    MyEnum e{MyEnum::B};
    std::cout << lexical_cast<std::string>(e) << std::endl;

    // This doesn't compile
    adjacency_list<vecS, vecS, directedS, EnumWrapper> graph;
    dynamic_properties dp;
    dp.property("test", get(&EnumWrapper::e, graph));

    return 0;
}

如果我尝试直接编写 enum,一切正常。当尝试将其放入动态属性时,出现以下错误:

In file included from /usr/include/boost/iterator/iterator_categories.hpp:22:0,
                 from /usr/include/boost/iterator/iterator_facade.hpp:14,
                 from /usr/include/boost/range/iterator_range_core.hpp:27,
                 from /usr/include/boost/lexical_cast.hpp:30,
                 from main.cpp:3:
/usr/include/boost/lexical_cast/detail/converter_lexical.hpp: In instantiation of ‘struct boost::detail::deduce_target_char_impl<boost::detail::deduce_character_type_later<MyEnum> >’:
/usr/include/boost/lexical_cast/detail/converter_lexical.hpp:270:89:   required from ‘struct boost::detail::deduce_target_char<MyEnum>’
/usr/include/boost/lexical_cast/detail/converter_lexical.hpp:407:92:   required from ‘struct boost::detail::lexical_cast_stream_traits<std::__cxx11::basic_string<char>, MyEnum>’
/usr/include/boost/lexical_cast/detail/converter_lexical.hpp:468:15:   required from ‘struct boost::detail::lexical_converter_impl<MyEnum, std::__cxx11::basic_string<char> >’
/usr/include/boost/lexical_cast/try_lexical_convert.hpp:181:44:   required from ‘bool boost::conversion::detail::try_lexical_convert(const Source&, Target&) [with Target = MyEnum; Source = std::__cxx11::basic_string<char>]’
/usr/include/boost/lexical_cast.hpp:41:60:   required from ‘Target boost::lexical_cast(const Source&) [with Target = MyEnum; Source = std::__cxx11::basic_string<char>]’
/usr/include/boost/property_map/dynamic_property_map.hpp:44:38:   required from ‘Value boost::detail::read_value(const string&) [with Value = MyEnum; std::__cxx11::string = std::__cxx11::basic_string<char>]’
/usr/include/boost/property_map/dynamic_property_map.hpp:158:64:   required from ‘void boost::detail::dynamic_property_map_adaptor<PropertyMap>::do_put(const boost::any&, const boost::any&, mpl_::bool_<true>) [with PropertyMap = boost::vec_adj_list_vertex_property_map<boost::adjacency_list<boost::vecS, boost::vecS, boost::directedS, EnumWrapper>, boost::adjacency_list<boost::vecS, boost::vecS, boost::directedS, EnumWrapper>*, MyEnum, MyEnum&, MyEnum EnumWrapper::*>]’
/usr/include/boost/property_map/dynamic_property_map.hpp:186:11:   required from ‘void boost::detail::dynamic_property_map_adaptor<PropertyMap>::put(const boost::any&, const boost::any&) [with PropertyMap = boost::vec_adj_list_vertex_property_map<boost::adjacency_list<boost::vecS, boost::vecS, boost::directedS, EnumWrapper>, boost::adjacency_list<boost::vecS, boost::vecS, boost::directedS, EnumWrapper>*, MyEnum, MyEnum&, MyEnum EnumWrapper::*>]’
main.cpp:40:1:   required from here
/usr/include/boost/lexical_cast/detail/converter_lexical.hpp:243:13: error: static assertion failed: Target type is neither std::istream`able nor std::wistream`able
             BOOST_STATIC_ASSERT_MSG((result_t::value || boost::has_right_shift<std::basic_istream<wchar_t>, T >::value),

Target type is neither std::istreamable nor std::wistreamable 向我暗示,lexical_cast 没有正确特化。然而,直接使用词法转换似乎工作得很好。

最佳答案

Target = MyEnum; Source = std::__cxx11::basic_string<char> . Boost 正在尝试从 string 转换 MyEnum ;你只提供了相反的方向。

您需要提供来自的转化 string MyEnum :

namespace boost{
    template<>
    MyEnum lexical_cast(const std::string & s) {
        return MyEnum::A;    // for example
    }
} /* boost */

Example .

关于c++ - 为 boost::property_maps 中的枚举专门化 boost::lexical_cast,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36937912/

相关文章:

c++ - 使用 CString 实例化枚举

.NET:为什么不检查Enum的范围/值?

c++ - Boost.Graph 如何合并两个顶点/契约(Contract)边

c++ - 提供从根到每个分支端的所有顶点路由的访问者

c++ - 在windows下用c++列出目录

c++ - i+=(i&-i) 是做什么的?它是可移植的吗?

C++ 删除 base 或 dynamic_cast 指针?

c - "Grouping"C 中的枚举值

c++ - 加权无向图上的 Floyd Warshall(所有对最短路径) - Boost Graph

c++ - 不按回车进入下一条指令