8.7 C++二进制文件读写操作
参考:http://www.weixueyuan.net/view/6413.html
总结:
二进制文件的读写稍微麻烦一些,对二进制文件的读写同样需要打开文件和关闭文件,打开和关闭方式与文本文件相同,只不过需要在打开方式上加上ios::binary以指明以二进制方式进行读写。
对于文本文件而言,我们只能用ofstream类定义对象用于输出到文件,用ifstream类定义对象用于从文件中输入,而对于二进制文件而言,除了可以这么做以外,我们还可以用fstream类定义对象既能用于从文件输入,又能输出到文件中。
针对二进制文件的读写,输入输出类中定义了专门的函数read和write,这两个都是类的成员函数。
istream & read ( char * buffer, int size );
ostream & write ( const char * buffer, int size );
char指针buffer是指向内存中的一段存储空间,size是存储空间的大小,也即需要读写的内容的字节数。
有时我们只需要获取某个指定位置的数据而已,此时如果需要将数据全部读入再找到对应位置数据实在是效率太低。为此系统为我们提供了一些操作文件读写指针位置的成员函数,我们可以使用这些函数,将文件读写指针移动到指定位置并操作其中的数据。
函数 | 用途 |
---|---|
gcount ( ) | 返回最后一次输入所读入的字节数 |
tellg ( ) | 返回输入指针的当前位置 |
seekg ( 文件中的位置 ) | 将输入文件中指针移动到指定的位置 |
seekg ( 位移量, 参照位置 ) | 以参照位置为标准移动指针 |
tellp ( ) | 返回输出文件指针当前位置 |
seekp ( 文件中的位置 ) | 将输出文件中指针移动到指定的位置 |
seekp ( 位移量, 参照位置 ) | 以参照位置为为标准移动指针 |
---------------------------------------------------------
二进制文件的读写稍微麻烦一些,对二进制文件的读写同样需要打开文件和关闭文件,打开和关闭方式与文本文件相同,只不过需要在打开方式上加上ios::binary以指明以二进制方式进行读写。
对于文本文件而言,我们只能用ofstream类定义对象用于输出到文件,用ifstream类定义对象用于从文件中输入,而对于二进制文件而言,除了可以这么做以外,我们还可以用fstream类定义对象既能用于从文件输入,又能输出到文件中。
针对二进制文件的读写,输入输出类中定义了专门的函数read和write,这两个都是类的成员函数,它们的函数原型为:
istream & read ( char * buffer, int size ); ostream & write ( const char * buffer, int size );
char指针buffer是指向内存中的一段存储空间,size是存储空间的大小,也即需要读写的内容的字节数。有了这两个用于读写二进制文件的函数,我们通过示例来看一下如何使用这两个函数。
例1:
#include <iostream> #include <fstream> using namespace std; int main() { int A[ 3 ][ 10 ] = { { 1, 2, 3, 4, 5, 6, 7, 8, 9, 0 }, { 1, 2, 3, 4, 7, 7, 7, 7, 7, 7 }, { 9, 8, 7, 6, 5, 4, 3, 2, 1, 0 } }; int B[ 3 ][ 10 ]; int i, j; for( i = 0; i < 3; i++ ) { for( j = 0 ; j < 10; j++ ) { cout << A[ i ][ j ] <<" "; } cout << endl; } cout << endl; //******************************************* ofstream output( "test.txt", ios::out | ios::binary ); if( ! output ) { cerr << "Open output file error!" << endl; exit( -1 ); } output.write ((char *) A, sizeof( A ) ); for( i = 0; i < 3; i++ ) { for( j = 0 ; j < 10; j++ ) { //output.write ( ( char * ) & A [ i ][ j ], sizeof( A [ i ][ j ] ) ); } } output.close(); //******************************************* ifstream input( "test.txt", ios::in | ios::binary ); if( ! input ) { cerr << "Open input file error!" << endl; exit( -1 ); } input.read( ( char * ) & B , sizeof( B ) ); for( i = 0; i < 3; i++ ) { for( j = 0 ; j < 10; j++ ) { //input.read ( ( char * ) & B[ i ][ j ], sizeof( B[ i ][ j ] )); } } for( i = 0; i < 3; i++ ) { for( j = 0 ; j < 10; j++ ) { cout << B[ i ][ j ] <<" "; } cout << endl; } //******************************************* int temp; input.seekg( 20 * sizeof( int ), ios::beg ); input.read( ( char * ) & temp, sizeof( int )); cout << temp << endl; input.close(); return 0; }
这个例子的主要功能就是先将一个二维数组以二进制的方式写入到一个test.txt文件中,之后再从这个文件中读出其中的数据赋值给另外一个数组。我们来细看一下这个函数的主函数,我们一开始定义了一个数组A[ 3 ][ 10 ],并赋了初值,之后又定义了另一个数组B[ 3 ][ 10 ],用于存放从文件中读入的数据。程序先将A[ 3 ][ 10 ]数组中的数据打印出来。之后定义一个ofstream类的对象,以二进制方式打开文件,这种文件打开方式和文本文件打开方式是一样的,检测文件是否打开方法也是一样的。文件正常打开之后,我们就开始将A[ 3 ][ 10 ]数组中的数据以二进制的形式输出到文件中,此时我们通过ofstream类的对象调用write函数,因为write函数的第一个参数为char指针,因此需要进行一次强制类型转换,第二个参数给出了数组A[ 3 ][ 10 ]在内存中所占空间大小。因为数组在内存中是连续存储的,因此我们可以只用这么一个语句,就将所有的数组内容以二进制的方式写入到文本中,当然如果我们想一个一个数据的输出到文件也不是不可以的,注释掉的那个语句就是一个一个元素输出的。之后输出数据操作结束,关闭文件。
接下来程序开始从文件中读入数据,打开文件也是和打开文本文件是一样的,只是在打开方式处添加了ios::binary以指明以二进制方式打开文件。同样的,我们也可以利用一个语句就将文件中的所有数据都读入到B[ 3 ][ 10 ]数组中。之后我们将B[ 3 ][ 10 ]数组中的数据打印出来。
当然,在程序设计过程中,我们有时候并不会像本例这样的,每次都将一堆数据全部存入文件或者将一堆数据全部读入到内存,很可能我们只需要获取某个指定位置的数据而已,此时如果需要将数据全部读入在找到对应位置数据实在是效率太低。为此系统为我们提供了一些操作文件读写指针位置的成员函数,我们可以使用这些函数,将文件读写指针移动到指定位置并操作其中的数据,具体函数见下表。
函数 | 用途 |
---|---|
gcount ( ) | 返回最后一次输入所读入的字节数 |
tellg ( ) | 返回输入指针的当前位置 |
seekg ( 文件中的位置 ) | 将输入文件中指针移动到指定的位置 |
seekg ( 位移量, 参照位置 ) | 以参照位置为标准移动指针 |
tellp ( ) | 返回输出文件指针当前位置 |
seekp ( 文件中的位置 ) | 将输出文件中指针移动到指定的位置 |
seekp ( 位移量, 参照位置 ) | 以参照位置为为标准移动指针 |
表中所列函数以“g”(get的首字母)结尾的函数都是用于输入的函数,而以“p”(put的首字母)结尾的函数都是用于输出的函数。函数中有提到参照位置,在系统中定义了三种参照位置:
- ios::beg 所打开文件的开头,这是默认值
- ios::cur 文件读写指针当前的位置
- ios::end 文件结尾
在本节例1中程序主函数的最后我们使用seekg将文件读写指针以文件开头为标准移动20个int型数据,此时指针指向的是第三行第一列的数据,然后我们用read函数将数据读入到temp中,输出temp结果确实是第三行第一列中的数据9。