- Python 菜鸟教程
- Python 教程
- Python - 概述
- Python - 历史
- Python - 特性
- Python 与 C++
- Python - Hello World 程序
- Python - 应用领域
- Python 解释器及其模式
- Python - 环境设置
- Python - 虚拟环境
- Python - 基本语法
- Python - 变量
- Python - 数据类型
- Python - 类型转换
- Python - Unicode 系统
- Python - 文字
- Python - 运算符
- Python - 算术运算符
- Python - 比较运算符
- Python - 赋值运算符
- Python - 逻辑运算符
- Python - 按位运算符
- Python - 成员资格运算符
- Python - 身份运算符
- Python - 运算符优先级
- Python - 注释
- Python - 用户输入
- Python - 数字
- Python - 布尔值
- Python 控制语句
- Python - 控制流
- Python - 决策
- Python - if 语句
- Python - if-else 语句
- Python - 嵌套 if 语句
- Python - Match-Case 语句
- Python - 循环
- Python - For 循环
- Python for-else 循环
- Python - While 循环
- Python - break 语句
- Python - Continue 语句
- Python - pass 语句
- Python - 嵌套循环
- Python 函数和模块
- Python - 函数
- Python - 默认参数
- Python - 关键字参数
- Python - 仅关键字参数
- Python - 位置参数
- Python - 仅位置参数
- Python - 任意或可变长度参数
- Python - 变量范围
- Python - 函数注释
- Python - 模块
- Python - 内置函数
- Python 字符串
- Python - 字符串
- Python - 切片字符串
- Python - 修改字符串
- Python - 字符串连接
- Python - 字符串格式化
- Python - 转义字符
- Python - 字符串方法
- Python - 字符串练习
- Python 列表
- Python - 列表
- Python - 访问列表项
- Python - 更改列表项
- Python - 添加列表项
- Python - 删除列表项
- Python - 循环列表
- Python - 列表推导式
- Python - 排序列表
- Python - 复制列表
- Python - 联接列表
- Python - 列表方法
- Python - 列表练习
- Python 元组
- Python - 元组(Tuple )
- Python - 访问元组项
- Python - 更新元组
- Python - 解压缩元组项
- Python - 循环元组
- Python - 联接元组
- Python - 元组方法
- Python - 元组练习
- Python 集
- Python - 集(sets)
- Python - 访问 Set Items
- Python - 添加 Set Items
- Python - 删除 Set Items
- Python - 循环 Set Items
- Python - 联接 Sets
- Python - 复制 Set
- Python - Set 运算符
- Python - Set 方法
- Python - Set 的练习
- Python 字典
- Python - 字典
- Python - 访问字典项
- Python - 更改字典项
- Python - 添加字典项
- Python - 删除字典项
- Python - 字典视图对象
- Python - 循环字典
- Python - 复制字典
- Python - 嵌套字典
- Python - 字典方法
- Python - 字典练习
- Python 数组
- Python - 数组
- Python - 访问数组项
- Python - 添加数组项
- Python - 删除数组项
- Python - 循环数组
- Python - 复制数组
- Python - 反向数组
- Python - 对数组进行排序
- Python - 连接数组
- Python - 数组方法
- Python - 数组练习
- Python 文件处理
- Python - 文件处理
- Python - 写入文件
- Python - 读取文件
- Python - 重命名和删除文件
- Python - 目录
- Python - 文件方法
- Python OS 文件/目录方法
- Python - os.path 方法
- 面向对象编程
- Python - OOP 概念
- Python - 类和对象
- Python - 类属性
- Python - 类方法
- Python - 静态方法
- Python - 构造函数
- Python - 访问修饰符
- Python - 继承
- Python - 多态性
- Python - 方法覆盖
- Python - 方法重载
- Python - 动态绑定
- Python - 动态类型
- Python - 抽象
- Python - 封装
- Python - 接口
- Python - 软件包
- Python - 内部类
- Python - 匿名类和对象
- Python - 单例类
- Python - 包装类
- Python - 枚举
- Python - 反射
- Python 错误和异常
- Python - 语法错误
- Python - 异常处理
- Python - try-except 块
- Python - try-finally 块
- Python - 引发异常
- Python - 异常链接
- Python - 嵌套 try 块
- Python - 用户定义的异常
- Python - 日志记录
- Python - 断言
- Python - 内置异常
- Python 多线程
- Python - 多线程
- Python - 线程生命周期
- Python - 创建线程
- Python - 启动线程
- Python - 联接线程
- Python - 命名线程
- Python - 线程调度
- Python - 线程池
- Python - 主线程
- Python - 线程优先级
- Python - 守护程序线程
- Python - 同步线程
- Python 同步
- Python - 线程间通信
- Python - 线程死锁
- Python - 中断线程
- Python 网络
- Python - 网络编程
- Python - 套接字编程
- Python - URL 处理
- Python - 泛型
- Python 杂项
- Python - 日期和时间
- Python - math 模块
- Python - 迭代器
- Python - 生成器
- Python - 闭包(closures)
- Python - 装饰器( Decorators)
- Python - 递归
- Python - 正则表达式
- Python - PIP
- Python - 数据库访问
- Python - 弱引用
- Python - 序列化
- Python - 模板
- Python - 输出格式
- Python - 性能测量
- Python - 数据压缩
- Python - CGI 编程
- Python - XML 处理
- Python - GUI 编程
- Python - 命令行参数
- Python - 文档字符串
- Python - JSON
- Python - 发送电子邮件
- Python - 更多扩展
- Python - 工具/实用程序
- Python - 图形用户界面
- Python 高级概念
- Python - 抽象基类
- Python - 自定义异常
- Python - 高阶函数
- Python - 对象内部
- Python - 内存管理
- Python - 元类
- Python - 使用 Metaclasses 进行元编程
- Python - 模拟和存根
- Python - 猴子修补
- Python - 信号处理
- Python - 类型提示
- Python - 自动化教程
- Python - 人性化软件包
- Python - 上下文管理器
- Python - 协程
- Python - 描述符
- Python - 诊断和修复内存泄漏
- Python - 不可变数据结构
Python - 更多扩展
您使用任何编译语言(如 C、C++ 或 Java)编写的任何代码都可以集成或导入到另一个 Python 脚本中。此代码被视为 “扩展”。
Python 扩展模块只不过是一个普通的 C 库。在 Unix 计算机上,这些库通常以 .so 结尾(对于共享对象)。在 Windows 计算机上,您通常会看到 .dll(对于动态链接库)。
编写扩展的先决条件
要开始编写扩展,您将需要 Python 头文件。
- 在 Unix 计算机上,这通常需要安装特定于开发人员的软件包。
- Windows 用户在使用二进制 Python 安装程序时,会将这些标头作为包的一部分获取。
此外,假设您对 C 或 C++ 有很好的了解,可以使用 C 编程编写任何 Python 扩展。
首先看一下 Python 扩展
首次查看 Python 扩展模块时,您需要将代码分为四个部分 -
- 头文件 Python.h.
- 你想要作为模块接口公开的 C 函数..
- 一个表格映射了你的函数名称,因为 Python 开发人员将它们视为扩展模块内的 C 函数。
- 初始化函数。
头文件 Python.h
您需要在 C 源文件中包含 Python.h 头文件,这样您就可以访问用于将模块挂接到解释器中的内部 Python API。
确保在您可能需要的任何其他标头之前包含 Python.h。您需要在包含后面跟着要从 Python 调用的函数。
C 函数
函数的 C 实现的签名始终采用以下三种形式之一 -
static PyObject *MyFunction(PyObject *self, PyObject *args);
static PyObject *MyFunctionWithKeywords(PyObject *self,
PyObject *args,
PyObject *kw);
static PyObject *MyFunctionWithNoArgs(PyObject *self);
上述每个声明都返回一个 Python 对象。Python 中没有 C 中那样的 void 函数。如果您不希望函数返回值,请返回 Python 的 None 值的 C 等效项。Python 标头定义了一个 宏 Py_RETURN_NONE,它为我们执行此操作。
你的 C 函数的名称可以是你喜欢的任何名称,因为它们永远不会在扩展模块之外看到。它们被定义为 static 函数。
您的 C 函数通常是通过将 Python 模块和函数名称组合在一起来命名的,如下所示 -
static PyObject *module_func(PyObject *self, PyObject *args) {
/* Do your stuff here. */
Py_RETURN_NONE;
}
这是 module 模块中一个名为 func 的 Python 函数。您将把指向 C 函数的指针放入源代码中通常下一个模块的方法表中。
方法映射表
此方法表是 PyMethodDef 结构的简单数组。该结构看起来像这样 -
struct PyMethodDef {
char *ml_name;
PyCFunction ml_meth;
int ml_flags;
char *ml_doc;
};
以下是此结构的成员的描述 -
- ml_name − 这是 Python 解释器在 Python 程序中使用时显示的函数名称。
- ml_meth − 这是具有上一节中描述的任何一个签名的函数的地址。
- ml_flags − 这告诉解释器ml_meth正在使用三个签名中的哪一个。
- 此标志的值为 METH_VARARGS。
- 如果要允许关键字参数进入函数,则可以使用 METH_KEYWORDS 按位 OR 运算此标志。
- 这也可以具有 METH_NOARGS 值,表示您不想接受任何参数。
- mml_doc − 这是函数的文档字符串,如果不这样做,则可能为 NULL 想写一个。
此表需要以 sentinel 终止,该 sentinel 由相应成员的 NULL 和 0 值组成。
例对于上面定义的函数,我们有以下方法映射表 -
static PyMethodDef module_methods[] = {
{ "func", (PyCFunction)module_func, METH_NOARGS, NULL },
{ NULL, NULL, 0, NULL }
};
初始化函数
扩展模块的最后一部分是初始化函数。加载模块时,Python 解释器会调用此函数。要求将函数命名为 initModule,其中 Module 是模块的名称。
初始化函数需要从您将要构建的库中导出。Python 头文件定义了 PyMODINIT_FUNC 以包含适当的咒语,以便针对我们正在编译的特定环境执行此操作。您所要做的就是在定义函数时使用它。
您的 C 初始化函数通常具有以下整体结构 -
PyMODINIT_FUNC initModule() {
Py_InitModule3(func, module_methods, "docstring...");
}
以下是Py_InitModule3函数的描述 -
- func − 这是要导出的函数。
- module_methods − 这是上面定义的映射表名称。
- docstring − 这是你想在扩展中给出的注释。
将所有这些放在一起,它看起来像下面 -
#include <Python.h>
static PyObject *module_func(PyObject *self, PyObject *args) {
/* Do your stuff here. */
Py_RETURN_NONE;
}
static PyMethodDef module_methods[] = {
{ "func", (PyCFunction)module_func, METH_NOARGS, NULL },
{ NULL, NULL, 0, NULL }
};
PyMODINIT_FUNC initModule() {
Py_InitModule3(func, module_methods, "docstring...");
}
例
一个利用上述所有概念的简单示例 -
#include <Python.h>
static PyObject* helloworld(PyObject* self)
{
return Py_BuildValue("s", "Hello, Python extensions!!");
}
static char helloworld_docs[] =
"helloworld( ): Any message you want to put here!!\n";
static PyMethodDef helloworld_funcs[] = {
{"helloworld", (PyCFunction)helloworld,
METH_NOARGS, helloworld_docs},
{NULL}
};
void inithelloworld(void)
{
Py_InitModule3("helloworld", helloworld_funcs,
"Extension module example!");
}
这里 Py_BuildValue 函数用于构建 Python 值。将上面的代码保存在 你好.c 文件中。我们将了解如何编译和安装这个要从 Python 脚本调用的模块。
构建和安装扩展
distutils 包使以标准方式分发 Python 模块(包括纯 Python 模块和扩展模块)变得非常容易。模块以源代码形式分发,通过通常称为 setup.pyas 的安装脚本构建和安装。
对于上述模块,您需要准备以下 setup.py 脚本 -
from distutils.core import setup, Extension
setup(name='helloworld', version='1.0', \
ext_modules=[Extension('helloworld', ['hello.c'])])
现在,使用以下命令,该命令将使用正确的编译器和链接器命令和标志执行所有需要的编译和链接步骤,并将生成的动态库复制到适当的目录中 -
在基于 Unix 的系统上,您很可能需要以 root 身份运行此命令,以便获得写入 site-packages 目录的权限。这在 Windows 上通常不是问题。
导入扩展
安装扩展后,您将能够在 Python 脚本中导入和调用该扩展,如下所示 -
import helloworld
print helloworld.helloworld()
这将产生以下输出 -
传递函数参数
由于您很可能希望定义接受参数的函数,因此您可以对 C 函数使用其他签名之一。例如,以下接受一定数量参数的函数将按如下方式定义 -
static PyObject *module_func(PyObject *self, PyObject *args) {
/* Parse args and do something interesting here. */
Py_RETURN_NONE;
}
包含新函数条目的方法表将如下所示 -
static PyMethodDef module_methods[] = {
{ "func", (PyCFunction)module_func, METH_NOARGS, NULL },
{ "func", module_func, METH_VARARGS, NULL },
{ NULL, NULL, 0, NULL }
};
您可以使用 API PyArg_ParseTuple 函数从传递到 C 函数的一个 PyObject 指针中提取参数。
PyArg_ParseTuple的第一个参数是 args 参数。这是您将要解析的对象。第二个参数是一个格式字符串,用于描述您期望的参数显示方式。每个参数由格式字符串中的一个或多个字符表示,如下所示。
static PyObject *module_func(PyObject *self, PyObject *args) {
int i;
double d;
char *s;
if (!PyArg_ParseTuple(args, "ids", &i, &d, &s)) {
return NULL;
}
/* Do something interesting here. */
Py_RETURN_NONE;
}
编译模块的新版本并导入它,使您能够使用任意数量的任何类型的参数调用新函数 -
module.func(1, s="three", d=2.0)
module.func(i=1, d=2.0, s="three")
module.func(s="three", d=2.0, i=1)
你可能会想出更多的变化。
PyArg_ParseTuple 函数
re 是 PyArg_ParseTuple 函数的标准签名 -
int PyArg_ParseTuple(PyObject* tuple,char* format,...)
此函数对错误返回 0,对于成功,返回不等于 0 的值。Tuple 是 PyObject*,它是 C 函数的第二个参数。这里的 format 是一个 C 字符串,用于描述强制和可选参数。
以下是 PyArg_ParseTuple 函数的格式代码列表 -
Code | C type | 意义 |
---|---|---|
c | char | 长度为 1 的 Python 字符串变为 C 字符。 |
d | double | Python 浮点数变为 C 双精度。 |
f | float | Python 浮点数变为 C 浮点数。 |
i | int | Python int 变为 C int。 |
l | long | Python int 变为 C long。 |
L | long long | Python int 变为 C long long。. |
O | PyObject* | 获取对 Python 参数的非 NULL 借用引用。 |
S | char* | C char* 中没有嵌入 null 的 Python 字符串。 |
s# | char*+int | 任何 Python 字符串到 C 地址和长度。 |
t# | char*+int | 只读单段缓冲区到 C 地址和长度。 |
u | Py_UNICODE* | C 中没有嵌入 null 的 Python Unicode。 |
u# | Py_UNICODE*+int | 任何 Python Unicode C 地址和长度。 |
w# | char*+int | 将单段缓冲区读/写到 C 地址和长度。 |
z | char* | 与 s 一样,也接受 None (将 C char* 设置为 NULL)。 |
z# | char*+int | 与 s# 一样,也接受 None(将 C char* 设置为 NULL)。 |
(...) | as per ... | Python 序列被视为每个项目的一个参数。 |
| | 以下参数是可选的。 | |
: | 格式为 end,后跟错误消息的函数名称。 | |
; | 格式为 end,后跟整个错误消息文本。 |
返回值
Py_BuildValue 接受格式字符串,就像 PyArg_ParseTuple 一样。而不是 传入您正在构建的值的地址,您将传入实际值。下面是一个演示如何实现 add 函数的示例。
static PyObject *foo_add(PyObject *self, PyObject *args) {
int a;
int b;
if (!PyArg_ParseTuple(args, "ii", &a, &b)) {
return NULL;
}
return Py_BuildValue("i", a + b);
}
这是用 Python 实现的样子 -
def add(a, b):
return (a + b)
您可以从函数中返回两个值,如下所示。这将使用 Python 中的列表捕获。
static PyObject *foo_add_subtract(PyObject *self, PyObject *args) {
int a;
int b;
if (!PyArg_ParseTuple(args, "ii", &a, &b)) {
return NULL;
}
return Py_BuildValue("ii", a + b, a - b);
}
这是用 Python 实现的样子 -
def add_subtract(a, b):
return (a + b, a - b)
Py_BuildValue 函数
以下是Py_BuildValue函数的标准签名 -
PyObject* Py_BuildValue(char* format,...)
此处 format 是一个 C 字符串,用于描述要构建的 Python 对象。Py_BuildValue 的以下参数是构建结果的 C 值。PyObject* 结果是一个新引用。
下表列出了常用的代码字符串,其中零个或多个字符串联接为字符串格式。
Code | C type | 意义 |
---|---|---|
c | char | C char 变为长度为 1 的 Python 字符串。 |
d | double | C double 变为 Python 浮点数。 |
f | float | C 浮点数变为 Python 浮点数。 |
i | int | C int 变为 Python int |
l | long | C long 变为 Python int |
N | PyObject* | 传递 Python 对象并窃取引用。 |
O | PyObject* | 传递 Python 对象并照常对其进行 INCREF。 |
O& | convert+void* | 任意转换 |
s | char* | C 以 0 结尾的 char* 更改为 Python 字符串,或 NULL 更改为 None。 |
s# | char*+int | C char* 和 length 设置为 Python 字符串,或 NULL 设置为 None。 |
u | Py_UNICODE* | C 范围的以 null 结尾的字符串设置为 Python Unicode,或将 NULL 设置为 None。 |
u# | Py_UNICODE*+int | C 范围的字符串和长度设置为 Python Unicode,或将 NULL 设置为 None。 |
w# | char*+int | 将单段缓冲区读/写到 C 地址和长度。 |
z | char* | 与 s 一样,也接受 None (将 C char* 设置为 NULL)。 |
z# | char*+int | 与 s# 一样,也接受 None(将 C char* 设置为 NULL)。 |
(...) | as per ... | 从 C 值构建 Python 元组。 |
[...] | as per ... | 从 C 值构建 Python 列表。 |
{...} | as per ... | 从 C 值构建 Python 字典,交替键和值。 |
代码 {...} 从偶数个 C 值(交替键和值)构建字典。
例如,Py_BuildValue(“{issi}”,23,“zig”,“zag”,42) 返回类似于 Python 的 {23:'zig','zag':42} 的字典