使用ctypes在Python中调用C++动态库

入门操作

使用ctypes库可以直接调用C语言编写的动态库,而如果是调用C++编写的动态库,需要使用extern关键字对动态库的函数进行声明:

#include <iostream>
using namespace std; extern "C" {
void greet() {
cout << "hello python" << endl;
}
}

将上述的C++程序编译成动态链接库:

g++ hello.cpp -fPIC -shared -o hello.so

在Python代码中使用ctypes导入动态库,调用函数:

# -*- coding: utf-8 -*- #
from ctypes import CDLL hello = CDLL('./hello.so') if __name__ == '__main__':
hello.greet()

运行上述Python程序:

xujijun@pc:~/codespace/python$ python3 hello.py
hello python

参数传递

编写一个整数加法函数

#include <iostream>
using namespace std; extern "C" {
int add(int a, int b) {
return a + b;
}
}

编译得到动态库,在Python代码中调用:

# -*- coding: utf-8 -*- #
from ctypes import CDLL hello = CDLL('./hello.so') if __name__ == '__main__':
a = input('input num1: ')
b = input('input num2: ')
print('output: %d' % hello.add(int(a), int(b)))

运行上述代码,得到输出:

xujijun@pc:~/codespace/python$ python3 hello.py
input num1: 12
input num2: 34
output: 46

尝试传递字符串参数

#include <iostream>
#include <cstdio>
using namespace std; extern "C" {
void print_name(const char* name) {
printf("%s\n", name);
}
}

Python代码调用:

# -*- coding: utf-8 -*- #
from ctypes import CDLL hello = CDLL('./hello.so') if __name__ == '__main__':
name = input('input name: ')
hello.print_name(name.encode('utf-8')) # 此处需要将Python中的字符串按照utf8编码成bytes

查看输出:

xujijun@pc:~/codespace/python$ python3 hello.py
input name: yanhewu
yanhewu

面向对象

使用C++编写动态链接库我们肯定会想到如何在Python中调用C++的类,由于ctypes只能调用C语言函数(C++中采用extern "C"声明的函数),我们需要对接口进行一定的处理:

C++代码示例

#include <iostream>
#include <cstdio>
#include <string>
using namespace std; class Student {
private:
string name;
public:
Student(const char* n);
void PrintName();
}; Student::Student(const char* n) {
this->name.assign(n);
} void Student::PrintName() {
cout << "My name is " << this->name << endl;
} extern "C" {
Student* new_student(const char* name) {
return new Student(name);
} void print_student_name(Student* stu) {
stu->PrintName();
}
}

Python代码中调用:

# -*- coding: utf-8 -*- #
from ctypes import CDLL hello = CDLL('./hello.so') class Student(object):
def __init__(self, name):
self.stu = hello.new_student(name.encode('utf-8')) def print_name(self):
hello.print_student_name(self.stu) if __name__ == '__main__':
name = input('input student name: ')
s = Student(name)
s.print_name()

输出:

xujijun@pc:~/codespace/python$ python3 hello.py
input student name: yanhewu
My name is yanhewu

内存泄漏?

上一部分我们我们尝试了如何使用ctypes调用带有类的C++动态库,这里我们不禁会想到一个问题,我们在动态库中使用new动态申请的内存是否会被Python的GC清理呢?这里我们完全可以猜想,C++动态库中的动态内存并不是使用Python中的内存申请机制申请的,Python不应该对这部分内存进行GC,如果真的是这样,C++的动态库就会出现内存泄漏的问题了。那事实是不是这样呢?我们可以使用内存检查工具Valgrind来检查上面的Python代码:

命令:

valgrind python3 hello.py

最终结果输出:

==17940== HEAP SUMMARY:
==17940== in use at exit: 647,194 bytes in 631 blocks
==17940== total heap usage: 8,914 allocs, 8,283 frees, 5,319,963 bytes allocated
==17940==
==17940== LEAK SUMMARY:
==17940== definitely lost: 32 bytes in 1 blocks
==17940== indirectly lost: 0 bytes in 0 blocks
==17940== possibly lost: 4,008 bytes in 7 blocks
==17940== still reachable: 643,154 bytes in 623 blocks
==17940== suppressed: 0 bytes in 0 blocks
==17940== Rerun with --leak-check=full to see details of leaked memory
==17940==
==17940== For counts of detected and suppressed errors, rerun with: -v
==17940== Use --track-origins=yes to see where uninitialised values come from
==17940== ERROR SUMMARY: 795 errors from 86 contexts (suppressed: 0 from 0)

