Github

https://github.com/gongluck/Windows-Core-Program.git

//第10章 同步设备IO与异步设备IO.cpp: 定义应用程序的入口点。
// #include "stdafx.h"
#include "第10章 同步设备IO与异步设备IO.h" //可提醒IO回调
VOID WINAPI funComplete(
_In_ DWORD dwErrorCode,
_In_ DWORD dwNumberOfBytesTransfered,
_Inout_ LPOVERLAPPED lpOverlapped
)
{ } //IO完成端口工作线程
DWORD WINAPI workthread(LPVOID lpThreadParameter)
{
DWORD NumberOfBytesTransferred;
DWORD CompletionKey;
OVERLAPPED* pOverlapped;
BOOL bres = GetQueuedCompletionStatus(lpThreadParameter, &NumberOfBytesTransferred, &CompletionKey, &pOverlapped, INFINITE);
//GetQueuedCompletionStatusEx可获取多个IO请求结果
DWORD dres = GetLastError(); return 0;
} int APIENTRY wWinMain(_In_ HINSTANCE hInstance,
_In_opt_ HINSTANCE hPrevInstance,
_In_ LPWSTR lpCmdLine,
_In_ int nCmdShow)
{
//打开(创建)文件
HANDLE hFile = CreateFile(TEXT("第10章 同步设备IO与异步设备IO.cpp"), GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ, nullptr, OPEN_EXISTING, 0, nullptr);
if (hFile == INVALID_HANDLE_VALUE)
{
MessageBox(nullptr, TEXT("打开文件失败!"), TEXT("error"), MB_OK);
return 0;
} //文件类型
DWORD dres = GetFileType(hFile);
//dres == FILE_TYPE_DISK //文件大小
LARGE_INTEGER li;
BOOL bres = GetFileSizeEx(hFile, &li);//逻辑大小
li.LowPart = GetCompressedFileSize(TEXT("第10章 同步设备IO与异步设备IO.cpp"), (DWORD*)&li.HighPart);//物理大小 //文件指针
li.QuadPart = 0;
bres = SetFilePointerEx(hFile, li, nullptr, FILE_END);
bres = SetFilePointerEx(hFile, li, &li, FILE_CURRENT);
bres = SetEndOfFile(hFile);
li.QuadPart = 0;
bres = SetFilePointerEx(hFile, li, nullptr, FILE_BEGIN); //同步IO
char* buf = new char[100];
bres = ReadFile(hFile, buf, 100, &dres, nullptr);
delete []buf;
buf = nullptr; //刷新缓冲区
bres = FlushFileBuffers(hFile); //CancelSynchronousIo
//取消线程未完成的同步IO请求 //关闭文件
CloseHandle(hFile);
hFile = nullptr; //异步IO
HANDLE hFile2 = CreateFile(TEXT("第10章 同步设备IO与异步设备IO.cpp"), GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ, nullptr, OPEN_EXISTING, FILE_FLAG_OVERLAPPED, nullptr);
OVERLAPPED ol;
ol.Offset = 1024;
ol.OffsetHigh = 0;
ol.hEvent = CreateEvent(nullptr, FALSE, FALSE, nullptr);
char* buf2 = new char[1024];
bres = ReadFile(hFile2, buf2, 1024, nullptr, &ol);//异步方式总是返回FALSE
dres = GetLastError();
switch (dres)
{
case ERROR_IO_PENDING:
//IO请求添加成功
//WaitForSingleObject(hFile2, INFINITE);
WaitForSingleObject(ol.hEvent, INFINITE);
break;
case ERROR_INVALID_USER_BUFFER:
case ERROR_NOT_ENOUGH_MEMORY:
break;
case ERROR_NOT_ENOUGH_QUOTA:
break;
default:
break;
} //取消队列中的IO请求
//bres = CancelIo(hFile2);
bres = CancelIoEx(hFile2, &ol);
//bres = CancelIoEx(hFile2, nullptr);
//bres = CloseHandle(hFile2); if (ol.hEvent != nullptr)
{
CloseHandle(ol.hEvent);
ol.hEvent = nullptr;
} //可提醒IO
OVERLAPPED ol2;
ol2.Offset = 1024;
ol2.OffsetHigh = 0;
ol2.hEvent = nullptr;
dres = ReadFileEx(hFile2, buf2, 1024, &ol2, funComplete);//回调函数在同一个线程空间
//置为可提醒状态
//APC队列中只要有一个,线程就不会进入睡眠
dres = SleepEx(INFINITE, TRUE);
//WaitForSingleObjectEx
//WaitForMultipleObjectsEx
//SignalObjectAndWait
//GetQueuedCompletionStatusEx
//MsgWaitForMultipleObjectsEx
dres = GetLastError();//WAIT_IO_COMPLETION //手动添加一项到APC队列
//可以让目标线程结束睡眠
dres = QueueUserAPC((PAPCFUNC)funComplete, GetCurrentThread(), NULL); //IO完成端口
HANDLE iocp = CreateIoCompletionPort(hFile2, nullptr, 1, 2);
HANDLE hts[2];
for (int i = 0; i < 2; ++i)
hts[i] = CreateThread(nullptr, 0, workthread, iocp, 0, nullptr);
bres = PostQueuedCompletionStatus(iocp, 100, 1, &ol); //模拟发送IO请求完成
bres = ReadFile(hFile2, buf2, 1024, nullptr, &ol2);
dres = GetLastError(); WaitForMultipleObjects(2, hts, TRUE, INFINITE);
for (int i = 0; i < 2; ++i)
{
CloseHandle(hts[i]);
hts[i] = nullptr;
}
CloseHandle(iocp);
iocp = nullptr; CloseHandle(hFile2);
hFile2 = nullptr;
delete[]buf2;
buf2 = nullptr; system("pause");
return 0;
}

