python 与c++相互调用实现

目录
  • 一、c++调用Python
    • 1.Python脚本
    • 2.C++调用python脚本
  • 二、接口方法
    • 1.规范化语法
  • 三、Pthon调用c++
    • 1.基于extern
    • 2.基于swig

一、c++调用Python

将Python安装目录下的includelibs文件夹引入到项目中,将libs目录下的python37.lib复制一份为python37_d.lib

1.Python脚本

def Hello():
    print("Hello")
     
def Add(a,b):
    return  a+b

2.C++调用python脚本

#include <Python.h>
using namespace std;
 
int main()
{
    Py_Initialize();              //初始化,创建一个Python虚拟环境
    if (Py_IsInitialized())
    {
        PyObject* pModule = NULL;
        PyObject* pFunc = NULL;
        pModule = PyImport_ImportModule("test_python");  //参数为Python脚本的文件名
        if (pModule)
        {
            pFunc = PyObject_GetAttrString(pModule, "Hello");   //获取函数
            PyEval_CallObject(pFunc, NULL);           //执行函数
        }
        else
        {
            printf("导入Python模块失败...\n");
        }
    }
    else
    {
        printf("Python环境初始化失败...\n");
    }
    Py_Finalize();
}

二、接口方法

Python3.6提供给C/C++接口函数,基本都是定义pylifecycle.hpythonrun.hceval.h中。

  • Py_Initialize() 和 Py_Finalize():必须先调用Py_Initialize()进行初始化,这个API用来分配Python解释器使用的全局资源,应用程序结束时需要调用Py_Finalize()来关闭Python的使用环境。
  • Py_IsInitialized():用来判断Python解释器是否初始化成功,true为成功,false为失败。
  • PyErr_Print() & PyErr_Clear():执行Python出错时,PyErr_Print()可将错误信息显示出来,PyErr_Clear()将错误信息在Python解释器的缓存清除。
  • PyRun_SimpleString():这个函数能够用来执行简单的Python语句。
  • PyEval_InitThreads():如果使用多线程调用Python脚本,就需要在初始化Python解释器时调用PyEval_InitThreads()来启用线程支持(导致Python内部启用线程锁),最好在主线程启动时就调用。该API同时也锁定全局解释锁,所以,还需要在初始化完成后需要自行释放锁。
  • 如果不需要使用多线程,不建议启用该选项,互斥锁也会不可避免的增加系统开销。

1.规范化语法

#include<Python.h> //添加python的声明
 
using namespace std;
 
int main()
{
Py_Initialize(); //1、初始化python接口
 
//初始化使用的变量
PyObject* pModule = NULL;
PyObject* pFunc = NULL;
PyObject* pName = NULL;
 
//2、初始化python系统文件路径,保证可以访问到 .py文件
PyRun_SimpleString("import sys");
PyRun_SimpleString("sys.path.append('./')");
 
//3、调用python文件名。当前的测试python文件名是test.py。在使用这个函数的时候,只需要写文件的名称就可以了。不用写后缀。
pModule = PyImport_ImportModule("test");
 
//4、调用函数
pFunc = PyObject_GetAttrString(pModule, "AdditionFc");
 
//5、给python传参数
PyObject* pArgs = PyTuple_New(2);//函数调用的参数传递均是以元组的形式打包的,2表示参数个数。如果AdditionFc中只有一个参数时,写1就可以了。这里只先介绍函数必须有参数存在的情况。
 
 
PyTuple_SetItem(pArgs, 0, Py_BuildValue("i", 2)); //0:表示序号。第一个参数。
PyTuple_SetItem(pArgs, 1, Py_BuildValue("i", 4)); //1:也表示序号。第二个参数。i:表示传入的参数类型是int类型。
 
//6、使用C++的python接口调用该函数
PyObject* pReturn = PyEval_CallObject(pFunc, pArgs);
 
//7、接收python计算好的返回值
int nResult;
PyArg_Parse(pReturn, "i", &nResult);//i表示转换成int型变量。在这里,最需要注意的是:PyArg_Parse的最后一个参数,必须加上“&”符号。
 
//8、结束python接口初始化
Py_Finalize();
}

