windows中Read函数引发数据异常的问题
【摘要】
在Window C/S开发中少不了客户端与服务端数据通信的情况,每当客户端从服务端获取数据时会将数据读到本地本件或缓存中,例如通过CInternetFile类的Read函数会将网卡缓存中的数据读入到制定的缓存中;在某APP通讯录中会很频繁地用到该函数去读取数据,然而由于当初没有细究该函数的使用情况,会意想不到地造成了客户端UI显示的内容异常,比如读取xml格式的内容时出现多余的字符串或者显示的内容为空。为此,本文将结合该函数的源码和MSDN资料以及实验结果来说明该函数的正确使用方法。
【关键字】
Windows、CInternetFile、Read
【类别】
大分类:软件; 小分类:代码优化
一、引出问题
在某APP中查看组织通讯录时有的成员节点或者部门节点有时候会出现多余的字符串或者为空信息,当出现该情况时刷新一下数据又正常了;后续该现象复现的频率就很高了,几乎每次都会发生了。接下来,就针对该情况做数据分析。
二、发现问题
由于该模块的数据量比较大,单步调试的话不好弄,只能通过添加打印日志来跟踪。首先,在数据显示之前将名称、职位等信息打印出来,当出现该异常时取出日志,发现显示的数据本身就有问题,接着对比一下数据库里的内容,发现数据库里存的数据也是异常数据,这个时候就需要在解析数据那块添加打印日志了,当初发现数据解析模块没有对局部变量进行初始化操作,想当然地以为是未初始化引起的,但是当对相关变量初始化操作后进行试验,该异常情况依旧出现,所以为初始化操作不是问题的原因;这个时候又开始怀疑是数据解析方法的问题,但是逆推思想推翻了该猜疑,要是是数据解析的方法有问题的话,那么所有的数据显示会异常,而不是只有少数部分数据异常;既然排除了数据解析方面,那么在数据解析之前对数据进行打印,后来实验发现确实是输入的数据有问题;而这个数据的来源是从另一个线程发送过来的,所以只能去数据源的线程查找问题的根源了。
三、找出原因
通过阅读代码流程可知数据的来源于CInternetFile::Read()函数获取到的数据,代码如下:
string strHtml = ""; char tChars[+] = {}; while ((pFile->Read((void*)tChars, )) > ) { strHtml += tChars; memset(tChars, , + ); }
初步看着这代码,觉得是没啥问题的,每次读取到的数据放在tChars数组里,下次读取数据之前都会置零处理,这样看似完美的流程是不会被怀疑的,可事实就是让人陷阱这个“美人计”中,导致数据异常。通过MSDN资料了解到,Read()函数第一个参数用于存放读取到的数据,第二个参数为每次读取数据的长度,它的返回值为实际读取到的数据。到此,对问题有点眉头了,也就是说Read函数第一个参数存储的是数据内容,其长度为第二个参数值;当第二个参数值和该函数的返回值不相等时,那么第一参数存储的内容就有一些内容时多余的,假设第二个参数值为InputValue,返回值为OutputValue,那么tChars数组中存有(InputValue—OutputValue)个多余的字符,并且是在数组的后面。带着这个疑惑,进入Read()函数源码分析,发现该函数调用的是InternetReadFile()函数,此函数最后一个参数就是实际读取数据的长度,为了证实我们对Read()函数第二个参数和返回值不相等的假设,那么就在该处添加断点进行调试。最后,通过实验证实了上述假设,InternetReadFile()函数返回的实际数据长度与Read()函数第二个参数值有时候会不相等。此外,通过实验发现一个有趣的问题,就是Read()函数读取到的数据若是存有多余的字符或者字符串的话,那这些多余的字符或者字符串就是Read()函数读取到的有效数据内容最后字符或字符串的重复。现在问题缘由被找到了,接下来就如何解决问题了,请看后续部分。
四、解决问题
既然Read()函数读取到内容有两个数据长度值,那么我们就采用实际数据长度值即可,将每次读取到的数据再进行一次拷贝操作即可,方法如下:
string strHtml = ""; char tChars[ + ] = {}; char tCharsTmp[ + ] = {}; while (nReadSize > ) { nReadSize = ; memset(tChars, , + ); memset(tCharsTmp, , + ); nReadSize = pFile->Read((void*)tCharsTmp, ); memcpy(tChars, tCharsTmp, nReadSize); strHtml += tChars; } 或者 while (nReadSize > ) { nReadSize = pFile->Read((void*)tCharsTmp, ); strHtml.assign(tCharsTmp, , nReadSize); }
通过该方式的修改后进行试验操作,没有发现组织通讯录中存在数据显示异常的问题了;因此,问题得解。
五、得出结论
在引用此类API函数时最好要考虑到其数据的实际长度值,然后通过其实际的数据长度来拷贝出新的一份数据,这样可以去除原有数据中多余的内容。
------20191228闪
windows中Read函数引发数据异常的问题的更多相关文章
- 类中成员函数与数据成员private/pubic/protected
类中成员函数与数据成员private/pubic/protected
- 在MS单元测试中引发期望异常
首先准备一个引发异常的方法. public static void ThrowException() { throw new ArgumentException(); } 然后在单元测试项目中,写下测 ...
- antd pro中如何使用mock数据以及调用接口
antd pro的底层基础框架使用的是dva,dva采用effect的方式来管理同步化异步 在dva中主要分为3层 services models components models层用于存放数据 ...
- 关于调试器中int3断点引发异常的思考
INT3断点 INT3断点是利用0Xcc指令实现的,cpu在执行0xcc指令时会引发断点异常调试器会捕捉这个异常. INT3断点引发的异常属于陷阱型异常,在执行完0xcc指令后eip指向下一条指令.但 ...
- windows环境下nutch2.x 在eclipse中实现抓取数据存进mysql详细步骤
nutch2.x 在eclipse中实现抓取数据存进mysql步骤 最近在研究nutch,花了几天时间,也遇到很多问题,最终结果还是成功了,在此记录,并给其他有兴趣的人提供参考,共同进步. 对nutc ...
- WPF关于“在“System.Windows.Markup.StaticResourceHolder”上提供值时引发了异常。”问题解决办法
在WPF中添加样式,在MainWindow.xaml使用自定义按钮FButton时报错,报错信息如下: "System.Windows.Markup.XamlParseException&q ...
- Linux启动kettle及linux和windows中kettle往hdfs中写数据(3)
在xmanager中的xshell运行进入图形化界面 sh spoon.sh 新建一个job
- 选择目录,选择文件夹的COM组件问题。在可以调用 OLE 之前,必须将当前线程设置为单线程单元(STA)模式。请确保您的 Main 函数带有 STAThreadAttribute 标记。 只有将调试器附加到该进程才会引发此异常。
异常: 在可以调用 OLE 之前,必须将当前线程设置为单线程单元(STA)模式.请确保您的 Main 函数带有 STAThreadAttribute 标记. 只有将调试器附加到该进程才会引发此异常. ...
- System.Windows.Forms.AxHost.InvalidActiveXStateException”类型的异常在 ESRI.ArcGIS.AxControls.dll 中发生,但未在用户代码中进行处理
private void CopyAndOverwriteMap() { //IObjectCopy接口变量申明 IObjectCopy objectCopy = new ObjectCopyClas ...
随机推荐
- python环境开发
Python3 下载 Python3 最新源码,二进制文档,新闻资讯等可以在 Python 的官网查看到: Python 官网:https://www.python.org/ 你可以在以下链接中下载 ...
- 你一定看得懂的 DDD+CQRS+EDA+ES 核心思想与极简可运行代码示例
前言 随着分布式架构微服务的兴起,DDD(领域驱动设计).CQRS(命令查询职责分离).EDA(事件驱动架构).ES(事件溯源)等概念也一并成为时下的火热概念,我也在早些时候阅读了一些大佬的分析文,学 ...
- XAML 字符转义
在 写xaml的使用遇到了一些特殊字符,这里记录一下特殊字符转义. 这些特殊字符遵循用于编码的万维网联合会(W3C) XML 标准. 下表显示这组特殊字符的编码语法: 字符 语法 描述 < & ...
- spring boot 打包jar后访问classes文件夹的文件提示地址不存在
报错内容:class path resource [client.p12] cannot be resolved to absolute file path because it does not r ...
- centos7.x中安装SQL Server
本文内容是采集的好几位博主的博文进行的一个整合,内容更为精准和详尽,以下是我参照的几篇博文地址: 微软官方文档:https://docs.microsoft.com/zh-cn/sql/linux/s ...
- VMware 安装CentOS8 教程
安装一台Linux服务器 一.准备工作 1.准备一台服务器 1)下载VMware 百度下载自行安装 2.准备CentOS8 系统盘 1)CentOS8官网 https://www.centos.org ...
- css3神奇的圆角边框、阴影框及其图片边框
css3圆角,建议IE10以上 如果border-radius 单位是百分比,则参考为自身宽高,因此当宽高不一致时,圆角为不规则形状 如果border-radius 为50%,则为椭圆:当宽高一致时, ...
- 离线安装PostgreSQL11.6
因为客户最近有一台CentOS7的虚拟机,但是没有联网,需要安装离线安装PostgreSQL 1.首先去官网下载离线安装包 https://www.postgresql.org/download/ 说 ...
- 有关使用phpstudy搭建sqli-lab环境搭建时发生Uncaught Error: Call to undefined function mysql_connect()错误
文章更新于2020-1-30 问题描述 Uncaught Error: Call to undefined function mysql_connect() 分析 经查php手册可知 mysql_co ...
- 安装ubuntu到移动硬盘(UEFI+GPT),实现在别的电脑也可以使用(详细教程)
前置说明:博主小白,第一次安装ubuntu,参考了网上很多人的教程,发博记录一下自己的安装过程.由于有些地方博主理解较浅或者因为机器硬件等各方面原因,本教程适用有限,仅供参考. 1.准备工作 win系 ...