可以看到,definitely lost了32字节,确实出现了内存泄漏,但是是不是动态库的问题我们还要进一步验证:

C++代码加入析构函数定义:

#include <iostream>
#include <cstdio>
#include <string>
using namespace std; class Student {
private:
string name;
public:
Student(const char* n);
~Student();
void PrintName();
}; Student::Student(const char* n) {
this->name.assign(n);
} Student::~Student() {
cout << "Student's destructor called" << endl;
} void Student::PrintName() {
cout << "My name is " << this->name << endl;
} extern "C" {
Student* new_student(const char* name) {
return new Student(name);
} void print_student_name(Student* stu) {
stu->PrintName();
}
}

运行相同的Python代码:

xujijun@pc:~/codespace/python$ python3 hello.py
input student name: yanhewu
My name is yanhewu

从输出可以看到,Student的析构函数并没有被调用。这里可以确定,Python的GC并没有对动态库中申请的内存进行处理,也确实不能进行处理(毕竟不是Python环境下申请的内存,在C++动态库中可能会先释放这部分内存,如果GC再次释放就会出现内存问题)。但是内存泄漏的问题还是需要解决的,可以参照以下做法:

C++代码中添加内存释放接口:

#include <iostream>
#include <cstdio>
#include <string>
using namespace std; class Student {
private:
string name;
public:
Student(const char* n);
~Student();
void PrintName();
}; Student::Student(const char* n) {
this->name.assign(n);
} Student::~Student() {
cout << "Student's destructor called" << endl;
} void Student::PrintName() {
cout << "My name is " << this->name << endl;
} extern "C" {
Student* new_student(const char* name) {
return new Student(name);
} // 释放对象内存函数
void del_student(Student* stu) {
delete stu;
} void print_student_name(Student* stu) {
stu->PrintName();
}
}

Python代码中,在Student类中调用内存释放函数:

# -*- coding: utf-8 -*- #
from ctypes import CDLL hello = CDLL('./hello.so') class Student(object):
def __init__(self, name):
self.stu = hello.new_student(name.encode('utf-8')) def __del__(self):
# Python的对象在被GC时调用__del__函数
hello.del_student(self.stu) def print_name(self):
hello.print_student_name(self.stu) if __name__ == '__main__':
name = input('input student name: ')
s = Student(name)
s.print_name()

运行Python代码:

xujijun@pc:~/codespace/python$ python3 hello.py
input student name: yanhewu
My name is yanhewu
Student's destructor called

可以看到,C++动态库中的Student类的析构函数被调用了。再次使用Valgrind检查内存使用情况:

==23780== HEAP SUMMARY:
==23780== in use at exit: 647,162 bytes in 630 blocks
==23780== total heap usage: 8,910 allocs, 8,280 frees, 5,317,023 bytes allocated
==23780==
==23780== LEAK SUMMARY:
==23780== definitely lost: 0 bytes in 0 blocks
==23780== indirectly lost: 0 bytes in 0 blocks
==23780== possibly lost: 4,008 bytes in 7 blocks
==23780== still reachable: 643,154 bytes in 623 blocks
==23780== suppressed: 0 bytes in 0 blocks
==23780== Rerun with --leak-check=full to see details of leaked memory
==23780==
==23780== For counts of detected and suppressed errors, rerun with: -v
==23780== Use --track-origins=yes to see where uninitialised values come from
==23780== ERROR SUMMARY: 793 errors from 87 contexts (suppressed: 0 from 0)

可以看到,definitely lost已经变为0,可以确定动态库申请的内存被成功释放。

