0

0

如何正确设置 ctypes.CDLL 中任意函数的 restype 属性

心靈之曲

心靈之曲

发布时间:2026-01-02 19:14:02

|

277人浏览过

|

来源于php中文网

原创

如何正确设置 ctypes.CDLL 中任意函数的 restype 属性

在使用 c++types 调用 c/c++ dll 函数时,若通过 `mydll['func_name']` 动态获取函数对象,其 `restype` 设置不会持久生效;必须通过 `getattr(mydll, name)` 或点号访问(`mydll.func_name`)获取**同一函数指针实例**,才能成功配置返回类型。

在基于 ctypes.CDLL 构建通用 DLL 封装类时,一个常见且关键的痛点是:无法为运行时动态确定的函数名(如字符串变量)正确设置 restype 和 argtypes。例如,以下代码看似合理,实则无效:

import ctypes
mydll = ctypes.CDLL("mylib.dll")
func_name = "GET_LAST_MESSAGE"
mydll[func_name].restype = ctypes.c_char_p  # ❌ 失败:每次 mydll[...] 都返回新 FuncPtr 实例
message = mydll[func_name]()  # 仍按默认 c_long 解析返回值

根本原因在于 mydll['func_name'] 触发了 __getitem__ 方法,该方法每次调用都创建并返回一个全新的 _FuncPtr 对象,其 restype 始终重置为默认值(c_long)。而 mydll.func_name 或 getattr(mydll, func_name) 则通过 __getattribute__ 机制返回同一个缓存的函数对象,因此对 restype 的修改可持久生效。

✅ 正确做法:始终复用同一个函数对象引用

import ctypes

mydll = ctypes.CDLL("mylib.dll")
func_name = "GET_LAST_MESSAGE"

# ✅ 推荐:使用 getattr —— 标准、安全、符合 Python 惯例
func = getattr(mydll, func_name)
func.restype = ctypes.c_char_p
func.argtypes = None  # 如需设置参数类型
message = func()  # 返回 bytes(需 decode() 转 str)

# ✅ 等价写法(不推荐用双下划线)
# func = mydll.__getattribute__(func_name)

# ✅ 也可先点号访问再赋值(但需确保 func_name 是合法标识符)
# func = mydll.GET_LAST_MESSAGE

? 进阶封装示例:通用 DLL 函数调用器

一览AI绘图
一览AI绘图

一览AI绘图是一览科技推出的AIGC作图工具,用AI灵感助力,轻松创作高品质图片

下载
class DLLWrapper:
    def __init__(self, lib_path):
        self.dll = ctypes.CDLL(lib_path)

    def call_func(self, func_name, restype=None, argtypes=None, *args):
        """安全调用任意 DLL 函数"""
        func = getattr(self.dll, func_name)  # ✅ 关键:复用同一 FuncPtr
        if restype is not None:
            func.restype = restype
        if argtypes is not None:
            func.argtypes = argtypes
        return func(*args)

# 使用示例
wrapper = DLLWrapper("mylib.dll")
msg_bytes = wrapper.call_func(
    "GET_LAST_MESSAGE",
    restype=ctypes.c_char_p
)
message = msg_bytes.decode('utf-8') if msg_bytes else ""

⚠️ 注意事项:

  • 切勿混用访问方式:mydll['func'] 和 getattr(mydll, 'func') 返回的是不同对象,修改前者 restype 对后者无影响;
  • c_char_p 返回的是 bytes,C 函数返回的 C 字符串需确保以 \0 结尾,且内存由 DLL 管理(避免提前释放);
  • 若函数返回堆分配内存(如 malloc),建议改用 ctypes.POINTER(ctypes.c_char) + ctypes.cast 手动管理,或在 C 层提供配套的 free 函数;
  • Windows 下注意调用约定:CDLL 默认 cdecl,若 DLL 使用 stdcall(如 WinAPI),应改用 WinDLL。

总结:getattr(mydll, name) 是动态设置 restype/argtypes 的唯一可靠方式,它保证了函数对象的同一性。在设计可扩展的 ctypes 封装类时,应以此为基础构建抽象层,避免直接使用下标访问 mydll[...]。

相关专题

更多
js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

250

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

206

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1436

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

609

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

547

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

539

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

158

2025.07.29

c++字符串相关教程
c++字符串相关教程

本专题整合了c++字符串相关教程,阅读专题下面的文章了解更多详细内容。

77

2025.08.07

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

74

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 0.6万人学习

Django 教程
Django 教程

共28课时 | 2.7万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.0万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号