将Boost.Proto语法与类型匹配

时间:2011-06-25 19:29:21

标签: c++ boost expression-templates boost-proto

我正在尝试在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> >
{};

因此,如果我不能使用我自己的类型的通配符,我如何测试表达式是否是包含向量的终端?

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<_, _> > 
    >));
}

希望有所帮助!