【C++】ReadFile概述,及实践使用时ReadFile的速率影响研究

news2024/11/20 19:29:17

ReadFile 函数概述

ReadFile 是 Windows API 函数,用于从文件或设备(如串口、硬盘等)中读取数据。它是同步和异步 I/O 操作的基础函数。


函数原型

BOOL ReadFile(
    _In_ HANDLE hFile,                          // 文件或设备句柄
    _Out_writes_bytes_to_opt_(nNumberOfBytesToRead, *lpNumberOfBytesRead) LPVOID lpBuffer, // 缓冲区
    _In_ DWORD nNumberOfBytesToRead,            // 预期读取的字节数
    _Out_opt_ LPDWORD lpNumberOfBytesRead,      // 实际读取的字节数,会根据_In_ DWORD返回给他
    _Inout_opt_ LPOVERLAPPED lpOverlapped       // 异步操作参数
);

参数详解

1. HANDLE hFile
  • 含义:表示需要读取数据的目标。
    • 可以是文件句柄(通过 CreateFile 打开)或设备句柄(如串口 COM1)。
    • 例如,读取串口数据时,通过 CreateFile 获得串口句柄。
2. LPVOID lpBuffer
  • 含义:指向缓冲区的指针ReadFile 将读取的数据存储到该缓冲区中。
  • 大小要求:缓冲区大小应至少等于 nNumberOfBytesToRead,以避免越界。
  • 类型LPVOID,可转换为任意指针类型(如 char*)。
3. DWORD nNumberOfBytesToRead
  • 含义:预期读取的字节数,ReadFile 尝试从文件或设备读取该数量的数据。
  • 注意
    • 如果设备中数据不足,ReadFile 可能返回成功,但实际读取的字节数会少于此值(存储在 lpNumberOfBytesRead 中)。
4. LPDWORD lpNumberOfBytesRead
  • 含义:指向 DWORD 类型变量的指针,用于接收实际读取的字节数。
  • 可选性
    • 如果为 nullptr,表示调用方不关心读取了多少字节(不推荐)。
  • 返回值含义
    • 函数执行成功后,该变量存储实际读取的字节数。
5. LPOVERLAPPED lpOverlapped
  • 含义:指向 OVERLAPPED 结构的指针,用于异步操作。
  • 同步与异步
    • nullptr:表示同步操作,ReadFile 会阻塞,直到读取完成或超时。
    • nullptr:表示异步操作,ReadFile 会立即返回,读取操作会在后台完成。

返回值

  • BOOL 类型
    • TRUE:读取成功。
    • FALSE:读取失败,调用 GetLastError 获取错误代码。
  • 常见错误
    • ERROR_HANDLE_EOF:已到达文件末尾(EOF)。
    • ERROR_IO_PENDING:对于异步操作,表示读取请求已提交但尚未完成。

函数用途

ReadFile 广泛用于以下场景:

  1. 文件读取:从文件系统中读取内容。
  2. 串口通信:读取串口(COM)数据,常用于嵌入式设备通信。
  3. 网络通信(通过设备接口):读取基于设备接口的网络数据。
  4. 传感器数据读取:从硬件传感器中读取数据。

示例 1:从文件读取数据

从文件中读取内容并打印到控制台:

#include <windows.h>
#include <iostream>

int main() {
    // 打开文件
    HANDLE hFile = CreateFile(
        "example.txt",          // 文件路径
        GENERIC_READ,           // 读取权限
        0,                      // 共享模式
        NULL,                   // 安全属性
        OPEN_EXISTING,          // 打开已存在的文件
        FILE_ATTRIBUTE_NORMAL,  // 属性
        NULL                    // 模板文件句柄
    );

    if (hFile == INVALID_HANDLE_VALUE) {
        std::cerr << "无法打开文件,错误代码:" << GetLastError() << std::endl;
        return 1;
    }

    // 读取数据
    char buffer[128] = {0};  // 数据缓冲区
    DWORD bytesRead = 0;     // 实际读取的字节数

    if (ReadFile(hFile, buffer, sizeof(buffer) - 1, &bytesRead, NULL)) {
        std::cout << "成功读取 " << bytesRead << " 字节:" << std::endl;
        std::cout << buffer << std::endl;
    } else {
        std::cerr << "读取失败,错误代码:" << GetLastError() << std::endl;
    }

    // 关闭文件
    CloseHandle(hFile);
    return 0;
}

示例 2:串口数据读取

从串口(如 COM2)读取数据:

#include <windows.h>
#include <iostream>