三、Pthon调用c++

python调用c++一种是基于extern 的方式,另一种是swig

1.基于extern

初级版:

首先先看一下Python调用c

C代码:

#include <stdio.h> 
#include <stdlib.h> 
int foo(int a, int b) 
{ 
  printf("you input %d and %d\n", a, b); 
  return a+b; 
} 

Python代码:

import ctypes 
lib = ctypes.CDLL("./libpycall_c.so")   
lib.foo(1, 3) 
print '***finish***'

编译:

gcc -g -o libpycall_c.so -shared -fPIC pycall_c.c

然后基于c++改造上述代码(使用g++编译生成C动态库的代码中的函数或者方法,需要使用extern “C”来进行编译)

c++代码:

#include <iostream>
using namespace std;
int foo(int a, int b){
    cout << "the number you input:" << a << "\t" << b << endl;
    return a + b;
}
extern "C" {
   int foo_(int a, int b){
       foo(a, b);  
    }
}

python代码:

import ctypes 
lib = ctypes.CDLL("./libpycall.so")   
lib.foo_(1, 3) 
print '***finish***'

编译:

g++ -g -o libpycall.so -shared -fPIC pycall.cpp

升级版:

c++定义一个类,通过python调用c++类的方法

#include <iostream>

using namespace std;

class TestLib{
    private:
        int number = 0;

    public:
        void set_number(int num){
            number = num;
        }
        int get_number(){
            return number;
        }
}; 

extern "C" {
    TestLib obj;
    int get_number(){
        return obj.get_number();
    }
    void set_number(int num){
        obj.set_number(num);
    }
}

python 代码:

import ctypes

lib = ctypes.CDLL("./libpycall.so")
print lib.get_number()  #0
lib.set_number(10)
print lib.get_number()   #10

编译:

g++ -g -o libpycall.so -shared -fPIC -std=c++11 pycall.cpp

2.基于swig

Swig是一种软件开发工具,能让一些脚本语言调用C/C++语言的接口。它实现的方法是,通过编译程序将C/C++的声明文件(.i文件)编译成C/C++的包装器源代码(.c或.cxx)。通过直接调用这样的包装器接口,脚本语言可以间接调用C/C++语言的程序接口。

参考地址:https://github.com/swig/swig

首先安装,源码或者pip

案例:

有这样一段C的代码,文件名为example.c

/* File : example.c */

double  My_variable  = 3.0;

/* Compute factorial of n */
int  fact(int n) {
    if (n <= 1) return 1;
    else return n*fact(n-1);
}

/* Compute n mod m */
int my_mod(int n, int m) {
    return(n % m);
}

你想在你的脚本语言的代码里面调用fact函数。你可以通过一段非常简单的SWIG脚本,文件名为example.i:(这里的格式非常重要,即使第一行的注释也不能省略)

/* File : example.i */
%module example
%{
/* Put headers and other declarations here */
extern double My_variable;
extern int    fact(int);
extern int    my_mod(int n, int m);
%}

extern double My_variable;
extern int    fact(int);
extern int    my_mod(int n, int m);

这段.i文件分成3个部分:

  • 第一部分是%module example%module是SWIG脚本的一个命令,它表示生成的包装器将在一个模块内的名称。
  • 第二部分是%{… %},这一部分的内容会原封不动的插入到xxxx_wrap.cxxxx_wrap.cxx文件中。
  • 第三部分就是剩下的部分了。这部分就是C语言或者C++语言的接口声明了。和C/C++的语法是一样的。

接下来以linux操作系统下,为python语言生成接口为例:

swig -python example.i

执行上述语句会生成两个文件example.pyexample_wrap.cexample.py就是python语言可以调用的example模块,而example_wrap.c则封装了example.c的封装器。

然后执行第二步:

