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 ...
随机推荐
- 浅谈Linux与unix系统的来历
在今天的UNIX是商业化的,UNIX系统大多是与硬件配套的,也就是说,大多数UNIX系统如AIX.HP-UX等是无法安装在 x86 服务器和个人计算机上的,UNIX系统是一个分时系统,而UNIX是至关 ...
- log4j2 springboot 特点与使用方法
Apache Log4j2 is an upgrade to Log4j that provides significant improvements over its predecessor, Lo ...
- Emmet:HTML/CSS代码快速编写
html缩写: 1. 初始化 HTML文档需要包含一些固定的标签,比如<html>.<head>.<body>等,现在你只需要1秒钟就可以输入这些标签.比如输入“ ...
- HTTP 1.1状态代码及其含义
HTTP 1.1状态代码及其含义 100 Continue 初始的请求已经接受,客户应当继续发送请求的其余部分.(HTTP 1.1新) 101 Switching Protocols 服务器将 ...
- Docker深入浅出系列 | 容器数据持久化
Docker深入浅出系列 | 容器数据持久化 Docker已经上市很多年,不是什么新鲜事物了,很多企业或者开发同学以前也不多不少有所接触,但是有实操经验的人不多,本系列教程主要偏重实战,尽量讲干货,会 ...
- 你一定看得懂的 DDD+CQRS+EDA+ES 核心思想与极简可运行代码示例
前言 随着分布式架构微服务的兴起,DDD(领域驱动设计).CQRS(命令查询职责分离).EDA(事件驱动架构).ES(事件溯源)等概念也一并成为时下的火热概念,我也在早些时候阅读了一些大佬的分析文,学 ...
- centos7安装postgresql和postgis
1.安装步骤 -- 安装对应的rpm文件(其他系统的rpm包,请自行到https://yum.postgresql.org/下载)yum install -y https://download.pos ...
- clr via c# 程序集加载和反射(2)
查看,clr via c# 程序集加载和反射(1) 8,发现类型的成员: 字段,构造器,方法,属性,事件,嵌套类型都可以作为类型成员.其包含在抽象类MemberInfo中,封装了所有类型都有的一组属性 ...
- C#设计模式学习笔记:(15)迭代器模式
本笔记摘抄自:https://www.cnblogs.com/PatrickLiu/p/7903617.html,记录一下学习过程以备后续查用. 一.引言 今天我们要讲行为型设计模式的第三个模式--迭 ...
- 目前最全的Python的就业方向
Python是一门面向对象的编程语言,编译速度超快,从诞生到现在已经25个年头了.它具有丰富和强大的库,常被称为“胶水语言”,能够把用其他语言编写的各种模块(尤其是C/C++)很轻松地联结在一起.其特 ...