windows 使用SetFileInformationByHandle移动文件

6uxekuva  于 2023-11-21  发布在  Windows
关注(0)|答案(3)|浏览(245)

我尝试使用SetFileInformationByHandle移动文件。Niall道格拉斯在他的CppCon 2015演讲“Racing The File System”中提出了这种技术,作为原子移动/重命名文件的一种方法。然而,我很难提供正确的参数;它总是失败,GetLastError返回ERROR_INVALID_PARAMETER
我已经尝试了以下设置,使用Unicode字符集:

  • VS2015 U1,在Windows 10下运行exe
  • VS2015 U2,在Windows Server 2012下运行exe
  • VS 2013,在Windows 7下运行exe

但是行为是一样的。我确保可以访问测试文件夹和测试文件。

  1. #include <sdkddkver.h>
  2. #include <windows.h>
  3. #include <cstring>
  4. #include <iostream>
  5. #include <memory>
  6. int main()
  7. {
  8. auto const& filepath = L"C:\\remove_tests\\file.txt";
  9. auto const& destpath = L"C:\\remove_tests\\other.txt";
  10. // unclear if that's the "root directory"
  11. auto const& rootdir = L"C:\\remove_tests";
  12. // handles will be leaked but that should be irrelevant here
  13. auto const f_handle = CreateFile(filepath,
  14. GENERIC_READ | GENERIC_WRITE | DELETE,
  15. 0,
  16. NULL,
  17. CREATE_ALWAYS,
  18. FILE_ATTRIBUTE_NORMAL,
  19. NULL);
  20. if (f_handle == INVALID_HANDLE_VALUE)
  21. {
  22. auto const err = GetLastError();
  23. std::cerr << "failed to create test file: " << err;
  24. return err;
  25. }
  26. auto const parent_dir_handle = CreateFile(rootdir,
  27. GENERIC_READ | GENERIC_WRITE,
  28. FILE_SHARE_READ | FILE_SHARE_WRITE,
  29. NULL,
  30. OPEN_EXISTING,
  31. FILE_ATTRIBUTE_NORMAL | FILE_FLAG_BACKUP_SEMANTICS,
  32. NULL);
  33. if (parent_dir_handle == INVALID_HANDLE_VALUE)
  34. {
  35. auto const err = GetLastError();
  36. std::cerr << "failed to get handle to parent directory: " << err;
  37. return err;
  38. }
  39. auto const destpath_bytes_with_null = sizeof(destpath);
  40. // unclear if we need to subtract the one wchar_t of FileNameLength:
  41. auto const struct_size = sizeof(FILE_RENAME_INFO) + destpath_bytes_with_null;
  42. auto const buf = std::make_unique<char[]>(struct_size);
  43. auto const fri = reinterpret_cast<FILE_RENAME_INFO*>(buf.get());
  44. fri->ReplaceIfExists = TRUE; // as described by Niall Douglas
  45. fri->RootDirectory = parent_dir_handle;
  46. // with or without null terminator?
  47. fri->FileNameLength = destpath_bytes_with_null;
  48. std::memcpy(fri->FileName, destpath, destpath_bytes_with_null);
  49. BOOL res = SetFileInformationByHandle(f_handle, FileRenameInfo,
  50. fri, struct_size);
  51. if (!res)
  52. {
  53. auto const err = GetLastError();
  54. std::cerr << "failed to rename file: " << err;
  55. return err;
  56. }
  57. else
  58. std::cout << "success";
  59. }

字符串
我特别提出的问题是:

  • FILE_RENAME_INFO要求的“根目录”是什么?
  • 句柄需要哪些权限?
  • SetFileInformationByHandle产生的ERROR_INVALID_PARAMETER的潜在问题是什么?
db2dz4w8

db2dz4w81#

