C语言二进制文件的读取问题
UDP的通讯程序,其中要把Server和Client的接收和发送的数据以数据长度(要求必须为int)+数据内容(可以为char)以二进制数据流的形式存储到各自的文件中。文...
UDP的通讯程序,其中要把Server和Client的接收和发送的数据以 数据长度(要求必须为int)+数据内容 (可以为char) 以二进制数据流的形式存储到各自的文件中。文件肯定为二进制文件。文件写入数据fp = fopen("客户端.bin","wb")。由于是wb,无法直接在文件末尾加数据,必须要将以前的数据也要保存下来,就要先进行一次read将原有数据保存在缓存区中,在fp = fopen("客户端.bin","wb")之后再写入。不过发现read出来的数据有问题,只读取了数据长度,数据长度后的00和数据内容都没有读取,read 的fp的指针只到数据长度就停止了。具体请看截图。求指导,怎样将数据完整的读出。程序如下,绝对无语法和赋值错误,应该是逻辑的问题,实在想不出来。
while(!feof(fp))
{
ch=fgetc(fp);
msg[count++]=ch;
} //读取原始数据,放入msg数组中
中间的程序都省略
fp = fopen("客户端.bin","wb"); //打开文件,必须存在,二进制存储
if(fp == NULL) {
printf("can not open file\n");
exit(0);
}
fwrite(msg,strlen(msg),1,fp); //写入原来的数据
fwrite(&sendinglen,sizeof(int),1,fp); //然后写入现在的数据长度
fwrite(buffer,sendinglen,1,fp); //最后写入数据内容
我认为一定是读取的问题,但是不知道该怎么弄,0A 00 00 00这种数据长度的结构,在while(!feof(fp)) 判断的时候就认为0A 后的00就是'\0',fscanf也一样,就判断结束了,请问该如何解决。在线等啊,解决了加分 展开
while(!feof(fp))
{
ch=fgetc(fp);
msg[count++]=ch;
} //读取原始数据,放入msg数组中
中间的程序都省略
fp = fopen("客户端.bin","wb"); //打开文件,必须存在,二进制存储
if(fp == NULL) {
printf("can not open file\n");
exit(0);
}
fwrite(msg,strlen(msg),1,fp); //写入原来的数据
fwrite(&sendinglen,sizeof(int),1,fp); //然后写入现在的数据长度
fwrite(buffer,sendinglen,1,fp); //最后写入数据内容
我认为一定是读取的问题,但是不知道该怎么弄,0A 00 00 00这种数据长度的结构,在while(!feof(fp)) 判断的时候就认为0A 后的00就是'\0',fscanf也一样,就判断结束了,请问该如何解决。在线等啊,解决了加分 展开
4个回答
展开全部
C语言中二进制文件的读取要用fread和fwrite来实现。
fwrite()与fprintf()是不同的。
fwrite将写入的数据作为文件的磁盘内容保存。fprintf将写入的数据的每个字符所对应的ASCII码作为文件的磁盘内容保存。fprintf做了一个转换的工作。
当打开文件时,记事本会自动把文件的磁盘内容作为ASCII码转换成对应的字符,然后再显示出来,即显示的是文本内容而不是磁盘内容。
例如,用fwrite向文件写入“65”时,文件的磁盘内容就是保存的65(磁盘上以二进制表示)。当用记事本打开文件时,记事本会读到65,并把65看作一个ASCII码,再把对应的字符“A”显示出来。因此屏幕上看到的文本内容是“A”。
而用fprintf向文件写入“65”时,文件的磁盘内容保存的是“6”和“5”这两个字符对应的ASCII码,分别是54和53。因此文件的磁盘内容是54和53。当用记事本打开文件时,记事本读到54,就显示出对应的“6”。再读到53,就显示出对应的“5”。
fwrite()与fprintf()是不同的。
fwrite将写入的数据作为文件的磁盘内容保存。fprintf将写入的数据的每个字符所对应的ASCII码作为文件的磁盘内容保存。fprintf做了一个转换的工作。
当打开文件时,记事本会自动把文件的磁盘内容作为ASCII码转换成对应的字符,然后再显示出来,即显示的是文本内容而不是磁盘内容。
例如,用fwrite向文件写入“65”时,文件的磁盘内容就是保存的65(磁盘上以二进制表示)。当用记事本打开文件时,记事本会读到65,并把65看作一个ASCII码,再把对应的字符“A”显示出来。因此屏幕上看到的文本内容是“A”。
而用fprintf向文件写入“65”时,文件的磁盘内容保存的是“6”和“5”这两个字符对应的ASCII码,分别是54和53。因此文件的磁盘内容是54和53。当用记事本打开文件时,记事本读到54,就显示出对应的“6”。再读到53,就显示出对应的“5”。
展开全部
使用fread读取,并且fopen的时候一定要使用“rb”模式,不能使用文本模式。
fgetc只适合处理文本文件
fgetc只适合处理文本文件
更多追问追答
追问
我用的就是rb打开,读取的,而且查看的是fgetc能读二进制的。有没有什么读取方式是二进制写入追加的?我查了好像没查到。
追答
不建议你使用fgetc,fgetc可以读二进制文件是没错,但二进制文件中有很多不是有效字符,处理起来就会出问题,比如0,10等控制字符
ab+ 为读/写打开一个二进制文件,位置指针移到到文件末位,可以添加也可以读;
rb+ 为读/写打开一个二进制文件,但此文件一定要存在
wb+ 为读/写新建一个二进制文件,会删除原有文件数据
上述方式是ANSIC规定的,不是所有的编译系统都支持的,你可以试一下
本回答被提问者采纳
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
展开全部
由于数据结构是
数据长度(要求必须为int)+数据内容 (可以为char) 以二进制数据流的形式存储到各自的文件中
那么可以考虑以fread读取长度 然后再通过fread读出数据段
例如
while( fread( &data_len[i], sizeof(int) , 1 , fp )
{
fread( &data[i], data_len[i], 1, fp)
i++;
}
数据长度(要求必须为int)+数据内容 (可以为char) 以二进制数据流的形式存储到各自的文件中
那么可以考虑以fread读取长度 然后再通过fread读出数据段
例如
while( fread( &data_len[i], sizeof(int) , 1 , fp )
{
fread( &data[i], data_len[i], 1, fp)
i++;
}
追问
你牛B了,把我后面想干什么都猜出来了,呵呵,用ab+打开文件就可以了,谢谢啦
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
展开全部
这个很正常,这是操作系统的缓存(cache)在起作用。
缓存:为了解决CPU速度和内存速度的速度差异而产生(CPU运算速度比内存快的多)
当程序需要读取某个文件时,实际就是把数据读入内存由CPU运算,CPU首先去缓存中找寻,查找不到就会到内存中去读取同时复制到缓存中以便下次访问,这个时候速度自然很慢,当你第二次读取该文件时,缓存中已经存在,CPU再次访问这些数据就会变的非常快。
这是和系统读取数据的方式有关的,并不是因为某个函数效率低的原因,比较明显的:
你在某个磁盘下搜索某一个文件名,第一次会比较慢,第二次就会快很多,原因就是第二次CPU需要处理的数据已经存在缓存中,处理时效率会非常高。
缓存:为了解决CPU速度和内存速度的速度差异而产生(CPU运算速度比内存快的多)
当程序需要读取某个文件时,实际就是把数据读入内存由CPU运算,CPU首先去缓存中找寻,查找不到就会到内存中去读取同时复制到缓存中以便下次访问,这个时候速度自然很慢,当你第二次读取该文件时,缓存中已经存在,CPU再次访问这些数据就会变的非常快。
这是和系统读取数据的方式有关的,并不是因为某个函数效率低的原因,比较明显的:
你在某个磁盘下搜索某一个文件名,第一次会比较慢,第二次就会快很多,原因就是第二次CPU需要处理的数据已经存在缓存中,处理时效率会非常高。
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
推荐律师服务:
若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询