晒晒我家小院子

0%

Python3 调用C语言代码

开发环境
1
centos7+python3.6
备注
1
2
3
Python中的ctypes模块可能是Python调用C方法中最简单的一种。
ctypes模块提供了和C语言兼容的数据类型和函数来加载dll文件,
因此在调用时不需对源文件做任何的修改,也正是如此奠定了这种方法的简单性。
准备文件
1.准备C语言程序,保存为add.c
1
2
3
4
5
6
7
8
9
10
11
12
#include <stdio.h>

int add_int(int, int);
float add_float(float, float);

int add_int(int num1, int num2){
return num1 + num2;
}

float add_float(float num1, float num2){
return num1 + num2;
}
2.编译成so库

输入命令即可,会出现一个.so文件

1
gcc -shared -Wl,-soname,adder -o adder.so -fPIC add.c
3.准备python代码,保存为python-c.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
import ctypes

#load the shared object file
adder = ctypes.cdll.LoadLibrary('./adder.so')

#Find sum of integers
res_int = adder.add_int(4,5)
print("4 + 5 = " + str(res_int))

#Find sum of floats
a = ctypes.c_float(5.5)
b = ctypes.c_float(4.1)

add_float = adder.add_float
add_float.restype = ctypes.c_float

print("5.5 + 4.1 = " + str(add_float(a, b)))
4.执行
1
python3 python-c.py

执行结果如下

1
2
4 + 5 = 9
5.5 + 4.1 = 9.600000381469727
5.特殊说明
  • 在Python文件中,一开始先导入ctypes模块,然后使用cdll.LoadLibrary函数来加载我们创建的库文件。这样我们就可以通过变量adder来使用C类库中的函数了。当adder.add_int()被调用时,内部将发起一个对C函数add_int的调用。ctypes接口允许我们在调用C函数时使用原生Python中默认的字符串型和整型。
  • 而对于其他类似布尔型和浮点型这样的类型,必须要使用正确的ctype类型才可以。如向adder.add_float()函数传参时, 我们要先将Python中的十进制值转化为c_float类型,然后才能传送给C函数。这种方法虽然简单,清晰,但是却很受限。例如,并不能在C中对对象进行操作。

0. “&”

1
2
一般我们可在结尾加上"&"来将命令同时放入后台运行
但是关掉会话之后,进程就会被杀掉

1. nohup

1
2
3
4
nohup ping www.ibm.com &
# nohup 的用途就是让提交的命令忽略 hangup 信号
# 标准输出和标准错误缺省会被重定向到 nohup.out 文件中,也可使用 >filename 2>&1 来重定向文件名
nohup python3 run.py >/dev/null 2>&1 &

2. setsid

1
2
setsid ping www.ibm.com
# 将命令进程放到1号进程下面(就不是子进程了,不再接收hangup信号)

3. subshell 的小技巧

1
2
3
4
(ping www.ibm.com &)
# 括号包起来
# 让命令在subshell运行
# 父 ID(PPID)为1,不属于当前终端的子进程,从而也就不会受到当前终端的 HUP 信号的影响了。

相关链接

让进程在后台运行更可靠的几种方法

Technical Term

1
2
3
4
5
class 类
instance 实例
method 方法
subclass 子类
super class 父类

模块搜索路径

1
2
3
import sys
sys.path.append('模块路径')
# 这种方法是在运行时修改(添加),运行结束后失效

介绍模块相关一些知识

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
#!/usr/bin/env python3
# -*- coding: utf-8 -*-

' a test module '

__author__ = 'Jason '

import sys

def test():
args = sys.argv # 命令行参数,第一个默认为文件名
if len(args)==1:
print('Hello, world!')
elif len(args)==2:
print('Hello, %s!' % args[1])
else:
print('Too many arguments!')

if __name__=='__main__':
test()
1
2
#!/usr/bin/env python3
可以让文件直接在Unix/Linux/Mac上运行
1
2
# -*- coding: utf-8 -*-
统一使用标准utf-8编码
1
任何模块代码的第一个字符串都被视为模块的文档注释
1
__author__变量是作者

map

1
把func作用在iterator每一个元素上,并返回一个iterator

reduce

1
累积计算  # func(x,y)必须两个参数

filter

1
2
3
过滤函数 
# 把func作用在iterator每一个元素上,然后根据返回值是Ture还是False决定保留还是丢弃该元素
# 返回值是原来iterator的子序列

注意

1
2
map、reduce、filter都是两个参数(第一个func,第二个iterator)
返回值都是惰性iterator,需要list()才能转换为list