SetFilePointer 使用

摘要:
如果lpDistanceToMoveHigh未传入NULL,则lpDistanceToMoveHigh和lDistanceToMove将形成一个有符号的64位值。lDistanceToMove参数用作此值的低32位,lpDistanceToMoveHigh用作高32位,即,lpDistance ToMoveHigh是lDistanceToMove的符号扩展。为了从0移动到2G,lpDistanceToMoveHigh必须设置为NULL或用作lDistanceToMove的符号扩展。为了支持64位文件指针,可以将LONG作为64位文件指示器的高位传递,并将其传递给lpDistanceToMoveHigh。当然,最好编写一个函数来替换SetFilePointer。

今天在使用SetFilePointer 的时候出现了溢出的问题,使用了SetFilePointerEx,出现了指针错位的问题,其实只要设置LONGLONG64位,然后分别设置SetFilePointer 的高低位不会出现溢出的问题

下面是转的一篇使用这个接口的博文

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;

}

本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/itlionwoo/archive/2006/04/11/659492.aspx

免责声明:文章转载自《SetFilePointer 使用》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇.NetCore 简单的使用中间件Java 两个日期间的天数计算下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

队列管理

1.队列的特性 队列可以保存有限个具有确定长度的数据单元。队列可以保存最大数目被称为队列的深度。在创建队列时需要确定队列的长度和每个单元的大小。 xQueueHandle xQueueCreate(unsigned portBASE_TYPE uxQueueLength,//the depth of the queue unsigend portBASE_...

Git知识总览(三) 分支的创建、删除、切换、合并以及冲突解决

  前两篇博客集中的聊了git的一些常用命令,具体请参见《Git知识总览(一) 从 git clone 和 git status 谈起》、《Git知识总览(二) git常用命令概览》。本篇博客主要涉及了在git版本管理中的分支的创建、切换以及合并。并且罗列了在merge分支使发生冲突时的解决方式。同时还介绍了如何删除本地分支以及远程分支。本篇博客除了参考...

C++第三十九篇 -- 研究一下Windows驱动开发(二)-- 驱动程序中重要的数据结构

数据结构是计算机程序的核心,I/O管理器定义了一些数据结构,这些数据结构是编写驱动程序时所必须掌握的。驱动程序经常要创建和维护这些数据结构的实例。 一、驱动对象(DRIVER_OBJECT) 每个驱动程序会有唯一的驱动对象与之对应,并且这个驱动对象是在驱动加载的时候,被内核中的对象管理程序所创建的。 驱动对象用DRIVER_OBJECT数据结构表示,它作为...

终于理解二级指针的作用了

之前学习swap函数时,知道传递指针可以实现对要交换变量本尊的修改,而直接传递值做不到这一点.究其原因,是因为函数传递参数时是以拷贝的形式,因此函数内部对其拷贝进行操作,不会影响到本尊. 如果想要通过函数实现对一级指针的值进行修改该如何去做呢?如果直接把它传进去,其实修改的是它的拷贝,而对它并没有影响.这个时候就是二级指针出场的时候了. #include...

IsBadStringPtr、IsBadWritePtr

判断调用进程是否拥有对指定字符串指针的读取权限,函数原型如下: BOOL IsBadStringPtr(     LPCTSTR lpsz,     UINT_PTR ucchMax); 参数: lpsz: 输入参数,指向字符串。 ucchMax:输入参数,读取字符串的最大长度。 返回值: 返回BOOL值,表示当前进程是否拥有字符串指针指向的字符串的度操作...

C++11新特性之十:enable_shared_from_this

enable_shared_from_this是一个模板类,定义于头文件<memory>,其原型为: template< class T > class enable_shared_from_this;       std::enable_shared_from_this 能让一个对象(假设其名为 t ,且已被一个 std::sha...