MoveFileEx 不返回结果

MoveFileEx 不返回,一直挂在哪。
但是在windows server 2012 是ok,只在windows server 2016发现此问题。

重命名某个文件时用到了windows API : MoveFileExA(oldpath, newpath, MOVEFILE_WRITE_THROUGH)

线程一直挂在那里,也不报错,也不返回,栈信息如下:

0000003e938fdeb8 00007ffb27b476c5 ntdll!NtSetInformationFile+0x14

0000003e938fdec0 00007ffb2aadd802 KERNELBASE!MoveFileWithProgressTransactedW+0x265

0000003e938fe070 00007ffb2aadd66a KERNEL32!MoveFileWithProgressTransactedA+0x8e

0000003e938fe0d0 00007ffb0d4abf90 KERNEL32!MoveFileExA+0x1a

操作系统安装在虚拟机上,而且是有时候发生.

MoveFile, MoveFileEx, CopyFile的几点心得 C++

MoveFile, MoveFileEx, CopyFile, 文件只读属性, 文件隐藏属性,

GetFileAttributes, SetFileAttributes, 文件属性的读取与设置

  1. 文件的只读和隐藏属性对函数的影响

1.1 移动操作(MoveFile和MoveFileEx)

testcase 1: 设源文件为c:/file1, 目标文件file2, 且目标文件不存在。

测试结果: file1的只读和隐藏属性对上面两个函数没有影响,函数返回后file1被更名为file2。文件的属性不变

testcase 2: 目标文件file2存在,属性为只读

测试结果: MoveFile失败,且与file2的只读属性无关,因为它不支持覆盖现有文件。

MoveFileEx的调用标志设为MOVEFILE_REPLACE_EXISTING时,file2的只读属性决定调用的成功与否, 和隐藏属性无关

1.2 拷贝操作(CopyFile)

MSDN在两个函数的文档中均有如下声明:

This function fails with ERROR_ACCESS_DENIED if the destination file already exists and has the FILE_ATTRIBUTE_HIDDEN or FILE_ATTRIBUTE_READONLY attribute set.

只需注意即可.

  1. 跨卷操作

1.1 移动操作

MoveFile支持跨卷操作,参见MSDN中的说明:

A new file may be on a different file system or drive. A new directory must be on the same drive

MoveFileEx也支持跨卷操作,不过要设置MOVEFILE_COPY_ALLOWED标志

1.2 拷贝操作

支持跨卷操作

  1. 设置文件属性的方法,以只读属性为例

文件属性操作使用SetFileAttributes和GetFileAttributes函数

具体说明参见MSDN

操作实例

// 测试某个属性是否被设置,以只读属性为例,其它类同

// 获取现有的文件属性

CString strPathName = _T("path and name of some file");

TCHAR szErr[255];

DWORD dwAttrs = ::GetFileAttributes(strPathName);

if(dwAttrs != INVALID_FILE_ATTRIBUTES)

{

if(dwAttrs & FILE_ATTRIBUTE_READONLY)

{

    // 有只读属性

    // 取消只读属性

    dwAttrs &= ~FILE_ATTRIBUTE_READONLY;

}

else

{

    // 无只读属性

    // 设置只读属性

    dwAttrs |= FILE_ATTRIBUTE_READONLY;

    // 将属性设置到文件

    if(!::SetFileAttributes(strPathName, dwAttrs))

    {

       // 打印错误信息

       ::FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, NULL,                                   ::GetLastError(), 0, szErr, 255, NULL);

    }

}

}

else

{

::FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, NULL,                                   ::GetLastError(), 0, szErr, 255, NULL);

}

操作结果:

如果文件有只读属性,该属性将被取消,否则,该属性被设置

备注:

设置文件属性前要先获取当前的文件属性,然后再当前属性的基础上进行设置某或取消某项属性的操作,以免使其它属性被无意取消.

  1. 总结

使用MoveFileEx的时候,要考虑目标文件的只读属性,使用CopyFile的时候,要考虑目标文件的只读和隐藏属性.

MoveFile不支持覆盖目标文件的操作

MoveFile支持跨卷操作,可以把文件移动到另一个逻辑分区或物理介质上,MoveFileEx也支持同样的操作,但设置MOVEFILE_COPY_ALLOWED标志位

CopyFile支持跨卷操作

本文没有涉及CopyFileEx, 调用它的代码在我的环境里不能成功编译.

在文件操作中包含目标文件的时候,一定要考虑到文件的只读和隐藏属性.确定要执行拷贝和移动到目标文件的时候,可以先把它的只读和隐藏属性去除,操作完成后,根据需求决定是否再把属性恢复回去. 还要注意目标文件是否在另外的卷上.

