C++二进制文件读写

简介:

 今天在做项目时联想到了这两个问题,所以实际编程测试了一下,有一些新的收获:

<1> 我一直以为自己很熟悉如何使用C/C++中的二进制文件,可今天测试的时候突然发现程序生成的二进制文件和文本文件一样。
比如:
FILE* fp = fopen("binary","wb");
//FILE* fp = fopen("character.txt","w");
fprintf(fp,"count is %d",250);
上述代码一个使用的是text file mode,一个是binary file mode,但结果生成的二进制文件中保存的仍然是ASCII 码,直接用
记事本就可以打开查看。要说区别,主要是换行符的区别,binary file 的换行符为<LF>,而text file 的换行符为<CR><LF>,仅此而已,
我就想怎么回事啊,我明明是用二进制模式打开的文件呀,怎么里面直接保存的还是ASCII 码?也就是说保存125 这个数字还是会占用
3 个字节,而不是我想象中那样只占一个字节!
后来我就换用C++,结果还是一样,代码如下:
ofstream fs("binary",ios::binary);
//ofstream fs("character.txt");
int i = 32765;
fs<<i<<endl;
//fs.write((char*)&i,2);
fs.close();
无论以二进制文件模式打开还是以文本模式打开,文件中都是保存着文本!似乎C/C++中的binary 模式不起作用!?!
后来查阅资料才知道:要想在C/C++中将数据以二进制形式文件输出,与你打开文件时的模式没有关系,关键是取决于你调用哪个
函数往文件里写东西!!只有使用fwrite 和fs.write()函数才能以二进制形式输出到文件中,调用puts、fprintf、<<等函数输出的
都是ASCII 文本,尤其需要注意的是类似于上面代码段中那样,在C++中,即使你用fs<<i<<endl 语句来输出一个整数,输出到二进制
文件中的仍然是文本格式!<< operator 在输出之前会自动给你进行转换,把一个整数值转换成一位一位的数字字符!!而且我后来试
过了,即使我以文本模式打开一个文件,假如我用fwrite 函数输出的话,文件中仍然是二进制格式,呵呵,说明在输出数据到文件时,
的确与打开文件的模式没有关系,只与调用的输出函数有关!!
此外,要注意,你不能用>>来读取以二进制文件格式存储的整数!(注:中午我试过了,不行!这再次证明>>只能读入文本格式的
数字)
(补注:11:47am 我后来又想,所有上面的这些东西可以归结成一句话:你以什么模式打开文件根本不重要,因为你既改变不
了文件本身的内容,也改变不了C/C++中系统函数的工作方式,所以在编程的时候,你只要关心这个文件里的数据内容本身是二进制格
式还是文本格式就好了!如果内容是文本格式的,你就调用文本格式那一套函数,比如puts,gets,fscanf,fprintf,<<,>>等,如果内容
是二进制格式的,你就调用二进制格式那一套函数,比如fread,fwrite,ifstream.read(),ofstream.write()等。 只要保持文件内容与
处理函数相对应相一致就可以了,别管它用什么模式打开文件!! 假如你用<<向一个二进制文件中输入一个整数,那么其实里面保存
的是文本格式的数据,那么你就照样可以以二进制模式打开它,然后用>>来读取这个整数。相反,如果你的二进制文件里面是一个以二
进制形式保存的整数,那你肯定不能用>>来读取里面的整数了!!
<2>关于字节序的问题,我想用一张图来表示就足够了:
今天终于弄明白怎样使用C++读写二进制文件了。
要读取文件必须包含<fstream>头文件,这里包含了C++读写文件的方法。
可以使用fstream 类,这个类可以对文件进行读写操作。
1、打开文件。
打开文件可以有两种方式,第一种可以使用fstream 类的构造函数。
fstream file("test.dat",ios_base::in|ios_base::out|ios_base::app);
另外一种方法就是使用open 函数。
fstream file;
file.open("test.dat",ios_base::in|ios_base::out|ios_base::app);
这样就可以打开一个可读写的文件了。如果文件不存在的话,就会创建一个新文件并且以读写方式打开。
这里需要说明一点,如果文件不存在的话,open 函数中第二个参数必须包含ios_base::out|ios_base::app,
否则就不能正确创建文件。
2、写文件。
先进性写文件的操作否则读一个空文件是没有意义的。
既然是写二进制文件可以向文件中写入一个整形值。写二进制字符只能使用write 函数。
但是write 函数的原形是write(const char * ch, int size)。第一个参数是char *类型,所以需要把将要写入
文件的int 类型转换成char *类型。这里的转换困扰了我好几天,不过终于弄明白了。代码如下。
int temp;
file.write((char *)(&temp),sizeof(temp));
3、读文件。
可以写文件了,读文件就好办多了。读文件需要用到read 函数。其参数和write 大致相同,read(const char * ch, int size)。
要把内容读到int 类型变量中同样涉及到一个类型转换的问题。和写文件一样。
int readInt;
file.read((char *)(&readInt),sizeof(readInt));
这样文件中的int 值就读入到int 型变量readInt 中了。
4、文件指针。
在文件的读写过程中往往需要对文件进行选择性读取。所以需要进行文件指针的移动。这是需要用到seekg 和seekp 函数。
在fstream 类中有两个文件指针,一个是读取文件的指针,一个是写文件的指针分别用tellg 和tellp 文件来取得指针的位置。
同样seekg 和seekp 两个函数分别是对这两个指针进行移动的函数。这两个函数的参数都是一样的。
先对几个枚举类型进行一下说明:
ios_base::beg ——文件开始位置
ios_base::cur ——文件当前位置
ios_base::end ——文件末尾位置
下面以seekg 为例说明一下指针移动的方法:
file.seekg(3) ——指针移动到第三个字符的位置
file.seekg(ios_base::beg) ——指针移动到文件开头
file.seekg(ios_base::end) ——指针移动到文件末尾
file.seekg(-3,ios_base::cur) ——指针当前位置向前移动三个字符
file.seekg(3,ios_base::cur) ——指针当前位置向后移动三个字符
file.seekg(3,file.tellg()) ——指针当前位置向后移动三个字符
file.seekg(file.tellg()+3) ——指针当前位置向后移动三个字符
5、对文件操作完毕后别忘了关闭文件。
file.close();
以上5 个步骤就完成了对文件的读写操作。文本文件的操作是相同的,比二进制文件还要简单。











本文转hackfreer51CTO博客,原文链接:http://blog.51cto.com/pnig0s1992/563152 ,如需转载请自行联系原作者
相关文章
|
3月前
|
Java 编译器 C++
颠倒二进制位(C++)
颠倒二进制位(C++)
34 1
|
1月前
|
算法 网络协议 编译器
【C++ 14 新特性】C++14二进制字面量:深度探索与实践
【C++ 14 新特性】C++14二进制字面量:深度探索与实践
39 1
|
3月前
|
C++
二进制求和(C++)
二进制求和(C++)
38 1
|
18天前
|
存储 C++ iOS开发
C++文件操作(文本文件的读写+二进制文件的读写)
C++文件操作(文本文件的读写+二进制文件的读写)
|
1月前
|
Linux API C++
【Linux C/C++ 线程同步 】Linux API 读写锁的编程使用
【Linux C/C++ 线程同步 】Linux API 读写锁的编程使用
21 1
|
5月前
|
C++
《C++避坑神器·七》二进制读写自定义类型导致崩溃或数据读写不全问题
《C++避坑神器·七》二进制读写自定义类型导致崩溃或数据读写不全问题
50 0
|
3月前
|
Java Go Python
Golang每日一练(leetDay0103) 区域和检索1~3
Golang每日一练(leetDay0103) 区域和检索1~3
30 0
Golang每日一练(leetDay0103) 区域和检索1~3
|
3月前
|
算法 Java C++
Java每日一练(20230424) 二叉树中序遍历、交换链表节点、不同子序列
Java每日一练(20230424) 二叉树中序遍历、交换链表节点、不同子序列
36 0
Java每日一练(20230424) 二叉树中序遍历、交换链表节点、不同子序列
|
3月前
|
存储 移动开发 Linux
C++017-C++文件读写应用
C++017-C++文件读写应用
|
4月前
|
C++
C++文件的随机读写与特定格式输入输出
C++文件的随机读写与特定格式输入输出
27 0