int main() {
    // 打开串口
    HANDLE hSerial = CreateFile(
        "COM2",                // 串口名称
        GENERIC_READ,          // 读取权限
        0,                     // 共享模式
        NULL,                  // 安全属性
        OPEN_EXISTING,         // 打开已存在的设备
        0,                     // 属性
        NULL                   // 模板文件句柄
    );

    if (hSerial == INVALID_HANDLE_VALUE) {
        std::cerr << "无法打开串口,错误代码:" << GetLastError() << std::endl;
        return 1;
    }

    // 设置串口参数(波特率、数据位等)
    DCB dcb = {0};//Device Control Block
    dcb.DCBlength = sizeof(DCB);
    GetCommState(hSerial, &dcb);
    dcb.BaudRate = CBR_9600;
    dcb.ByteSize = 8;
    dcb.Parity = NOPARITY;
    dcb.StopBits = ONESTOPBIT;
    SetCommState(hSerial, &dcb);

    // 读取数据
    char buffer[128] = {0};
    DWORD bytesRead = 0;

    if (ReadFile(hSerial, buffer, sizeof(buffer) - 1, &bytesRead, NULL)) {
        std::cout << "成功读取 " << bytesRead << " 字节:" << std::endl;
        for (DWORD i = 0; i < bytesRead; ++i) {
            std::cout << "0x" << std::hex << static_cast<int>(buffer[i]) << " ";
        }
        std::cout << std::endl;
    } else {
        std::cerr << "读取失败,错误代码:" << GetLastError() << std::endl;
    }

    // 关闭串口
    CloseHandle(hSerial);
    return 0;
}

异步模式的应用

如果需要非阻塞读取,可以使用 OVERLAPPED 结构:

OVERLAPPED overlapped = {0};
overlapped.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);

BOOL result = ReadFile(hFile, buffer, sizeof(buffer), &bytesRead, &overlapped);
if (!result && GetLastError() == ERROR_IO_PENDING) {
    // 等待异步操作完成
    WaitForSingleObject(overlapped.hEvent, INFINITE);
    GetOverlappedResult(hFile, &overlapped, &bytesRead, FALSE);
}

总结

ReadFile 的作用
  • 从文件或设备读取数据,支持同步和异步模式。
典型应用
  • 文件操作:读取文本或二进制文件。
  • 串口通信:读取嵌入式设备数据。
  • 网络设备:读取基于接口的网络数据。
优点
  • 支持异步 I/O,提高程序性能。
  • 适用于广泛的文件和设备操作场景。
注意事项
  • 缓冲区大小要足够大以避免数据截断。
  • 在串口和异步模式中,需额外配置超时和事件处理机制。

补充 1 关于DBC

DCB dcb = {0}; 是一个用于初始化并配置串口通信参数的结构体。在 Windows API 中,DCBDevice Control Block)结构用于描述串口通信的控制设置,例如波特率、数据位、停止位、校验位等。


DCB 结构的定义

typedef struct _DCB {
    DWORD DCBlength;       // 结构体的大小(字节)
    DWORD BaudRate;        // 波特率(如 9600、115200)
    DWORD fBinary: 1;      // 是否使用二进制模式
    DWORD fParity: 1;      // 是否启用校验位
    DWORD fOutxCtsFlow: 1; // 是否启用 CTS 流控制
    DWORD fOutxDsrFlow: 1; // 是否启用 DSR 流控制
    DWORD fDtrControl: 2;  // DTR 控制流设置
    DWORD fDsrSensitivity: 1; // DSR 灵敏度
    DWORD fTXContinueOnXoff: 1; // 接收 XOFF 后是否继续传输
    DWORD fOutX: 1;        // 是否启用 XON/XOFF 流控制(输出)
    DWORD fInX: 1;         // 是否启用 XON/XOFF 流控制(输入)
    DWORD fErrorChar: 1;   // 是否替换错误字符
    DWORD fNull: 1;        // 是否丢弃 NULL 字节
    DWORD fRtsControl: 2;  // RTS 控制流设置
    DWORD fAbortOnError: 1;// 是否在错误时中止读/写操作
    DWORD fDummy2: 17;     // 保留位
    WORD wReserved;        // 保留字段
    WORD XonLim;           // XON 限制
    WORD XoffLim;          // XOFF 限制
    BYTE ByteSize;         // 每个字节的数据位数(4-8 位)
    BYTE Parity;           // 校验类型(无、奇、偶、标记、空)
    BYTE StopBits;         // 停止位(1、1.5 或 2 位)
    char XonChar;          // XON 字符
    char XoffChar;         // XOFF 字符
    char ErrorChar;        // 替换错误的字符
    char EofChar;          // 文件结束符
    char EvtChar;          // 事件字符
    WORD wReserved1;       // 保留字段
} DCB, *LPDCB;

