我有一个用于与C库交互的Python 2.7 GUI。我在GUI中做了一堆设置,然后按“go”按钮。然后,我正在查看结果,不再需要库代码。但我希望在更改库时保持所有GUI状态。
我使用so
导入dll
或ctypes
,这显然会打开文件进行阅读。但是,我想显式关闭该文件,以便重新编译并覆盖它。然后,当我再次按下“go”按钮时,我想导入新版本。
在最糟糕的情况下,我可以将文件复制到tempfile.NamedTemporaryFile
,但后来我处理了几十个文件,其中没有一个我可以清理。
我可以以某种方式显式关闭文件句柄吗?或者,我可以将文件的内容读入StringIO
对象并以某种方式指向ctypes
吗?
答案 0 :(得分:5)
您需要关闭DLL的句柄,以便首先释放它以便您可以使用该文件,您需要获取该库的句柄,然后将其传递给Windows上的FreeLibrary
,然后您就可以您需要的DLL文件:
from ctypes import *
file = CDLL('file.dll')
# do stuff here
handle = file._handle # obtain the DLL handle
windll.kernel32.FreeLibrary(handle)
预览:
这是一个测试DLL:
#include <windows.h>
#include <stdio.h>
BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved) {
switch( fdwReason ) {
case DLL_PROCESS_ATTACH:
puts("DLL loaded");
break;
case DLL_PROCESS_DETACH:
puts("DLL unloaded");
break;
case DLL_THREAD_ATTACH:
break;
case DLL_THREAD_DETACH:
break;
}
return TRUE;
}
__declspec(dllexport) void function(void) {
puts("Hello");
}
预览:
>>> from ctypes import *
>>>
>>> file = CDLL('file.dll')
DLL loaded
>>>
>>> # now it's locked
...
>>> file.function()
Hello
0
>>> windll.kernel32.FreeLibrary(file._handle)
DLL unloaded
1
>>> # not it's unlocked
在Linux上使用dlclose
,它将是:
from ctypes import *
file = CDLL('./file.so')
# do stuff here
handle = file._handle # obtain the SO handle
cdll.LoadLibrary('libdl.so').dlclose(handle)
这是一个类似的共享对象:
#include <stdio.h>
__attribute__((constructor)) void dlentry(void) {
puts("SO loaded");
}
void function(void) {
puts("Hello");
}
__attribute__((destructor)) void dlexit(void) {
puts("SO unloaded");
}
预览:
>>> from ctypes import *
>>>
>>> file = CDLL('./file.so')
SO loaded
>>>
>>> file.function()
Hello
6
>>> cdll.LoadLibrary('libdl.so').dlclose(file._handle)
SO unloaded
0
>>>