在pybind11中包装可变参数模板

时间:2018-12-12 22:11:15

标签: c++ c++11 boost-python pybind11 pybinding

我正在为glog之类的库编写python绑定,该库使用宏,并且具有cout之类的语法来记录日志。 LOG(LEVEL)<<" "<<" "...。 所以我正在使用此函数来调用宏

    template <typename Arg, typename... Args>
    void log(auto level, Arg&& arg, Args&&... args)
    {
        std::stringstream out;
        out << std::forward<Arg>(arg);
        using expander = int[];
        (void)expander{0, (void(out << ' ' << std::forward<Args>(args)), 0)...};
        LOG(level) << out.str();
    }

因此,为了将该函数包装到pybind11模块中,我需要显式指定模板类型。 有没有使用pybind11绑定此功能的可行方法或方法?我也愿意使用其他库,例如boost.python或cython(如果可能)。

1 个答案:

答案 0 :(得分:0)

最好在运行时完成模板和Python的混合,以确保您拥有将实际使用的所有模板实例化。您可以使用cppyy(http://cppyy.org)进行此操作,该程序使用下面的Cling(LLVM)实例化模板。

使用您的示例(用cerr代替您未发布的特定代码LOG

import cppyy

# variadic template example; the code can also live in a header which is loaded 
# with "cppyy.include()"
cppyy.cppdef(r"""
    template <typename Arg, typename... Args>
    void log(int level, Arg&& arg, Args&&... args)
    {   
        std::stringstream out;
        out << std::forward<Arg>(arg);
        using expander = int[];
        (void)expander{0, (void(out << ' ' << std::forward<Args>(args)), 0)...};
        std::cerr << out.str() << '\n';
    }
""")

level = 1
cppyy.gbl.log(level, 1, "string", 3.14)

预期结果为:

1 string 3.14