补充2 影响速度的因素

ReadFile 的读取缓冲区的速度和范围没有固定的值,它受到以下多个因素的影响。具体的读取速度和数据范围会因场景、设备以及系统配置而有所不同。


1. 影响 ReadFile 读取速度的因素

文件读取速度
  • 现代 SSD:几百 MB/s。
  • HDD:50-200 MB/s。
串口读取速度
  • 受波特率限制:
    • 9600 bps:约 960 字节/秒。
    • 115200 bps:约 11,520 字节/秒。
  • 调整缓冲区大小可以提升效率。
影响速度的关键点
  • 硬件速率:文件存储设备、串口波特率等。
  • 用户缓冲区大小:每次读取的最大字节数。
  • 系统缓冲区:通过 SetupComm 设置大小。
  • 数据可用性:设备中数据是否已准备好。

通过合理调整缓冲区大小与 I/O 参数,ReadFile 的速度可以优化到设备允许的上限范围。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2244208.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【vmware+ubuntu16.04】ROS学习_博物馆仿真克隆ROS-Academy-for-Beginners软件包处理依赖报错问题

首先安装git 进入终端&#xff0c;输入sudo apt-get install git 安装后&#xff0c;创建一个工作空间名为tutorial_ws&#xff0c; 输入 mkdir tutorial_ws#创建工作空间 cd tutorial_ws#进入 mkdir src cd src git clone https://github.com/DroidAITech/ROS-Academy-for-Be…

【澜舟科技-注册/登录安全分析报告】

前言 由于网站注册入口容易被机器执行自动化程序攻击&#xff0c;存在如下风险&#xff1a; 暴力破解密码&#xff0c;造成用户信息泄露&#xff0c;不符合国家等级保护的要求。短信盗刷带来的拒绝服务风险 &#xff0c;造成用户无法登陆、注册&#xff0c;大量收到垃圾短信的…

thinkphp6 --数据库操作 增删改查

一、数据库连接配置 如果是本地测试&#xff0c;它会优先读取 .env 配置&#xff0c;然后再读取 database.php 的配置&#xff1b; 如果禁用了 .env 配置&#xff0c;则会读取数据库连接的默认配置&#xff1a; # .env文件&#xff0c;部署服务器&#xff0c;请禁用我 我们可以…

Excel数据动态获取与映射

处理代码 动态映射 动态读取 excel 中的数据&#xff0c;并通过 json 配置 指定对应列的值映射到模板中的什么字段上 private void GetFreightFeeByExcel(string filePath) {// 文件名需要以快递公司命名 便于映射查询string fileName Path.GetFileNameWithoutExtension(fi…

Python学习29天

二分查找 # 定义函数冒泡排序法从大到小排列 def bbble_sort(list):# i控制排序次数for i in range(len(list) - 1):# j控制每次排序比较次数for j in range(len(list) - 1 - i):if list[j] < list[j 1]:list[j], list[j 1] list[j 1], list[j] # 定义二分查找函数 def…

nodemon入门介绍

以前&#xff0c;我们开发一个node后端服务时&#xff0c;每次更改文件&#xff0c;均需重启一下&#xff0c;服务才能生效。这使我们的开发效率降低了很多。nodemon的出现&#xff0c;可以随时监听文件的变更&#xff0c;自动重启服务&#xff0c;我们开发时只需关注代码即可&…

STM32设计防丢防摔智能行李箱-分享

目录 目录 前言 一、本设计主要实现哪些很“开门”功能&#xff1f; 二、电路设计原理图 1.电路图采用Altium Designer进行设计&#xff1a; 2.实物展示图片 三、程序源代码设计 四、获取资料内容 前言 随着科技的不断发展&#xff0c;嵌入式系统、物联网技术、智能设备…

Pytest 学习 @allure.severity 标记用例级别的使用

一、前言 使用allure.serverity注解&#xff0c;可以在allure报告中清晰的看到不同级别用例情况 使用等级介绍 allure提供的枚举类 二、等级介绍 二、等级介绍 blocker&#xff1a;阻塞缺陷&#xff08;功能未实现&#xff0c;无法下一步&#xff09; critical&#xff1a;…

[javascript]js的五子棋让红蓝双方自己跟自己下棋