使用ctypes在Python中调用C++动态库的更多相关文章

  1. JNI_Android项目中调用.so动态库

    JNI_Android项目中调用.so动态库 2014年6月3日 JNI学习 參考:http://blog.sina.com.cn/s/blog_4298002e01013zk8.html 上一篇笔者 ...

  2. CVI中调用VC动态库

    1.在VC环境中建立新工程,创建32位动态库(Win32 Dynamic-Link Library)  -> A simple DLL project 2.在工程中可加入别的动态库,在工程菜单中 ...

  3. JNI_Android项目中调用.so动态库实现详解

    转自:http://www.yxkfw.com/?p=7223 1. 在Eclipse中创建项目:TestJNI 2. 新创建一个class:TestJNI.java package com.wwj. ...

  4. JNI_Android项目中调用.so动态库实现详解【转】

    转自 http://www.cnblogs.com/sevenyuan/p/4202759.html 1. 在Eclipse中创建项目:TestJNI 2. 新创建一个class:TestJNI.ja ...

  5. JNI_Android 项目中调用.so动态库实现详解

    转自:http://www.yxkfw.com/?p=7223 1. 在Eclipse中创建项目:TestJNI 2. 新创建一个class:TestJNI.java package com.wwj. ...

  6. golang调用c动态库

    golang调用c动态库 简介 golang调用c语言动态库,动态方式调用,可指定动态库路径,无需系统目录下 核心技术点 封装c动态库 go语言调用c代码 实例代码 封装c动态库 头文件 test_s ...

  7. 如何在python中调用C语言代码

    1.使用C扩展CPython还为开发者实现了一个有趣的特性,使用Python可以轻松调用C代码 开发者有三种方法可以在自己的Python代码中来调用C编写的函数-ctypes,SWIG,Python/ ...

  8. python调用.net动态库

    # python调用.net动态库 ### pythonnet简介------------------------------ pythonnet是cpython的扩展- pythonnet提供了cp ...

  9. Python脚本传參和Python中调用mysqldump

    Python脚本传參和Python中调用mysqldump<pre name="code" class="python">#coding=utf-8 ...

随机推荐

  1. node+express搭建个人网站(2)

    node+express搭建个人网站(1)这一节中成功启动了一个网站但还很简陋,仅仅打印了一个helloworld的网页 作为个人网站,我们当然想输出自己设计好的网页, 我们借助 Express 应用 ...

  2. HDU4054_Hexadecimal View

    水题.直接八位八位地枚举即可. 注意控制输出,注意读数的时候要把s中的全部元素置零. #include <iostream> #include <cstdio> #includ ...

  3. ZOJ3113_John

    这个题目是一个典型的Anti_Sg.我也不知道为什么这么叫,呵呵,反正大家都这么叫,而且我也是听别人说,看别人的日志自己才知道的. 题目的意思是给你不同颜色的石子,每次可以去一种颜色的石子若干个(至少 ...

  4. web项目访问路径上为什么不能写上WebContent

    我们常常在WEB项目中要写很多的访问路径,比如说/good/target.jsp;目录结构中从来不会带有项目目录结构的WebContent?这到底的为什么呢? 我们知道WEB项目是放在容器上运行的,而 ...

  5. 【Java并发编程】之四:守护线程与线程阻塞的四种情况

    守护线程 Java中有两类线程:User Thread(用户线程).Daemon Thread(守护线程) ​ 用户线程即运行在前台的线程,而守护线程是运行在后台的线程. 守护线程作用是为其他前台线程 ...

  6. 个人博客开发-01-nodeJs项目搭建

    // window系统下 1.nodeJs 安装 nodeJs 安装 看  这里 , 先下载再下一步下一步就OK了,我的是在C盘里安装的. 安装完以后 按 win + R ,在弹出的小框框里输入 CM ...

  7. 漫谈ElasticSearch关于ES性能调优几件必须知道的事

    lasticSearch是现在技术前沿的大数据引擎,常见的组合有ES+Logstash+Kibana作为一套成熟的日志系统,其中Logstash是ETL工具,Kibana是数据分析展示平台.ES让人惊 ...

  8. Alpha 冲刺 —— 十分之二

    队名 火箭少男100 组长博客 林燊大哥 作业博客 Alpha 冲鸭鸭! 成员冲刺阶段情况 林燊(组长) 过去两天完成了哪些任务 协调各成员之间的工作 协助前端界面的开发 搭建测试用服务器的环境 完成 ...

  9. 测试开发面试的Linux面试题总结之二:常用命令

    (1)Linux的文件系统目录配置要遵循FHS规范,规范定义的两级目录规范如下:        /home  每个账号在该目录下都有一个文件夹,进行数据的管理        /usr 有点像windo ...

  10. Linux内核设计第三周学习总结 跟踪分析Linux内核的启动过程

    陈巧然 原创作品 转载请注明出处 <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 实验步骤 登陆实验楼虚 ...