使用Boost-Python计算python中定义的函数的导数

时间:2012-12-23 03:08:48

标签: c++ python numpy boost-python sympy

我想写一个Boost-Python程序来从用户那里获取一个符号python函数,并在我的程序中评估它的派生。

例如,User提供了一个定义函数的python文件(Function.py) F = sin(x)* cos(x)。

然后我希望使用Sympy的符号分化能力来访问F'(x)(F(x)的导数)。我不想使用数字微分。

有没有办法使用Boost-Python在C ++中访问这样的函数F'(x)。

2 个答案:

答案 0 :(得分:4)

以下是一些可以帮助您入门的代码。

main.cpp中:

#include <boost/python.hpp>
#include <iostream>

using namespace boost::python;

int main(void) {
    Py_Initialize();

    object main_module = import("__main__");
    object main_namespace = main_module.attr("__dict__");

    exec("from __future__ import division\n"
        "from sympy import *\n"
        "x = symbols('x')\n"
        "f = symbols('f', cls=Function)\n"
        "f = cos(x) * sin(x)\n"
        "f1 = lambda u: diff(f).subs(x, u);\n",
        main_namespace);

    exec("result = f1(1.0)", main_namespace);
    double res = extract<double>(main_namespace["result"]);
    std::cout << "Out: " << res << std::endl;

    return 0;
}

编译命令,替换为您的路径和编译器:

$ clang++ -I"/usr/local/Cellar/python/2.7.3/Frameworks/Python.framework/Versions/Current/Headers/" -L"/usr/local/Cellar/python/2.7.3/Frameworks/Python.framework/Versions/Current/lib/" -lpython2.7 main.cpp

它编译但现在对我不起作用。希望它有所帮助。

答案 1 :(得分:3)

我不是SymPy专家,但也许这可以帮到你:

您可以定义Python方法,如:

def f(x):
  return sin(x)*cos(x)

您可以使用以下方法创建可评估函数 f1 作为 f 的衍生物:

from sympy import *
x = symbols('x')
f1 = lambdify(x, diff(f(x)))

可以使用boost :: python从C ++调用此函数f1。您可以创建一个对象来运行f1,使用()运算符调用该函数,并使用 extract&lt;&gt; 将结果转换为double。

以下是一个例子:

  namespace py = boost::python;

  Py_Initialize();

  py::object main_module = py::import("__main__");
  py::object main_dict   = main_module.attr("__dict__");

  py::exec(
    "def f(x):\n"
    "  return  sin(x)*cos(x)\n",
    main_dict
  );

  py::exec(
    "from sympy import *\n"
    "x = symbols('x')\n"
    "f1 = lambdify(x, diff(f(x)))\n",
     main_dict
  );

  py::object f1 = main_dict["f1"];
  std::cout << py::extract<double>(f1(0.0)) << std::endl;
  std::cout << py::extract<double>(f1(1.0)) << std::endl;

  return 0;