我正在尝试在Boost.Proto中创建一个与矢量类型匹配的语法,但是当我给它一个该类型的终端时,它与语法不匹配。类型定义如下所示:
template <typename T, unsigned D>
struct vector
{
typedef T scalar;
enum { size = D };
scalar& operator[](unsigned i)
{
return m_components[i];
}
scalar const& operator[](unsigned i) const
{
return m_components[i];
}
private:
scalar m_components[size];
};
我想要匹配的语法看起来像这样:
namespace proto = boost::proto;
using proto::_;
using proto::N;
struct test:
proto::terminal<vector<_, N> >
{};
比赛失败:
int main ()
{
BOOST_MPL_ASSERT((proto::matches<proto::terminal<vector<float, 2> >::type, test>));
}
如何制作与特定类型匹配的语法?
编辑:
似乎proto :: _和proto :: N未在自定义类型中用作通配符。代码用这个语法编译(matches
断言传递):
struct test:
proto::terminal<vector<float, 2> >
{};
但是当其中任何一个通配符属于该类型时,它都不起作用:
struct test:
proto::terminal<vector<float, N> >
{};
或者:
struct test:
proto::terminal<vector<_, 2> >
{};
因此,如果我不能使用我自己的类型的通配符,我如何测试表达式是否是包含向量的终端?
答案 0 :(得分:7)
要将终端中的类型与类型进行比较,您可以使用类型特征。我有一些特征结构设置,如果给定的类型是向量,则评估为true:
template <typename T>
struct is_vector:
boost::mpl::false_
{};
template <typename T, unsigned Size>
struct is_vector <dev::math::vector <T, Size> >:
boost::mpl::true_
{};
然后你可以把它放在你的语法中:
proto::and_<
proto::terminal<_>,
proto::if_<is_vector<proto::_value>()>
>
之前我尝试过这种方法,但之所以不起作用是因为我转发了声明的向量&lt; ...&gt;在标题中错误的命名空间中,特征结构是在。
答案 1 :(得分:6)
Boost.Proto不适用于非类型模板参数。如果可以,请更改矢量类型以使用整数类型包装器,如下所示:
template <typename T, typename D>
struct vector
{
typedef T scalar;
enum { size = D::value };
scalar& operator[](unsigned i)
{
return m_components[i];
}
scalar const& operator[](unsigned i) const
{
return m_components[i];
}
private:
scalar m_components[size];
};
然后你应该能够匹配如下:
int main ()
{
BOOST_MPL_ASSERT((proto::matches<
proto::terminal<vector<float, mpl::int_<2> > >::type,
proto::terminal<vector<_, _> >
>));
}
希望有所帮助!