SetFilePointer 文件定位函数的使用---重点在于对大文件(2G/4G以上)的支持
来源:互联网 发布:维多利亚贝克汉姆知乎 编辑:程序博客网 时间:2024/05/01 13:11
近期在做关于文件断点续传的功能,发现SetFilePointer ()函数先是对大于2G文件断点定位溢出,对断点值做了位运算后又变成大于4G溢出,反复研究了好几天,今天终于搞定,做个总结记录一下。
下面是转的一篇介绍此函数的博文,很受启发:
DWORD SetFilePointer(
HANDLE hFile,
LONG lDistanceToMove,
PLONG lpDistanceToMoveHigh,
DWORD dwMoveMethod
);
文中对于第二个和第三个参数进行了详细的说明:
lpDistanceToMoveHigh 参数是用来管理大文件,如果要移到文件中任何位置,我们就必须设置这个参数的值。假如我们传入NULL值,那么lDistanceToMove 的最大值是2^31–2(2G-2),因为所有文件指针的值是有符号的。因此,就算文件只有很少的机会能够达到这个大小,我们最好还是把文件当成是一个大文件,并且在程序中使用64位的指针(就是lpDistanceToMoveHigh的值不是NULL)。如果我们有一个压缩的并且文件很少的NTFS文件系统中,即使当前盘的空间不是很大的情况下,也很可能会有一些大文件。
假如lpDistanceToMoveHigh传入的不是NULL,那么lpDistanceToMoveHigh和lDistanceToMove会组成一个有符号的64位值。lDistanceToMove参数是被作为这个值的低32们,lpDistanceToMoveHigh作为高32位,也就是说lpDistanceToMoveHigh是lDistanceToMove的符号扩展名。
为了从0移动到2G位置,lpDistanceToMoveHigh必须设置为NULL或当作lDistanceToMove的符号扩展名。为了移动到大于2G的位置,就要使用lpDistanceToMoveHigh和lDistanceToMove合成一个有符号的64位值。举个例子:为了从2G位置移到4G的位置,我们需要设置lpDistanceToMoveHigh的值为0或-1,让它作为lDistanceToMove的符号扩展名。
为了支持64位文件指针,你可以传一个LONG,把它当作64位文件指针的高位,并把它传给lpDistanceToMoveHigh。这就意味着你必须把两个不同的变量当作一个操作单元,要不然这可能会出错。最好还是使用LARGE_INTEGER结构来创建一个64位值,并且把其中两个union元素作为参数传入。
当然,最好还是写一个函数来替代SetFilePointer。下面是一个代码例子,用来演示你想要的功能。
__int64 myFileSeek (HANDLE hf, __int64 distance, DWORD MoveMethod)
{
LARGE_INTEGER li;
li.QuadPart = distance;
li.LowPart = SetFilePointer (hf,
li.LowPart,
&li.HighPart,
MoveMethod);
if (li.LowPart == INVALID_SET_FILE_POINTER && GetLastError()
!= NO_ERROR)
{
li.QuadPart = -1;
}
return li.QuadPart;
}
博客地址:http://www.cnblogs.com/rainbowzc/archive/2011/06/16/2083192.html
以下是我实现的方法:
int64 m_CurrPosition ; //定位的断点值
LARGE_INTEGER li;
li.QuadPart = m_CurrPosition;
li.LowPart = SetFilePointer(hRecvFile, li.LowPart, &li.HighPart, FILE_BEGIN);
if (li.LowPart == INVALID_SET_FILE_POINTER && GetLastError()!=NO_ERROR)
//张帅:大文件传输断点续传(0--4G)功能完善
// LONG lMoveHigh = (LONG)(m_CurrPosition >> 32); //张帅添加
// if (::SetFilePointer(hRecvFile, m_dwCurrPosition & 0xffffffff, &lMoveHigh, FILE_BEGIN) != m_dwCurrPosition) //张帅:支持断点位置小于4G的文件断点续传
//
// if (::SetFilePointer(hRecvFile, m_dwCurrPosition , 0, FILE_BEGIN) != m_dwCurrPosition) //张帅:只支持断点位置小于2G的文件断点续传
{
li.QuadPart = -1;//zhansghuai:大于4G断点续传
//移动偏移位置出错
if (hRecvFile)
{
::CloseHandle(hRecvFile);
hRecvFile = INVALID_HANDLE_VALUE;
}
return;
} else
{
::SetEndOfFile(hRecvFile); //设置文件大小
}
- SetFilePointer 文件定位函数的使用---重点在于对大文件(2G/4G以上)的支持
- 如何让intel fortran支持2G以上大文件
- aix无法创建2G以上大文件的问题
- 32位Linux下使用2G以上大文件的几个相关宏的关系
- fopen打开2G以上大文件
- PHP 大文件下载,文件传输,支持断点续传。 2g以上超大文件也有效
- 使用MappedByteBuffer读取大文件(1G以上)和释放MappedByteBuffer的资源
- 为什么电脑磁盘无法支持下载4G以上的文件
- 32位Linux下操作64位(4G以上)大文件的总结
- linux下使用stat函数不可以取得超过2G的大文件的解决方法
- 如何用U盘拷贝4G以上的文件?
- u盘无法存放单个4g以上的文件
- ULONGLONG类型的使用,适合存储文件的大小(4G以上的)
- 用fopen打开2G以上大文件
- WinCE支持大容量(4G及以上)SD卡的解决办法
- Windows上快速创建一个1G以上的大文件
- 超大文件下载类 支持2g以上文件 支持断点续传
- 关于fseek不能定位大于2G文件的问题
- IntelliJ IDEA 将 Maven 构建的 Java 项目打包
- HDU3191 次短路条数
- 十进制转化为其他进制(C语言)
- Vim快捷键笔记
- centos 解决在编译的时候打印乱码的问题
- SetFilePointer 文件定位函数的使用---重点在于对大文件(2G/4G以上)的支持
- CMake学习
- ip地址分类及各网段私有地址
- Android内存泄漏case
- android源码下载和编译
- 改变状态栏的颜色
- Android设置横屏或竖屏
- JDBC连接执行 MySQL 存储过程报权限错误:User does not have access to metadata required to determine stored procedur
- Unity3D占用内存太大的解决方法