MoveFileEx函数功能很强大,但它只支持WindowsXP及更高版本的操作系统.所以对MoveFileEx需要谨慎使用.

CopyFileEx, GetFileAttributesEx, SetFileAttributesEx支持Windows98及更高版本的操作系统.

文件操作十分常用,注意这些细节,否则程序可能因为一个移动文件操作失败而崩溃.或因使用了高级函数而根本不能在宿主系统上运行

关键字:

MoveFile, MoveFileEx, CopyFile, 文件只读属性, 文件隐藏属性,

GetFileAttributes, SetFileAttributes, 文件属性的读取与设置

  1. 文件的只读和隐藏属性对函数的影响

1.1 移动操作(MoveFile和MoveFileEx)

testcase 1: 设源文件为c:/file1, 目标文件file2, 且目标文件不存在。

测试结果: file1的只读和隐藏属性对上面两个函数没有影响,函数返回后file1被更名为file2。文件的属性不变

testcase 2: 目标文件file2存在,属性为只读

测试结果: MoveFile失败,且与file2的只读属性无关,因为它不支持覆盖现有文件。

MoveFileEx的调用标志设为MOVEFILE_REPLACE_EXISTING时,file2的只读属性决定调用的成功与否, 和隐藏属性无关

1.2 拷贝操作(CopyFile)

MSDN在两个函数的文档中均有如下声明:

This function fails with ERROR_ACCESS_DENIED if the destination file already exists and has the FILE_ATTRIBUTE_HIDDEN or FILE_ATTRIBUTE_READONLY attribute set.

只需注意即可.

  1. 跨卷操作

1.1 移动操作

MoveFile支持跨卷操作,参见MSDN中的说明:

A new file may be on a different file system or drive. A new directory must be on the same drive

MoveFileEx也支持跨卷操作,不过要设置MOVEFILE_COPY_ALLOWED标志

1.2 拷贝操作

支持跨卷操作

  1. 设置文件属性的方法,以只读属性为例

文件属性操作使用SetFileAttributes和GetFileAttributes函数

具体说明参见MSDN

操作实例

// 测试某个属性是否被设置,以只读属性为例,其它类同

// 获取现有的文件属性

CString strPathName = _T("path and name of some file");

TCHAR szErr[255];

DWORD dwAttrs = ::GetFileAttributes(strPathName);

if(dwAttrs != INVALID_FILE_ATTRIBUTES)

{

if(dwAttrs & FILE_ATTRIBUTE_READONLY)

{

    // 有只读属性

    // 取消只读属性

    dwAttrs &= ~FILE_ATTRIBUTE_READONLY;

}

else

{

    // 无只读属性

    // 设置只读属性

    dwAttrs |= FILE_ATTRIBUTE_READONLY;

    // 将属性设置到文件

    if(!::SetFileAttributes(strPathName, dwAttrs))

    {

       // 打印错误信息

       ::FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, NULL,                                   ::GetLastError(), 0, szErr, 255, NULL);

    }

}

}

else

{

::FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, NULL,                                   ::GetLastError(), 0, szErr, 255, NULL);

}

操作结果:

如果文件有只读属性,该属性将被取消,否则,该属性被设置

备注:

设置文件属性前要先获取当前的文件属性,然后再当前属性的基础上进行设置某或取消某项属性的操作,以免使其它属性被无意取消.

  1. 总结

使用MoveFileEx的时候,要考虑目标文件的只读属性,使用CopyFile的时候,要考虑目标文件的只读和隐藏属性.

MoveFile不支持覆盖目标文件的操作

MoveFile支持跨卷操作,可以把文件移动到另一个逻辑分区或物理介质上,MoveFileEx也支持同样的操作,但设置MOVEFILE_COPY_ALLOWED标志位

CopyFile支持跨卷操作

本文没有涉及CopyFileEx, 调用它的代码在我的环境里不能成功编译.

在文件操作中包含目标文件的时候,一定要考虑到文件的只读和隐藏属性.确定要执行拷贝和移动到目标文件的时候,可以先把它的只读和隐藏属性去除,操作完成后,根据需求决定是否再把属性恢复回去. 还要注意目标文件是否在另外的卷上.

MoveFileEx函数功能很强大,但它只支持WindowsXP及更高版本的操作系统.所以对MoveFileEx需要谨慎使用.

CopyFileEx, GetFileAttributesEx, SetFileAttributesEx支持Windows98及更高版本的操作系统.

文件操作十分常用,注意这些细节,否则程序可能因为一个移动文件操作失败而崩溃.或因使用了高级函数而根本不能在宿主系统上运行

http://www.xuebuyuan.com/1547393.html

这个不返回的问题在 windows server 2012是OK,windows server 2016有问题。