Windows核心编程:第10章 同步设备IO与异步设备IO的更多相关文章

  1. windows核心编程 第8章201页旋转锁的代码在新版Visual Studio运行问题

    // 全局变量,用于指示共享的资源是否在使用 BOOL g_fResourceInUse = FALSE; void Func1() { //等待访问资源 while(InterlockedExcha ...

  2. windows核心编程 第5章job lab示例程序 解决小技巧

    看到windows核心编程 第5章的最后一节,发现job lab例子程序不能在我的系统(win8下)正常运行,总是提示“进程在一个作业里”         用process explorer程序查看 ...

  3. windows核心编程---第七章 用户模式下的线程同步

    用户模式下的线程同步 系统中的线程必须访问系统资源,如堆.串口.文件.窗口以及其他资源.如果一个线程独占了对某个资源的访问,其他线程就无法完成工作.我们也必须限制线程在任何时刻都能访问任何资源.比如在 ...

  4. 《Windows核心编程》第九章——用内核对象进行线程同步

    先举一个有bug的例子: #include <iostream> #include <windows.h> #include <process.h> using n ...

  5. Windows核心编程 第十一章 线程池的使用

    第11章 线程池的使用 第8章讲述了如何使用让线程保持用户方式的机制来实现线程同步的方法.用户方式的同步机制的出色之处在于它的同步速度很快.如果关心线程的运行速度,那么应该了解一下用户方式的同步机制是 ...

  6. Windows核心编程 第七章 线程的调度、优先级和亲缘性(下)

    7.6 运用结构环境 现在应该懂得环境结构在线程调度中所起的重要作用了.环境结构使得系统能够记住线程的状态,这样,当下次线程拥有可以运行的C P U时,它就能够找到它上次中断运行的地方. 知道这样低层 ...

  7. Windows核心编程 第27章 硬件输入模型和局部输入状态

    第27章 硬件输入模型和局部输入状态 这章说的是按键和鼠标事件是如何进入系统并发送给适当的窗口过程的.微软设计输入模型的一个主要目标就是为了保证一个线程的动作不要对其他线程的动作产生不好的影响. 27 ...

  8. Windows核心编程 第十七章 -内存映射文件(下)

    17.3 使用内存映射文件 若要使用内存映射文件,必须执行下列操作步骤: 1) 创建或打开一个文件内核对象,该对象用于标识磁盘上你想用作内存映射文件的文件. 2) 创建一个文件映射内核对象,告诉系统该 ...

  9. Windows核心编程 第六章 线程基础知识 (上)

    第6章 线程的基础知识 理解线程是非常关键的,因为每个进程至少需要一个线程.本章将更加详细地介绍线程的知识.尤其是要讲述进程与线程之间存在多大的差别,它们各自具有什么作用.还要介绍系统如何使用线程内核 ...

  10. Windows核心编程 第三章 内核对象

    第3章内核对象 在介绍Windows API的时候,首先要讲述内核对象以及它们的句柄.本章将要介绍一些比较抽象的概念,在此并不讨论某个特定内核对象的特性,相反只是介绍适用于所有内核对象的特性. 首先介 ...

