1. 簡(jiǎn)介
遇到大數(shù)據(jù)時(shí),往往讀寫(xiě)文件成了程序運(yùn)行速度的瓶頸,需要更快的讀取方式。相信幾乎所有的C++學(xué)習(xí)者都在cin機(jī)器緩慢的速度上栽過(guò)跟頭,有很多案例中提供幾個(gè)數(shù)據(jù),卻在后臺(tái)測(cè)評(píng)卻提供了近千,近萬(wàn)的數(shù)據(jù)量是常事,而很多人會(huì)發(fā)現(xiàn),明明算法正確的問(wèn)題,卻總是在超時(shí),但把自己的輸入換成scanf,輸出換成printf之后莫名其妙又可以通過(guò)了,于是便冷眼相對(duì)C++的cin與cout。
在C++中,cin與cout往往不需要我們手動(dòng)設(shè)置格式而變得靈活,因此更趨向于我們便捷式的使用,但這并不是說(shuō)cin與cout就一定比scanf和printf慢,我們可以通過(guò)C++輸入輸出流解除綁定的方式進(jìn)行加速,使其提升至C語(yǔ)言scanf和printf般的速度。
2.原理:
cin在為了與scanf保持同步,設(shè)置了一個(gè)緩沖區(qū),為了保證各位混用兩者的情況不會(huì)出錯(cuò),利用這個(gè)緩沖區(qū)進(jìn)行同步,不至于發(fā)生指針錯(cuò)誤造成亂碼,因此cin會(huì)犧牲一點(diǎn)點(diǎn)效率,而這一點(diǎn)點(diǎn)的效率,在大數(shù)據(jù)讀取和運(yùn)算的時(shí)候也會(huì)產(chǎn)生極大的影響,我們可以通過(guò)sync_with_stdio(false)的方式取消這個(gè)緩沖區(qū),讓cin變成和scanf一樣的效率。
a) sync_with_stdio
這個(gè)函數(shù)是一個(gè)“是否兼容stdio”的開(kāi)關(guān),C++為了兼容C,保證程序在使用了std::printf和std::cout的時(shí)候不發(fā)生混亂,將輸出流綁到了一起,默認(rèn)情況為sync_with_stdio(ftrue),即開(kāi)啟。
b)cin.tie(0),cout.tie(0);
cin.tie(NULL);cout.tie(NULL);只解除的是C++運(yùn)行庫(kù)層面的對(duì)數(shù)據(jù)傳輸?shù)慕壎?stdin和stdout應(yīng)該在更底層的操作系統(tǒng)層面有綁定,沒(méi)有解除,也就是說(shuō),cin.tie(0)與cout.tie(0)的方式是繼續(xù)松綁c++傳輸?shù)男省?/p>
因此我們可以在自己的代碼中建立一個(gè)如此的模板:
#include<iostream> using namespace std; int main(){ ios::sync_with_stdio(false); cin.tie(0); cout.tie(0); /* 寫(xiě)上你想寫(xiě)入的代碼,并使用cin,cout輸入輸出 */ return 0; }
也可以用宏定義的方式簡(jiǎn)寫(xiě)這段代碼:
#define jiasu ios::sync_with_stdio(false),cin.tie(0),cout.tie(0);
在主函數(shù)進(jìn)行引用即可。
根據(jù)最近的輸出速度測(cè)試,分別在不同的平臺(tái)進(jìn)行存粹的輸出測(cè)試,使用解除綁定的cout相對(duì)而言是最快的輸出方式,而輸入則因?yàn)槟承┦芟薜脑虿⒉皇亲羁斓?,為了使輸入?shù)據(jù)變快,請(qǐng)看下一章,快速讀取。
備注:
1.NULL在C/C++語(yǔ)言中對(duì)應(yīng)0,因此可以拿0代替NULL(空指針)
2.相關(guān)測(cè)試數(shù)據(jù):
WINDWOS下[1e5的數(shù)據(jù)量]:
使用解除綁定的cout : 30.719000 秒 , 29.518000 秒 , 29.446000 秒
不使用解除綁定cout : 51.749000 秒 , 49.383000 秒 , 47.605000 秒
C++文件的printf : 84.962000 秒 , 76.131000 秒 , 77.639000 秒
C語(yǔ)言文件的printf : 29.776000 秒 , 29.327000 秒 , 29.862000 秒
LINUX系統(tǒng)下[1e5的數(shù)據(jù)量]:
使用解除綁定的cout : 0.199213 秒 , 0.195920 秒 , 0.195387 秒
不使用解除綁定cout : 0.199305 秒 , 0.188013 秒 , 0.199603 秒
C++文件的printf : 0.195575 秒 , 0.197582 秒 , 0.197400 秒
C語(yǔ)言文件的printf : 0.195144 秒 , 0.200245 秒 , 0.215732 秒
【在WINDOWS上面運(yùn)行很久的程序在LINUX全部低于一秒鐘結(jié)束】
LINUX系統(tǒng)下[1e7(一百萬(wàn))的數(shù)據(jù)量]:
使用解除綁定的cout : 18.359119 秒 , 18.066489 秒 , 18.309879 秒
不使用解除綁定cout : 18.655116 秒 , 18.655820 秒 , 18.657697 秒
C++文件的printf : 18.354496 秒 , 18.592422 秒 , 18.312166 秒
C語(yǔ)言文件的printf : 18.463724 秒 , 18.475845 秒 , 18.495757 秒
數(shù)據(jù)來(lái)源: https://blog.dotcpp.com/a/60828
C語(yǔ)言網(wǎng)提供由在職研發(fā)工程師或ACM藍(lán)橋杯競(jìng)賽優(yōu)秀選手錄制的視頻教程,并配有習(xí)題和答疑,點(diǎn)擊了解:
一點(diǎn)編程也不會(huì)寫(xiě)的:零基礎(chǔ)C語(yǔ)言學(xué)練課程
解決困擾你多年的C語(yǔ)言疑難雜癥特性的C語(yǔ)言進(jìn)階課程
從零到寫(xiě)出一個(gè)爬蟲(chóng)的Python編程課程
只會(huì)語(yǔ)法寫(xiě)不出代碼?手把手帶你寫(xiě)100個(gè)編程真題的編程百練課程
信息學(xué)奧賽或C++選手的 必學(xué)C++課程
藍(lán)橋杯ACM、信息學(xué)奧賽的必學(xué)課程:算法競(jìng)賽課入門(mén)課程
手把手講解近五年真題的藍(lán)橋杯輔導(dǎo)課程