2016-12-06 182 views
0

我试图重现类似的事情在On Windows, how to open for writing a file already opened for writing by another process? 所以我后面的Piotr Dobrogost答案,从Using a struct as a function argument with the python ctypes module modyfing代码,更改标志和属性,以我的目的(从这里https://msdn.microsoft.com/en-us/library/windows/desktop/aa363858(v=vs.85).aspx这里http://unix.superglobalmegacorp.com/Net2/newsrc/sys/fcntl.h.html拍摄),并加入了Python (使用3.3版本)open功能: 从操作系统导入路径 从ctypes的导入* 从ctypes.wintypes导入*在Python中,如何传递ctypes文件描述符来打开文件?

GENERIC_READ = 0x80000000 
GENERIC_WRITE = 0x40000000 

FILE_SHARE_DELETE = 0x00000004 
FILE_SHARE_READ = 0x00000001 
FILE_SHARE_WRITE = 0x00000002 
FILE_SHARE_READ_WRITE = (FILE_SHARE_READ | FILE_SHARE_WRITE) 

OPEN_EXISTING = 3 

FILE_ATTRIBUTE_NORMAL = 128 
FILE_ATTRIBUTE_TEMPORARY = 256 

O_RDONLY = 0x0000  # open for reading only 
O_WRONLY = 0x0001  # open for writing only 
O_RDWR = 0x0002  # open for reading and writing 
O_ACCMODE = 0x0003  # mask for above modes 
O_APPEND = 0x0008  # set append mode 

INVALID_HANDLE_VALUE = -1 
LPOVERLAPPED = c_void_p 
LPSECURITY_ATTRIBUTES = c_void_p 

NULL = 0 
FALSE = BOOL(0) 
TRUE = BOOL(1) 

def CreateFile(filename, access, sharemode, creation, flags): 
    return HANDLE(windll.kernel32.CreateFileW(
     LPWSTR(filename), 
     DWORD(access), 
     DWORD(sharemode), 
     LPSECURITY_ATTRIBUTES(NULL), 
     DWORD(creation), 
     DWORD(flags), 
     HANDLE(NULL) 
    )) 


def translate_path(fpath): 
    fpath = path.abspath(fpath) 
    if fpath[len(fpath)-1] == '\\' and fpath[len(fpath)-2] == ':': 
     fpath = fpath[:len(fpath)-1] 
    return '\\??\\%s' % fpath 

link_name = 'G:\\MATLAB\\Chronos_Python\\test.txt' 
link_name = path.abspath(link_name) 

hFile = CreateFile(link_name, GENERIC_READ, FILE_SHARE_WRITE, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL) 

if hFile == HANDLE(INVALID_HANDLE_VALUE): 
    raise Exception('Failed to open directory for junction creation.') 

cFile = ctypes.cdll.msvcrt._open_osfhandle(hFile,O_RDONLY) 
pyFile = open(cFile,'r') 
pyFile.read() 
pyFile.close() 
ctypes.cdll.msvcrt._close(cFile) 

windll.kernel32.CloseHandle(hFile) 

但在该行pyFile = open(cFile,'r')我得到

OSError: [Errno 9] Bad file descriptor

这可能是非常基本的东西,因为我是一个Python的新手... 我非常感谢,如果有人能帮我修复它。

+0

这里最大的错误是调用msvcrt.dll中的'_open_osfhandle'函数 - Windows系统DLL的私有C运行时(很多MinGW滥用)。这个CRT与CPython无关。你应该使用Python的'msvcrt'模块来分配文件描述符。 – eryksun

+0

@eryksun谢谢!我已经使用了'msvcrt.open_osfhandle(hFile.value,O_RDONLY)'(我不得不使用'.value'来获得'int'类型,否则我得到TypeError),现在它工作。 – Remolek

回答

0

假设句柄文件正确生成,我认为你的问题可能是在这里:

pyFile = open(cFile,'r') 

你试图打开一个C运行时文件描述符,所以你需要使用在蟒蛇的fdopen功能:

pyFile = os.fdopen(cFile, 'r') 

你可能想看看这里的文档 - 蟒暴露了一些的MSVCRT功能通过标准库在Windows上: https://docs.python.org/3.6/library/msvcrt.html#msvcrt.open_osfhandle