运行效果&#xff08;这是未分出胜负&#xff09;&#xff1a; 这是分出胜负&#xff1a; 源代码&#xff0c;把下边的代码放到1.html&#xff0c;然后用浏览器打开&#xff0c;就可以&#xff0c;然后刷新网页&#xff1a; <!DOCTYPE html> <html><body>&l…

Go语言中AES加密算法的实现与应用

一、前言 在当今的软件开发领域&#xff0c;数据安全至关重要。加密技术作为保护数据机密性的关键手段&#xff0c;被广泛应用于各个方面。AES&#xff08;高级加密标准&#xff09;作为一种对称加密算法&#xff0c;以其高效性和安全性在众多加密场景中占据重要地位。本文将详…

单片机GPIO中断+定时器 实现模拟串口接收

单片机GPIO中断定时器 实现模拟串口接收 解决思路代码示例 解决思路 串口波特率9600bps,每个bit约为1000000us/9600104.16us&#xff1b; 定时器第一次定时时间设为52us即半个bit的时间&#xff0c;其目的是偏移半个bit时间&#xff0c;之后的每104us采样并读取1bit数据。使得…

近几年新笔记本重装系统方法及一些注意事项

新笔记本怎么重装系统&#xff1f; 近几年的新笔记本默认开启了raid on模式或vmd选项&#xff0c;安装过程中会遇到问题&#xff0c;新笔记本电脑重装自带的系统建议采用u盘方式安装&#xff0c;默认新笔记本有bitlocker加密机制&#xff0c;如果采用一键重装系统或硬盘方式安装…

【支持向量机(SVM)】:算法原理及核函数

文章目录 1 SVM算法原理1.1 目标函数确定1.2 约束条件优化问题转换1.3 对偶问题转换1.4 确定超平面1.5 计算举例1.6 SVM原理小节 2 SVM核函数2.1 核函数的作用2.2 核函数分类2.3 高斯核函数2.3 高斯核函数API2.4 超参数 γ \gamma γ 1 SVM算法原理 1.1 目标函数确定 SVM思想…

自动驾驶系列—告别眩光烦恼:智能大灯如何守护夜间行车安全

&#x1f31f;&#x1f31f; 欢迎来到我的技术小筑&#xff0c;一个专为技术探索者打造的交流空间。在这里&#xff0c;我们不仅分享代码的智慧&#xff0c;还探讨技术的深度与广度。无论您是资深开发者还是技术新手&#xff0c;这里都有一片属于您的天空。让我们在知识的海洋中…

无插件H5播放器EasyPlayer.js RTSP播放器chrome/edge等浏览器如何使用独立显卡

随着互联网的快速发展和视频内容的日益丰富&#xff0c;HTML5视频播放器已成为网页视频播放的主流技术。EasyPlayer.js播放器视频播放技术不仅支持多种浏览器和设备&#xff0c;还提供了丰富的功能和更好的用户体验。 那么chrome/edge等浏览器如何使用独立显卡&#xff1f; 在…

@Autowired 和 @Resource思考(注入redisTemplate时发现一些奇怪的现象)

1. 前置知识 Configuration public class RedisConfig {Beanpublic RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) {RedisTemplate<String, Object> template new RedisTemplate<>();template.setConnectionFactory(facto…

STM32低功耗设计NFC与无线距离感应智能钥匙扣-分享

目录 目录 前言 一、本设计主要实现哪些很“开门”功能&#xff1f; 二、电路设计原理图 1.电路图采用Altium Designer进行设计&#xff1a; 2.实物展示图片 三、程序源代码设计 四、获取资料内容 前言 智能钥匙扣作为一种小巧而实用的智能设备&#xff0c;凭借其便携性…

微服务day09

DSL查询 快速入门 GET /items/_search {"query": {"match_all": {}} } 叶子查询 GET /items/_search {"query": {"match_all": {}} }GET /items/_search {"query": {"multi_match": {"query": "脱…

Golang语言整合jwt+gin框架实现token

1.下载jwt go get -u github.com/dgrijalva/jwt-go2.新建生成token和解析token文件 2.1 新建common文件夹和jwtConfig文件夹 新建jwtconfig.go文件 2.2 jwtconfig.go文件代码 /* Time : 2021/8/2 下午3:03 Author : mrxuexi File : main Software: GoLand */ package jwtC…

Orcad 输出有链接属性的PDF

安装adobe pdf安装Ghostscript修改C:\Cadence\SPB_16.6\tools\capture\tclscripts\capUtils\capPdfUtil.tcl ​ 设置默认打印机为 Adobe PDF ​ 将Ghostscript的路径修改正确 打开cadence Orcad &#xff0c;accessories->candece Tcl/Tk Utilities-> Utilities->PD…