我改变了几个想法:
1)我不使用根句柄(我将其设置为NULL)
2)我更改了您的FILE_RENAME_INFO内存分配代码
注意:在Windows 8中检查,在同一卷(磁盘)中移动文件

  1. auto const& filepath = L"C:\\remove_tests\\file.txt";
  2. auto const& destpath = L"C:\\remove_tests\\other.txt";
  3. // unclear if that's the "root directory"
  4. auto const& rootdir = L"C:\\remove_tests";
  5. // handles will be leaked but that should be irrelevant here
  6. auto const f_handle = CreateFile(filepath,
  7. GENERIC_READ | GENERIC_WRITE | DELETE,
  8. 0,
  9. NULL,
  10. CREATE_ALWAYS,
  11. FILE_ATTRIBUTE_NORMAL,
  12. NULL);
  13. if (f_handle == INVALID_HANDLE_VALUE)
  14. {
  15. auto const err = GetLastError();
  16. std::cerr << "failed to create test file: " << err;
  17. return err;
  18. }
  19. /*auto const parent_dir_handle = CreateFile(rootdir,
  20. GENERIC_READ | GENERIC_WRITE | DELETE,
  21. FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
  22. NULL,
  23. OPEN_EXISTING,
  24. FILE_ATTRIBUTE_NORMAL | FILE_FLAG_BACKUP_SEMANTICS,
  25. NULL);
  26. if (parent_dir_handle == INVALID_HANDLE_VALUE)
  27. {
  28. auto const err = GetLastError();
  29. std::cerr << "failed to get handle to parent directory: " << err;
  30. return err;
  31. }*/
  32. auto const destpath_bytes_withOUT_null = _tcslen(destpath);
  33. // unclear if we need to subtract the one wchar_t of FileNameLength:
  34. auto const struct_size = sizeof(FILE_RENAME_INFO) + (destpath_bytes_withOUT_null + 1) * sizeof(WCHAR);
  35. FILE_RENAME_INFO* fri = (FILE_RENAME_INFO*)new BYTE[struct_size];
  36. fri->ReplaceIfExists = TRUE; // as described by Niall Douglas
  37. fri->RootDirectory = NULL;//parent_dir_handle;
  38. // with or without null terminator?
  39. fri->FileNameLength = destpath_bytes_withOUT_null;// No include null
  40. _tcscpy_s(fri->FileName, destpath_bytes_withOUT_null + 1, destpath);
  41. BOOL res = SetFileInformationByHandle(f_handle, FileRenameInfo,
  42. fri, struct_size);
  43. delete fri;
  44. if (!res)
  45. {
  46. auto const err = GetLastError();
  47. std::cerr << "failed to rename file: " << err;
  48. return err;
  49. }
  50. else
  51. std::cout << "success";

字符串

展开查看全部
fykwrbwg

fykwrbwg2#

包含FileRenameInfoFILE_RENAME_INFOSetFileInformationByHandle的文档包含一些错误。FILE_RENAME_INFO.FileNameLength必须设置为复制到FILE_RENAME_INFO.FileName的字符数(不包括终止零),FILE_RENAME_INFO.RootDirectory必须为null,即使将文件从一个目录移动到另一个目录。

  1. #include <sdkddkver.h>
  2. #include <windows.h>
  3. #include <cstring>
  4. #include <iostream>
  5. #include <memory>
  6. int _tmain( int argc, _TCHAR* argv [] )
  7. {
  8. wchar_t* filename = L"C:\\remove_tests\\file.txt";
  9. wchar_t* destFilename = L"C:\\remove_tests2\\other.txt";
  10. // handles will be leaked but that should be irrelevant here
  11. auto fileHandle = CreateFile( filename,
  12. GENERIC_READ | GENERIC_WRITE | DELETE,
  13. FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
  14. NULL,
  15. OPEN_EXISTING,
  16. FILE_ATTRIBUTE_NORMAL,
  17. NULL );
  18. if ( fileHandle == INVALID_HANDLE_VALUE )
  19. {
  20. auto const err = GetLastError( );
  21. std::cerr << "failed to create test file: " << err;
  22. return err;
  23. }
  24. auto destFilenameLength = wcslen( destFilename );
  25. auto bufferSize = sizeof( FILE_RENAME_INFO ) + ( destFilenameLength*sizeof( wchar_t ));
  26. auto buffer = _alloca( bufferSize );
  27. memset( buffer, 0, bufferSize );
  28. auto const fri = reinterpret_cast<FILE_RENAME_INFO*>( buffer );
  29. fri->ReplaceIfExists = TRUE;
  30. fri->FileNameLength = destFilenameLength;
  31. wmemcpy( fri->FileName, destFilename, destFilenameLength );
  32. BOOL res = SetFileInformationByHandle( fileHandle, FileRenameInfo, fri, bufferSize );
  33. if ( !res )
  34. {
  35. auto const err = GetLastError( );
  36. std::cerr << "failed to rename file: " << err;
  37. return err;
  38. }
  39. else
  40. std::cout << "success";
  41. }

字符串

展开查看全部
xytpbqjk

xytpbqjk3#

有点偏离了主要问题,但是为什么不尝试C++17filesystem库作为标准的可移植方式呢?你可以简单地使用**std::filesystem::rename,MSVC也支持。我没有提供一个例子,因为它本质上是上面第一个链接中cppreference**例子的副本。它有一个更简单直接的API。

相关问题