1 #include <iostream>
2 #include <fstream>
3 #include <cassert>
4
5 using namespace std;
6 int main()
7 {
8 ifstream in("test.txt");
9 assert(in.is_open());
10
11 //基地址為文件結束處,偏移地址為0,於是指針定位在文件結束處
12 in.seekg(0, ios::end);
13 //sp為定位指針,因為它在文件結束處,所以也就是文件的大小
14 streampos sp = in.tellg();
15 cout<<"file size:"<<endl<<sp<<endl;
16
17 //基地址為文件末,偏移地址為負,於是向前移動sp/3個字節
18 in.seekg(-sp/3, ios::end);
19 streampos sp2 = in.tellg();
20 cout<<"from file to point:"<<endl<<sp2<<endl;
21
22 //基地址為文件頭,偏移量為0,於是定位在文件頭; 從頭讀出文件內容
23 in.seekg(0, ios::beg);
24 cout<<in.rdbuf()<<endl;
25
26 //從sp2開始讀出文件內容
27 in.seekg(sp2);
28 cout<<in.rdbuf()<<endl;
29
30 in.close();
31
32 return 0;
33 }
1 #include <iostream>
2 #include <fstream>
3 #include <cassert>
4 #include <streambuf>
5
6 int main(void)
7 {
8 std::ofstream log("test.txt");
9 assert(log.is_open());
10
11 // 返回cout原來的的流緩沖指針,使cout重定向到log文件的流緩沖
12 std::streambuf* x = std::cout.rdbuf(log.rdbuf());
13
14 // 寫入到文件中
15 std::cout << "xiaolouyiyetingchunyu\n";
16
17 log.close();
18
19 std::cout.rdbuf(x); // 恢復cout的流對象指針
20
21 std::cout << "yuanyuewandao\n"; // 寫入cout
22
23 return 0;
24 }
我們使用STL編程的時候有時候會想到把一個流對象指向的內容用另一個流對象來輸出,比如想把一個文件的內容輸出到顯示器上,我們可以用簡單的兩行代碼就可以完成。
ifstream infile("test.txt");
cout << infile.rdbuf();
上面的代碼就把infile流對象中的流重定向到標准輸出cout上,您可以在屏幕上看到test.txt的內容。
下面的例子來自MSDN,清晰的描述了rdbuf函數的使用方法
// basic_ios_rdbuf.cpp // compile with: /EHsc #include <ios> #include <iostream> #include <fstream> int main( ) { using namespace std; ofstream file( "rdbuf.txt" ); streambuf *x = cout.rdbuf( file.rdbuf( ) ); cout << "test" << endl; // Goes to file cout.rdbuf(x); cout << "test2" << endl; }
rdbuf函數有兩種調用方法
basic_streambuf<Elem, Traits> *rdbuf( ) const;
basic_streambuf<Elem, Traits> *rdbuf( basic_streambuf<E, T> *_Sb);
1)無參數。返回調用者的流緩沖指針。
2)參數為流緩沖指針。它使調用者與參數(流緩沖指針)關聯,返回自己當前關聯的流緩沖區指針。
假如我們用C語言寫一個文件復制程序,比如一個mp3文件,我們首先考慮的是C語言的文件輸入輸出功能,其思路是建一個指定大小緩沖區,我們從源文件中循環讀取緩沖區大小的數據,然后寫進目的文件。而在C++中,我們拋棄了這種用字符緩沖區的按字節復制的方法,因為這種方法看起來很繁瑣,而且效率一點也不高。下面可以對比這兩種方法(程序可以直接執行):
C:
#include<stdlib.h>
#include<stdio.h>
int main()
{
char buf[256];
FILE *pf1, *pf2;
if((pf1 = fopen("1.mp3", "rb")) == NULL)
{
printf("源文件打開失敗/n");
return 0;
}
if((pf2 = fopen("2.mp3","wb")) == NULL)
{
printf("目標文件打開失敗/n");
return 0;
}
while(fread(buf,1,256,pf1), !feof(pf1))
{
fwrite(buf,1,256,pf2);
}
fclose(pf1);
fclose(pf2);
return 0;
}
在C++中:
#include<fstream>
#include<iostream>
using namespace std;
int main()
{
fstream fin("1.mp3",ios::in|ios::binary);
if(!fin.is_open())
{
cout << "源文件打開失敗" << endl;
return 0;
}
fstream fout("2.mp3",ios::out|ios::binary);
if(! fin.is_open())
{
cout << "目標文件打開失敗!" << endl;
return 0;
}
fout<<fin.rdbuf();
fin.close();
fout.close();
return 0;
}
看起來是不是清晰多了呢,這就是C++中的流緩沖的威力了,程序通過把源文件的流重定向到關聯到目的文件的流對象,通過 fout<<fin.rdbuf();一句代碼就完成了在C語言中的循環讀寫緩沖區的功能,而且C++中使用的是底層的流緩沖,效率更高!