windows编程之文件操作
总所周知,C/C++标准库里边,有fread,fopen等文件操作,而在windows环境下,微软当然也给我们提供了这样的API,而且比C/C++运行库更好用,效率更高。
首先,要进行文件操作,你得打开一个文件或者创建一个文件,而CreateFile这个API就是用来打开和创建一个文件(因为在操作系统看来,系统上的所有东西都是一个个文件,因此,那些串口,硬盘都可以用CreateFile打开)。
若该函数失败,返回INVALID_HANDLE_VALUE,不返回NULL。成功返回文件的句柄。 打开完文件之后,就可以进行读写了,这涉及到ReadFile和WriteFile两个API,他们的声明如下:
HANDLE CreateFile( LPCTSTR lpFileName, // 文件名 DWORD dwDesiredAccess, // 访问权限 DWORD dwShareMode, // 共享模式 LPSECURITY_ATTRIBUTES lpSecurityAttributes, // 安全属性,一般为0 DWORD dwCreationDisposition, // 怎样创建 DWORD dwFlagsAndAttributes, // 文件属性和标志,若读写是要用到异步方式,则设为FILE_FLAG_OVERLAPPED HANDLE hTemplateFile // 模板创建文件的句柄,一般用不到,给0 );
若该函数失败,返回INVALID_HANDLE_VALUE,不返回NULL。成功返回文件的句柄。 打开完文件之后,就可以进行读写了,这涉及到ReadFile和WriteFile两个API,他们的声明如下:
BOOL ReadFile( HANDLE hFile, // 文件的句柄 LPVOID lpBuffer, // 用于接收数据的缓冲区 DWORD nNumberOfBytesToRead, // 指定读多少字节 LPDWORD lpNumberOfBytesRead, // 接收读取的字节数 LPOVERLAPPED lpOverlapped // 若该参不为NULL,则为异步方式读取 ); BOOL WriteFile( HANDLE hFile, // 文件的句柄 LPCVOID lpBuffer, // 指向要写入到文件的缓冲区 DWORD nNumberOfBytesToWrite, // 要写的字节数 LPDWORD lpNumberOfBytesWritten, // 已写入的字节数的指针 LPOVERLAPPED lpOverlapped // 是否已异步方式进行写入 );在读写的时候我们可以指定从哪里开始读或写,我们可以调用SetFilePointer这个函数,其声明如下:
DWORD SetFilePointer( HANDLE hFile, // 文件句柄 LONG lDistanceToMove, // 移动的字节数,低32位 PLONG lpDistanceToMoveHigh, // 若移动的字节数超过4GB,则高32位在此指定 DWORD dwMoveMethod // 在哪开始偏移,开头,结尾,当前 );若要读写整个文件,我们可以先获取一下整个文件的大小,我们可以调用GetFileSize这个函数,声明 如下:
DWORD GetFileSize( HANDLE hFile, // 文件的句柄 LPDWORD lpFileSizeHigh // 若文件大小超过4GB的大小,则该参数不应为0 );若你要进行异步方式读或写,则应该指定OVERLAPPED这个结构体,声明如下:
typedef struct _OVERLAPPED { ULONG_PTR Internal; //状态码(该参不能以命名进行理解,这是微软当时遗留的问题,最开始,微软并不是状态码,后来把它改了,命名却遗留了下来) ULONG_PTR InternalHigh; //指定了总共写入或读取的字节数(该参和上一个参数一样,都是微软的遗留问题) union { struct { DWORD Offset; //偏移量,从哪里开始读写,低32位 DWORD OffsetHigh; //偏移量,从哪里开始读写,高32位 } DUMMYSTRUCTNAME; PVOID Pointer; //指向文件的位置,操作系统使用 } DUMMYUNIONNAME; HANDLE hEvent; //系统的内核事件句柄 } OVERLAPPED, *LPOVERLAPPED;下面是一段简单的代码,对以上进行练习:
#include <windows.h> #include <string> using namespace std; int _tmain(int argc, _TCHAR* argv[]) { HANDLE hfile = CreateFile(_T("test.txt"), GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0, OPEN_EXISTING,FILE_FLAG_OVERLAPPED, 0); if (hfile == INVALID_HANDLE_VALUE) { printf("open file error "); return -1; } DWORD dwSize = 0; DWORD dwSizeHigh = 0; //获取文件的大小 dwSize = GetFileSize(hfile, &dwSizeHigh); // 创建缓冲区 byte * buf = new byte[dwSize+dwSizeHigh+1]; // 清空缓冲区 ZeroMemory(buf, dwSize + dwSizeHigh + 1); //从开头进行读写 SetFilePointer(hfile, 0, 0, FILE_BEGIN); DWORD dwRead = 0; // 使用异步方式读取文件 OVERLAPPED ov = { 0 }; ov.Offset = 2; // 读取的偏移 2 ov.OffsetHigh = 0; ov.hEvent = CreateEvent(0, true, false, 0); //创建一个事件内核对象 //读文件 ReadFile(hfile, buf, dwSize + dwSizeHigh, &dwRead, &ov); //因为是以异步方式进行读取的,所以dwRead读取的字节数应为0 if (dwRead == dwSize) { printf("read file"); } //我们要等待所有的文件大小读完,应调用下面这个等待函数 WaitForSingleObject(ov.hEvent, INFINITE); // INFINITE代表无限等待 //应为偏移量为2,所以读取的字节数应与文件的大小少2个字节 if (ov.InternalHigh != dwSize -2) { printf("read failed "); return -1; } //将读取的内容输出 printf("%s ", buf); CloseHandle(hfile); // 关闭句柄 return 0; }
声明:该文观点仅代表作者本人,牛骨文系教育信息发布平台,牛骨文仅提供信息存储空间服务。
- 上一篇: 结合MongoDB开发LBS应用
- 下一篇: MongoDB的地埋空间数据存储、空间索引以及空间查询