Обертывание шаблонной функции с boost.python - PullRequest
4 голосов
/ 30 июня 2011

Я пытаюсь представить следующую функцию c ++ для python, используя boost.python:

template <typename genType> 
    genType refract(
        genType const & I, 
        genType const & N, 
        typename genType::value_type const & eta);

, и я получил следующее:

template<typename N>
    N reflect(N const & i, N const & n, typename N::value_type const & eta)
    {
        return glm::N refract(i,n,eta);
    }

BOOST_PYTHON_MODULE(foo)
{
    def("reflect", reflect<float>);
    def("reflect", reflect<double>);
}

и у меня есть следующееошибка при компиляции:

ошибка C2780: 'void boost :: python :: def (const char *, F, const A1 &, const A2 &, const A3 &)': ожидается 5 аргументов - 2 при условии *

Как мне его обернуть?

----- edit ------

Работает:

template<class T>
T floor(T x)
{
    return glm::core::function::common::floor(x);
}

BOOST_PYTHON_MODULE(busta)
{
def("floor", floor<double>);
def("floor", floor<float>);
}

из ссылки, floor() определение является следующим:

template< typename genType >
genType floor (genType const &x)

Я могу построить это как DLL, затем импортировать ее в python и использовать оттуда floor ().Жизнь так приятна ... но ..

Это не сработает, и я хотел бы понять, почему:

template<class genType >
genType reflect (genType i, genType n, genType eta)
{
    return glm::core::function::geometric::refract(i, n,eta);
}

BOOST_PYTHON_MODULE(busta)
{
def("reflect", reflect<float>);
}

refract () - определение в верхней части этого поста.

ошибка, которую я получаю сейчас, такова:

1>foo.cpp(37): error C2893: Failed to specialize function template 'genType glm::core::function::geometric::refract(const genType &,const genType &,const genType::value_type &)'
1>          With the following template arguments:
1>          'float'
1>          foo.cpp(60) : see reference to function template instantiation 'genType 
`anonymous-namespace'::reflect<float>(genType,genType,genType)' being compiled
1>          with
1>          [
1>              genType=float
1>          ]
1>
1>Build FAILED.

1 Ответ

3 голосов
/ 10 июля 2011

Это не идеальный ответ, так как требует злоупотребления системой типов и написания большого количества дополнительного связующего кода.
Вы можете попробовать определить шаблон-обертку, чтобы украсить ваш целевой тип так, чтобы он имел необходимые typedef для удовлетворениявызывающая функция (отражение).

Этот пример демонстрирует недостатки такого подхода.Обратите внимание, что эта функция отражения выполняет простое сложение;однако, чтобы C ++ распознал оператор + для шаблонного типа N, оболочка должна явно его определить.

#include <iostream>   
using namespace std;

template<class N>
N reflect(const N& n, const typename N::value_type& t)
{
  return n + t;
}

template<class N>
struct wrapper
{
  typedef N value_type;

  wrapper(const N& n):_n(n){}
  operator N& () { return _n; }
  N operator+ (const N& r) const { return _n + r; }

  N _n;
};

int main(int,char**)
{
  cout << reflect( wrapper<double>(1), 2.0) << endl;
  return 0;
}
...