OS实验九:采用异步方式实现文件读写

09 采用异步方式实现文件读/写
4.3.1 实验目的
(1)了解Windows系统异步文件读/写的概念。
(2)熟悉Windows系统文件读/写相关的API。
(3)掌握采用异步方式实现文件读/写的相关参数设置。
4.3.2 实验准备知识:文件异步传输及相关API函数介绍

  1. 文件异步传输基本原理
    文件异步传输是一种改变指令执行顺序的机制。一般而言,指令是顺序执行的,下一条指令必须在上一条指令执行完毕才可执行。因此当CPU遇到一条访问磁盘的指令时。应用程序需要等待磁盘访问结束后才能进行后续的工作。但如果后续工作与访问磁盘操作无关,这样的等待就显得很没有必要。Windows XP 系统中提供了异步传输机制可以解决这个问题。它通过打开文件时设置标志位表明文件采用异步传输方式,这样,进程不等待读写操作而继续执行。当指令必须用到磁盘访问结果的数据时,可通过一条Wait指令进行等待。
    文件异步传输时,访问磁盘指令和等待指令之间的指令与磁盘访问并发进行。从而大大加快了系统处理I/O的速度。
  2. 相关API函数介绍
    函数GetOverlappedResult()返回指定文件 命名通道或通信设备上OVERLAPPED操纵的结果。
    原型:
    Bool GetOverlappedResult(
    HANDLE hFile //文件 命名通道或通信设备的句柄,
    LPOVERLAPPED lpOverlapped, //指向OVERLAPPED结构的指针
    LPDWORD lpNumberOfBytesTransferred, //指向实际传输字节数的指针
    BOOL bWait //等待标志
    );
    参数说明:
    (1) hFile:文件 命名通道或通信设备的句柄,。
    (2) lpOverlaooed:指向OVERLAPPED结构的指针。
    (3) lpNumberOfBytesTransferred,:32位变量指针, 指向实际传输字节数。
    (4) bWait : /等待标志。指定函数是否应等待被挂起的 要完成的OVERLAPPED操作。若为TURE,则OVERLAPPED操作完成之前该函数不返回;若为FASLE,则OVERLAPPED 被挂起,则函数返回FASLE,调用GetlastError()函数应返回ERROR_IO_INCOMPLETE。
    返回值:
    如果函数调用成功,则返回值为非0值。如果函数调用失败,则返回值为0.若要得到更多的错误信息,则调用函数GetLastError()。
    4.3.3 实验内容
    建立一个函数,使用该函数将原文件source.txt中的内容读出。再写到目标文件overlapped.txt中去。
    4.3.4 实验要求
    采用异步方式实现文件的读/写。
    4.3.5 实验指导
    由于要采用异步方式对文件进行操作,在使用函数CreateFile()建立文件时其参数dwFlagsAndAttributes选用FILE-FLAG_NO_BUFFERING| FILE-FLAG_OVERLAPPED。
    4.3.6 实验总结
    该试验完成异步方式的文件读/写操作。先创建两个文件即source.txt和overlapped.txt,然后反复从文件source.txt中读取数据块,并写到overlapped.txt中去,直到文件尾结束。
    4.3.7 源程序
    //File_Overlapped.cpp:Defines the entry point for the console application,
    //
    #include “stadfx.h”
    #include “File_Overlapped.h”
    #ifdef+DEBUG
    #define new DEBUG_NEW
    #undef THIS_FILE
    Static char THIS_FILE[]=FILE;
    #endif
    DWORD BufferSize=1024;
    Char buf[1024];

    //The one and only application object
    CWinApp theApp;
    Using namespace std;
    Void FileReadWrite_Overlapped(char source,char destination);
    Int_tmain(int argc,TCHAR
    argv[],TCHAR
    envp[])
    {int nRetCode=0;
    Printf(“Call FileReadWrite_Overlapped!\n”);
    FileReadWrite_Overlapped(“source.txt”,”nobuffer.txt”);
    Return nRetCode;
    }
    Void FileReadWrite_Overlapped(char *source,char *destination)
    { HANDLE handle_src,handle_dst;
    DWORD NumberOfByteRead,NumberOfByteWrite,Error;
    BOOL cycle;
    Char *buffer;
    buffer=buf;
    OVERLAPPED overlapped;
    //建立文件
    handle_src=CreateFile(spurce,
    GEBERIC_READ,
    0,
    NULL,
    OPEN_EXISTING,
    FILE-FLAG_NO_BUFFERING| FILE-FLAG_OVERLAPPED,
    NULL);
    Handle_dst=CreateFile(destination,
    GENERIC_WRITE,
    NULL,
    CREATE_ALWAYS,
    NULL,
    NULL);
    If (handl_srcINVALID_HANDLE+VALUE||
    Handle_dst
    INVALID_HANDLE+VALUE)
    {printf(“File Create Fail!\n”);
    exit(1);
    }
    cycle=TURE;
    overlapped.hEvent=NULL;
    overlapped.Offset=-BufferSize;
    overlapped.OffsetHigh=0;
    while (cycle)
    { overlapped.Offset= overlapped.Offset+BufferSize;
    NumberOfByteRead=BufferSize;
    //读文件
    if (!ReadFile(handle-src,
    buffer,
    NumberOfByteRead,
    & NumberOfByteRead,
    &overlapped))
    {switch(Error=GetLastError)
    {case ERROR_HANDLE_EOF; //若到文件尾
    cycle=FASLE;
    break;
    case ERROR_IO_PENDING; //若进程挂起
    if(!GetOverlappedResult(handle_src,
    &overlapped,
    &NumberOfByteRead,
    TURE))
    {printf(“GetOverlappedResult!%d\n”,GetLastError());
    exit(1);
    }
    break;
    default:
    break;
    }

}
If(NumberOfByteRead<BufferSize) cycle=FALE;
//写文件
if(!WriteFile(handle_dst,
buffer,
NumberOfByteRead,
&NumberOfByteWrite,
NULL))
{printf(“Write File Error!%d\n”,GetLastError());
exit(1);
}
}
//关闭句柄
CloseHandle(handle_src);
CloseHandle(handle_dst);
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值