求助! Python 如何调用 C 语言.SO 动态库(并且被调用的动态库还依赖其它的动态库)

2018-05-28 11:28:51 +08:00
 wsds

最近要做 SDK 测试,SDK 是由 C 语言封装成的,函数封装在了 .so动态库中,而且该动态库还依赖其他的第三方动态库,不知道该怎么调用,求助各位大神,给个 demo

ps: linux平台的库

#call_c.py

import ctypes
from ctypes import *

ll = ctypes.cdll.LoadLibrary
lib = ll("./libIFaceRecSDK.so")

单独调用就大概是上面这样子了,但 libIFaceRecSDK.so 这个动态库还依赖其他库,且带有头文件时,我该怎么调用啊??你们看,这个库还依赖以下的这多么的库,还有头文件!!求救!

6883 次点击
所在节点    Python
24 条回复
mashiro233
2018-05-28 11:47:56 +08:00
遇到这种情况,我的解决方法就是要么自己封装一个 C 模块给 py,要么 FFI。前者偏向调用复杂模块的时候,后者偏向调用简单模块。
另外这个依赖和你没多少关系,系统或者 libffi 会帮你解决。

给个 CFFI 参考链接。
https://eli.thegreenplace.net/2013/03/09/python-ffi-with-ctypes-and-cffi/
wsds
2018-05-28 11:57:12 +08:00
@mashiro233
/尴尬,不会 C 语言,现在手里就一份 SDK 内置函数文档,还有些库文件,头文件,我要做的就是用 python 调 SDK 中的这些方法;
我看一下这个 FFI,谢谢哈
wellsc
2018-05-28 12:03:39 +08:00
Cffi 了解一下
wsds
2018-05-28 17:40:38 +08:00
@mashiro233 调不通啊,报错
qieqie
2018-05-28 18:33:50 +08:00
你只需要知道返回值和参数列表的类型,不需要头文件。
ld-linux.so 会帮你找到依赖。

```
import ctypes

lib = ctypes.CDLL('libm.so.6')
func = getattr(lib, 'pow')
func.argtypes = [ctypes.c_double, ctypes.c_double]
func.restype = ctypes.c_double

print (func(2, 10))
```
Shazoo
2018-05-28 18:51:45 +08:00
我觉得有点像比较初级的库路径问题。可能是你的 LD_LIBRARY 设置有些问题。
一般做这种工作的步骤大概是:
1、先写个 C 的程序,简单调用下 SDK 功能。如果 C 程序链接、运行有问题,那就及时解决。
2、C 程序搞定后,再弄成 python 库。

一般来说,库都会依赖不少第三方和系统库。这部分依赖,一般做的好的 SDK 是无需关心的。只要能搜索到就可以。做的不好的 SDK,也应该可以在步骤 1 解决掉。

参考下?
http://oi.0w0.io/2018/01/05/Ubuntu-16-04-Python3-%E9%85%8D%E7%BD%AESQLite3-%E7%9A%84-icu-%E5%88%86%E8%AF%8D/#more
wsds
2018-05-28 19:22:41 +08:00
@qieqie 首先就是用的 ctypes 的,调用 so 报错:undefined symbol: _ZN6apache6thrift12GlobalOutputE
然后才换的 cffi
wsds
2018-05-28 19:23:06 +08:00
@Shazoo 看报错像是路径有问题,但我代码跟 so 就在同级目录下啊,不该是路径问题啊
wsds
2018-05-28 19:24:08 +08:00
@qieqie 函数名,入参,出参,都有文档,可现在首要是调不通 so
wevsty
2018-05-28 19:34:26 +08:00
@wsds
undefined symbol: _ZN6apache6thrift12GlobalOutputE
表示的是找不到_ZN6apache6thrift12GlobalOutputE 这个函数,这个函数可能是哪个 so 中引用了,但是你系统里面没有。
google 一下这个函数大概是来自这些 so 文件
/usr/lib64/libthriftnb.so.0.0.0
/usr/lib/libthriftz-1.0.0-dev.so
所以你得看你系统是否安装了 libthrift ?或者版本是否正确?
gnaggnoyil
2018-05-28 19:37:53 +08:00
_ZN6apache6thrift12GlobalOutputE 不出意外应该是来自 Apache Thrift 的库的 symbol.LZ 你 Apache Thrift 装了没?

@wsds 人家说的是你那个库的依赖库没配置好,你为什么要强调你的"代码怎么着怎么着"......
lonccc
2018-05-28 19:38:57 +08:00
pybind11 了解一下
wsds
2018-05-28 19:49:37 +08:00
@wevsty lib 下有这个库 libthrift.so
wsds
2018-05-28 19:50:08 +08:00
@wevsty 不过,目录有什么要求吗?我是从别的地方拷过来的
wevsty
2018-05-28 20:11:30 +08:00
@wsds
Linux 默认不会搜索工作目录这样,如果 so 不是在 /usr/lib 或者 /lib 这样的地方,那么就需要你指定路径。
可以修改 /etc/ld.so.conf 这样来加载,但是既然缺少库,用包管理安装一下不是更好么?
wsds
2018-05-28 20:14:44 +08:00
@wevsty 我试过绝对路径,也是还是提示:undefined symbol: _ZN6apache6thrift12GlobalOutputE

```
from cffi import FFI

ffi = FFI()

# ffi.cdef()
lib_IOTCAPIs = ffi.dlopen("/home/install/linux/x86_64/lib/libthrift.so")
lib_IOTCAPIs = ffi.dlopen("/home/lib/cpu/libIFaceRecSDK.so")
print('Loaded lib {0}'.format(lib))
```
wevsty
2018-05-28 22:01:32 +08:00
@wsds 那只能说明这个函数可能不在这个 so 文件里,再去检查这个库有没有其他的 so 文件吧。
qieqie
2018-05-29 00:12:04 +08:00
@wsds export LD_LIBRARY_PATH=/home/install/linux/x86_64/lib:$LD_LIBRARY_PATH
Shazoo
2018-05-29 09:40:11 +08:00
@wsds 都说啦~写个简单 c 程序。调试通了,再跑 python。这样就能排除库本身问题。善用 readelf 之类的工具,导出引用表看看。

当然,如果已经通了,那就实验下 LD_LIBRARY
wsds
2018-05-29 10:55:04 +08:00
@Shazoo 加载通了,但调里边 C 的方法,提示 AttributeError: function

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/458300

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX