windows编程之文件操作

来源:互联网 发布:淘宝手机详情视频时间 编辑:程序博客网 时间:2024/05/01 12:47
总所周知,C/C++标准库里边,有fread,fopen等文件操作,而在windows环境下,微软当然也给我们提供了这样的API,而且比C/C++运行库更好用,效率更高。
首先,要进行文件操作,你得打开一个文件或者创建一个文件,而CreateFile这个API就是用来打开和创建一个文件(因为在操作系统看来,系统上的所有东西都是一个个文件,因此,那些串口,硬盘都可以用CreateFile打开)。
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\n");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读取的字节数应为0if (dwRead == dwSize){printf("read file");}//我们要等待所有的文件大小读完,应调用下面这个等待函数WaitForSingleObject(ov.hEvent,  INFINITE); // INFINITE代表无限等待//应为偏移量为2,所以读取的字节数应与文件的大小少2个字节if (ov.InternalHigh != dwSize -2){printf("read failed\n");return -1;}//将读取的内容输出printf("%s\n", buf);CloseHandle(hfile); // 关闭句柄return 0;}


0 0