gcc -c -fPIC example.c example_wrap.c -I/usr/include/python2.7

执行该步会生成两个o文件,example.oexample_wrap.o

最后执行:

g++ -shared example.o example_wrap.o -o _example.so

这一步会将上面两个o文件封装成一个新的动态库,_example.so。在这之后就可以在python内直接调用example.c提供的接口了。

import example
print example.fact(3)
print example.cvar.My_variable   #注意这里的参数不能直接用,得用cvar

到此这篇关于python 与c++相互调用实现的文章就介绍到这了,更多相关python 与c++相互调用内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 如何在C++中调用Python

    Python的安装 为了使用Python.h这个扩展项,我们需要安装一个python*-dev而不是python*,这两者略有区别,下面的案例展示的是在Ubuntu20.04下安装python3.9-dev的方法: dechin@ubuntu2004:~/projects/gitlab/dechin/$ sudo apt install python3.9-dev 正在读取软件包列表... 完成 正在分析软件包的依赖关系树 正在读取状态信息... 完成 下列软件包是自动安装的并且现在不需要了:

  • Python调用C/C++的方法解析

    Python是解释性语言, 底层就是用c实现的, 所以用python调用C是很容易的, 下面就总结一下各种调用的方法, 给出例子, 所有例子都在ubuntu9.10, python2.6下试过. 1. Python 调用 C (base) 想在python中调用c函数, 如这儿的fact #include <Python.h> int fact(int n) { if (n <= 1) return 1; else return n * fact(n - 1); } PyObject*

  • python直接调用和使用swig法方调用c++库

    c++运算速度快于python,python简单易写.很多时候对于已有的c++代码也不想用python重写,此时就自然而然地想到用python调用c或者c++,两全其美.然而根据这些博客的说法,python只能实现c的调用,如果需要调用c++,还需要对c++代码进行额外的处理. 首先是python调用c代码: //gcc -g -o libpycall_c.so -shared -fPIC pycall_c.c #include <stdio.h>  #include <stdlib.h

  • C++调用python(执行py文件)的全过程

    1.首先要配好vs开发工程 注意版本:我这使用32位的python那么我vs工程这边也选择32位的编译环境去配置 注意点:需要将python安装目录的一些文件拷过来作为vs工程使用. 2.C++调用Python结果 py代码 这里引用了cdll库也需要放置到运行目录,py文件也是需要放置到运行目录(也就是exe生成所在目录) import os import time from ctypes import * def testDLL(): pDll = CDLL("./pythonTestCDl

  • 如何在C++中调用python代码你知道吗

    目录 一.环境设置 二.VS项目中设置 (1)首先在acaconda中找到include文件夹和libs文件夹,如图所示 (2)点击链接器,然后输入,附加依赖项,添加python36_d.lib的路径 (3)python代码 三.可能会出现的问题 总结 一.环境设置 windows VS2015 python的话用的是acaconda自带的python环境,不同版本的acaconda自带的python不同,我的是python3.6(这一步很重要,如果使用acaconda创建的虚拟环境的pytho

  • python 与c++相互调用实现

    目录 一.c++调用Python 1.Python脚本 2.C++调用python脚本 二.接口方法 1.规范化语法 三.Pthon调用c++ 1.基于extern 2.基于swig 一.c++调用Python 将Python安装目录下的include和libs文件夹引入到项目中,将libs目录下的python37.lib复制一份为python37_d.lib 1.Python脚本 def Hello():     print("Hello")       def Add(a,b):

  • Python和java 如何相互调用

    目录 1.Java调用Python的函数 2.在java中调用本机python脚本中的函数 3.使用java直接执行python脚本 1.Java调用Python的函数 在java类中直接执行python语句 class="highlight"> import org.python.util.PythonInterpreter; /** * Created by wjf on 2018/3/16. */ public class java2python { public stat

  • Python和java 如何相互调用

    目录 1.Java调用Python的函数 2.在java中调用本机python脚本中的函数 3.使用java直接执行python脚本 1.Java调用Python的函数 在java类中直接执行python语句 class="highlight"> import org.python.util.PythonInterpreter; /** * Created by wjf on 2018/3/16. */ public class java2python { public stat

  • Python父目录、子目录的相互调用方法

    最近在使用Python的过程中经常遇到找不到该模块的问题.其中一个就是父目录子目录之间相互调用的情况.下面简单总结下. 我们在F:\Code文件夹下面创建一个test文件夹 而test文件夹里面如下 包含两个子目录 a.py def showdata(): print("this is a") def plus(): a=1 b=2 print(a+b) b.py def show(): print("this is b") 从父目路test.py调用a和b fro

  • Python与C/C++的相互调用案例

    一.问题 Python模块和C/C++的动态库间相互调用在实际的应用中会有所涉及,在此作一总结. 二.Python调用C/C++ 1.Python调用C动态链接库 Python调用C库比较简单,不经过任何封装打包成so,再使用python的ctypes调用即可. (1)C语言文件:pycall.c /***gcc -o libpycall.so -shared -fPIC pycall.c*/ #include <stdio.h> #include <stdlib.h> int f

  • 胶水语言Python与C/C++的相互调用的实现

    准备工作: python:https://www.python.org/downloads/ Dev-C++:https://sourceforge.net/projects/orwelldevcpp/ gcc和g++:http://mingw-w64.org/doku.php notepad++:https://notepad-plus.en.softonic.com/ 一.Python调用C 步骤1:Csayhello.c #include<stdio.h> void show_hello

  • python模块与C和C++动态库相互调用实现过程示例

    目录 Python调用C/C++ 1.Python调用C动态链接库 C语言文件:pycall.c gcc编译生成动态库libpycall.so Python调用动态库的文件:pycall.py 运行结果: 2.Python调用C/C++原生态导出 3.Python调用C/C++通过boost实现 4.Python调用C/C++通过导出类 5.Python调用C/C++通过导出变参函数 6.Python调用C/C++通过导出带Python对象的接口 Python调用C/C++ 1.Python调用

  • C#和lua相互调用的方法教程

    前言 自从ulua在官网上出来后,lua 就被u3d开发人员喜爱.国内有几个高手把lua拿过来 接着进行了封装.很多都是新手转过来.lua语法一看遍知,但是大多数人还是不明白两个语言之间的互相调用是怎么一回事,这也是难点和重点.所以今天想跟大家分享一下这方面的知识,让大家少走弯路吧. Lua是一种很好的扩展性语言,Lua解释器被设计成一个很容易嵌入到宿主程序的库.LuaInterface则用于实现Lua和CLR的混合编程. C与lua交互面临以下几个问题: 1.由于lua里面的数据都是动态加载的

  • python实现根据窗口标题调用窗口的方法

    本文实例讲述了python实现根据窗口标题调用窗口的方法.分享给大家供大家参考.具体分析如下: 当你知道一个windows窗口的标题后,可以用下面的代码调用窗口,甚至向窗口内写入内容. #-*-coding:utf-8-*- import win32gui,win32con #下面的是窗口的标题名称,这样是一定错的,但在控制台就可以正常使用 #写在文件里要用U编码 a=u"jb51.net" dlg=win32gui.FindWindow(None,a) //用控件的ID取得控件的句柄

  • Objective-C 代码与Javascript 代码相互调用实例

    现在好的IOS 的 APP有时候会跟网页内容相关联,当然也是增加用户体验,及更好的,有效的体现APP 的功能. 由于本人在项目中会用到这部分功能,做下记录! JS调用OC 很多应用里面或多或少的调用了网页,来达到绚丽的效果,所谓的JS调用OC.....举个例子吧,网页上有个按钮 点击按钮跳转界面,跳转的动作由OC的代码实现. OC调用JS 还是举个例子,我们OC代码创建了输入框比如输入用户名,输入完成后显示在网页上,显示用户的用户名 一.利用webView的代理方法实现OC和JS的相互调用 创建

随机推荐