我可以明确关闭ctypes CDLL吗?

时间:2013-10-23 16:21:20

标签: python ctypes

我有一个用于与C库交互的Python 2.7 GUI。我在GUI中做了一堆设置,然后按“go”按钮。然后,我正在查看结果,不再需要库代码。但我希望在更改库时保持所有GUI状态。

我使用so导入dllctypes,这显然会打开文件进行阅读。但是,我想显式关闭该文件,以便重新编译并覆盖它。然后,当我再次按下“go”按钮时,我想导入新版本。

在最糟糕的情况下,我可以将文件复制到tempfile.NamedTemporaryFile,但后来我处理了几十个文件,其中没有一个我可以清理。

我可以以某种方式显式关闭文件句柄吗?或者,我可以将文件的内容读入StringIO对象并以某种方式指向ctypes吗?

1 个答案:

答案 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
>>>