随机推荐

  1. ZIP压缩输入/输出流

    ZIP是压缩文件的格式,使用ZIP可以节省空间 java将压缩/解压缩文件的方法都封装在java.util.zip包下,java实现了I/O数据流和网络数据流的单一接口,所以实现起来比较容易. 主要的 ...

  2. 《面向对象程序设计(Java)》第四周学习总结

    第一部分 第四章部分理论知识 1.面向对象程序设计概述:java是完全面向对象的,必须熟悉OOP才能编写java程序. 类:由类构造对象的过程称为创建类的实例. 封装:封装是将数据和行为组合在一个包中 ...

  3. python 爬虫启航

    1. 使用excel(简单使用) 数据- 自网站-导入 2.you-get python爬虫入门 1.环境配置 python,request,lxml 2.原理 爬虫的框架如下: 1.挑选种子URL: ...

  4. NIO总结

    NIO主要有三大核心部分:Channel(通道),Buffer(缓冲区), Selector.传统IO基于字节流和字符流进行操作,而NIO基于Channel和Buffer(缓冲区)进行操作,数据总是从 ...

  5. HTML-全局属性 / 事件属性(转)

    拷贝自:< http://www.runoob.com > HTML 全局属性 New : HTML5 新属性. 属性 描述 accesskey 设置访问元素的键盘快捷键. class 规 ...

  6. 零基础爬虫----python爬取豆瓣电影top250的信息(转)

    今天利用xpath写了一个小爬虫,比较适合一些爬虫新手来学习.话不多说,开始今天的正题,我会利用一个案例来介绍下xpath如何对网页进行解析的,以及如何对信息进行提取的. python环境:pytho ...

  7. mongodb文件损坏的恢复--无可恢复数据

    1.mongodb 启动异常error code 100,检查日志,数据文件损坏 2 检查collection-15-6548623434943640018.wt 可恢复数据,为空,不存在恢复的数据 ...

  8. 大数据入门到精通12--spark dataframe 注册成hive 的临时表

    一.获得最初的数据并形成dataframe val ny= sc.textFile("data/new_york/")val header=ny.firstval filterNY ...

  9. 【笔记】计算机原理,网络,Linux操作系统

    一,计算机原理 1,化繁为简的思想,产生二进制,产生把所有计算归结为加法运算 二,网络 1,分层思想,产生ISO七层协议,商用了TCP/IP的五层协议 三,Linux操作系统 1,分层思想,硬件-&g ...

  10. 实验吧“解码磁带”的write up

    在“实验吧”的做CTF题时遇到的一道题,地址在这里:http://ctf5.shiyanbar.com/misc/cidai.html 因为正在学python,做这道题的时候正好用python写个简单 ...