操作系统试验九、采用异步方式实现文件读/写

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

一、实验目的

(1)了解Windows系统异步文件读/写的概念。

(2)熟悉Windows系统文件读/写相关的API。

(3)掌握采用异步方式实现文件读/写的相关参数设置。

二、实验准备知识:文件异步传输及相关API函数介绍

  1. 文件异步传输基本原理

文件异步传输是一种改变指令执行顺序的机制。一般而言,指令是顺序执行的,下一条指令必须在上一条指令执行完毕才可执行。因此当CPU遇到一条访问磁盘的指令时。应用程序需要等待磁盘访问结束后才能进行后续的工作。但如果后续工作与访问磁盘操作无关,这样的等待就显得很没有必要。Windows XP 系统中提供了异步传输机制可以解决这个问题。它通过打开文件时设置标志位表明文件采用异步传输方式,这样,进程不等待读写操作而继续执行。当指令必须用到磁盘访问结果的数据时,可通过一条Wait指令进行等待。

文件异步传输时,访问磁盘指令和等待指令之间的指令与磁盘访问并发进行。从而大大加快了系统处理I/O的速度。

  1. 相关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()。

三、实验内容

(一)实验内容

建立一个函数,使用该函数将原文件source.txt中的内容读出。再写到目标文件overlapped.txt中去。

采用异步方式实现文件的读/写。

由于要采用异步方式对文件进行操作,在使用函数CreateFile()建立文件时其参数dwFlagsAndAttributes选用FILE-FLAG_NO_BUFFERING| FILE-FLAG_OVERLAPPED。

(二)主要代码


// 9.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"
#include "9.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","overlapped.txt");

	return nRetCode;
}

void FileReadWrite_Overlapped(char*source,char*destination)
{
	HANDLE handle_src,handle_dst;
	DWORD NumberOfByteWrite,NumberOfByteRead,Error;
	BOOL cycle;
	char*buffer;
	buffer=buf;
	OVERLAPPED overlapped;
	//创建文件
	handle_src=CreateFile(source,
		GENERIC_READ,
		0,
		NULL,
		OPEN_EXISTING,
		FILE_FLAG_NO_BUFFERING| FILE_FLAG_OVERLAPPED,
		NULL);
	handle_dst=CreateFile(destination,
		GENERIC_WRITE,
		NULL,
		NULL,
		CREATE_ALWAYS,
		NULL,
		NULL);
	if(handle_src==INVALID_HANDLE_VALUE || 
		handle_dst==INVALID_HANDLE_VALUE)
	{
		printf("File Create Fail!\n");
		exit(1);
	}
	cycle=TRUE;
	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=FALSE;
					break;
				case ERROR_IO_PENDING:
					if(!GetOverlappedResult(handle_src,
						&overlapped,
						&NumberOfByteRead,
						TRUE))
					{
						printf("GetOverlappedResult!%d\n",GetLastError());
						exit(1);
					}
					break;
				default:
					break;
			}			
		}
		if(NumberOfByteRead<BufferSize)
			cycle=FALSE;
		//写文件
		if(!WriteFile(handle_dst,buffer,NumberOfByteRead,&NumberOfByteWrite,NULL))
		{
			printf("Write File Error!%d\n",GetLastError());
			exit(1);
		}
	}
	CloseHandle(handle_src);
	CloseHandle(handle_dst);

}

四、实验结果与总结

image

该试验完成异步方式的文件读/写操作。先创建两个文件即source.txt和overlapped.txt,然后反复从文件source.txt中读取数据块,并写到overlapped.txt中去,直到文件尾结束。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 Java 中使用 gRPC 传输文件的异步方式,需要借助 gRPC 的流式传输能力。 以下是一个基本的 gRPC 文件传输的异步实现示例: 首先,在.proto文件中定义文件传输的方法: ``` service FileTransferService { rpc uploadFile(stream FileChunk) returns (UploadStatus); } message FileChunk { bytes data = 1; } message UploadStatus { bool success = 1; } ``` 接下来,在Java中实现异步上传文件的客户端: ``` // 定义异步上传文件的回调接口 public interface FileUploadCallback { void onSuccess(); void onError(Throwable t); } // 客户端异步上传文件 public class FileUploadClient { private final FileTransferServiceStub stub; public FileUploadClient(FileTransferServiceStub stub) { this.stub = stub; } public void uploadFile(File file, FileUploadCallback callback) { try { FileInputStream inputStream = new FileInputStream(file); StreamObserver<FileChunk> requestObserver = stub.uploadFile(new StreamObserver<UploadStatus>() { @Override public void onNext(UploadStatus status) { if (status.getSuccess()) { callback.onSuccess(); } else { callback.onError(new RuntimeException("Upload failed")); } } @Override public void onError(Throwable t) { callback.onError(t); } @Override public void onCompleted() { // Do nothing } }); byte[] buffer = new byte[4096]; int bytesRead = -1; while ((bytesRead = inputStream.read(buffer)) != -1) { FileChunk chunk = FileChunk.newBuilder().setData(ByteString.copyFrom(buffer, 0, bytesRead)).build(); requestObserver.onNext(chunk); } requestObserver.onCompleted(); } catch (IOException e) { callback.onError(e); } } } ``` 最后,在Java中实现异步接收文件的服务端: ``` // 服务端异步接收文件 public class FileTransferServiceImpl extends FileTransferServiceGrpc.FileTransferServiceImplBase { @Override public StreamObserver<FileChunk> uploadFile(StreamObserver<UploadStatus> responseObserver) { return new StreamObserver<FileChunk>() { private FileOutputStream outputStream; @Override public void onNext(FileChunk chunk) { try { if (outputStream == null) { File tempFile = File.createTempFile("temp", "upload"); outputStream = new FileOutputStream(tempFile); } outputStream.write(chunk.getData().toByteArray()); } catch (IOException e) { onError(e); } } @Override public void onError(Throwable t) { if (outputStream != null) { try { outputStream.close(); } catch (IOException e) { // Ignore } } responseObserver.onError(t); } @Override public void onCompleted() { try { outputStream.close(); responseObserver.onNext(UploadStatus.newBuilder().setSuccess(true).build()); responseObserver.onCompleted(); } catch (IOException e) { onError(e); } } }; } } ``` 以上就是一个基本的 gRPC 文件传输的异步实现示例。需要注意的是,在实际应用中,可能需要对上传和下载的文件进行分片,以便更好地管理和控制传输过程。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值