0x01 前置准备

所有代码依赖以下头文件,建议统一包含:

  • <cstdio>:提供 getchar()putchar()fread()fwrite()
  • <iostream>:提供 cincout
  • <cctype>:提供 isspace()

0x02 基础 I/O 优化:基于 cincout

优化步骤

  1. 关闭流同步
  • 实现:通过 ios::sync_with_stdio(false) 关闭 C++ 和 C 输入输出流的同步;
  • 解释:为了确保混用 C++ 的 cin / cout 和 C 的 printf / scanf 不会产生 I/O 混乱,C++ 和 C 的两种流之间进行了同步。这提高了兼容性,但是产生了大常数。关闭流同步之后就不要同时使用 cinscanf,也不要同时使用 coutprintf,否则会造成 I/O 混乱。但可以同时使用 cinprintf,也可以同时使用 scanfcout
  1. 解除绑定
  • 实现:通过 cin.tie(nullptr) 解除 cincout 的绑定;
  • 解释:在 C++ 中,cin 默认绑定的是 &cout,这意味着每次读入都会调用 flush()。可以用 cin.tie(nullptr) 函数解除这种绑定。
  1. 针对 endl 的优化
  • 实现:用 '\n' 替换 endl
  • 解释:endl 的作用是换行并刷新缓冲区,相当于 cout<<'\n'<<flush。而刷新缓冲区会带来一定开销。

其中前两步一般合称「关流」。后文会沿用这个称呼。

代码实现

以下两种写法是等价的:

// 写法1:链式调用
cin.tie(0)->sync_with_stdio(0);
// 写法2:分步调用
ios::sync_with_stdio(0);
cin.tie(0);

可以这样将所有 endl 替换为 '\n'

// 注意:该宏需在包含<iostream>之后定义,避免与std::endl声明冲突
#define endl '\n'

0x03 进阶优化:快读

普通快读:基于 getchar()

通过 getchar() 函数逐字符读取,手动解析整数或字符串。

void read(int &x){  // 读整数(支持负数)
int c,f=1;
while((c=getchar())<'0'||c>'9') if(c=='-') f=-1;
for(x=c^48;(c=getchar())>='0'&&c<='9';x=(x<<3)+(x<<1)+(c^48));
x*=f;
}
void read(char &c){ // 读一个非空字符
while(isspace(c=getchar()));
}
int read(char s[]){ // 读一个字符串,到空格/EOF为止,返回长度
int len=0;
char c;
while(isspace(c=getchar()));
do s[len++]=c;
while(!isspace(c=getchar())&&c!=EOF);
s[len]='\0'; // 补字符串结束符
return len;
}
int getline(char s[]){ // 读一行字符串,返回长度
int len=0;
char c;
while((c=getchar())!='\n'&&c!=EOF) s[len++]=c;
s[len]='\0'; // 补字符串结束符
return len;
}

缓冲区快读:基于 fread()

getchar() 每次从系统读取 1 个字符,频繁调用系统接口,开销大。fread() 一次性读取一整块数据到自定义缓冲区,后续从缓冲区取字符。这样可以减少系统调用次数,速度通常可以提升 5~10 倍。

缓冲区一般大小设为 1MB 左右,即 \(2^{20}\) 字节,这样既不会占太大空间,不会刷新太多次缓冲区。

由于 fread 可以一次整块读入,因此速度比 getchar 快多了。

char in[1<<20],*p1,*p2;
inline char gc(){ // 从缓冲区读1个字符,空则补充
return p1==p2&&(p2=(p1=in)+fread(in,1,1<<20,stdin))==in?EOF:*p1++;
}

加上这段代码,然后用 gc() 替换掉所有 getchar() 就可以了。

0x04 进阶优化:快写

普通快写:基于 putchar()

通过 putchar() 函数逐字符输出。

void write(int x){  // 写整数(支持负数)
if(x<0) putchar('-'),x=-x;
x<10?putchar(x|48):(write(x/10),putchar(x%10|48));
}
void write(char s[],int len){ // 写字符串,指定长度
for(int i=0;i<len;++i) putchar(s[i]);
}
void write(char s[]){ // 写字符串,直到'\0'为止
for(int i=0;s[i];++i) putchar(s[i]);
}

缓冲区快写:基于 fwrite()

和缓冲区快读差不多,自定义一个缓冲区,每次写一个字符到缓冲区,满了就刷新缓冲区,通过 fwrite() 一次性将整个缓冲区里的内容输出。

加上如下代码,再用 pc() 替换掉所有 putchar() 就可以了。

char out[1<<20],*p3=out;
inline void pc(char c){ // 向缓冲区写1个字符,满则刷新
if(p3-out==1<<20) fwrite(out,1,1<<20,stdout),p3=out;
*p3++=c;
}

但是程序结束时,缓冲区里可能还有东西,因此我们必须在结束前清空缓冲区。这一步千万不要忘!

fwrite(out,1,p3-out,stdout);

0x05 工程化实现:I/O 类封装

封装的核心目的

  • 自动管理缓冲区:析构函数自动调用 fwrite() 刷新输出缓冲区,避免忘记刷新。
  • 统一接口:将快读和快写整合到同一个类中,使用时直接调用 io.函数名(参数) 即可,无需再关注底层实现,适合作为缺省源使用。

代码实现

class IO{
#define SIZE 1<<20
private:
char in[SIZE],out[SIZE],*p1,*p2,*p3;
public:
IO():p1(in),p2(in),p3(out){}
~IO(){fwrite(out,1,p3-out,stdout);}
inline char gc(){ // 从缓冲区读1个字符,空则补充
return p1==p2&&(p2=(p1=in)+fread(in,1,SIZE,stdin))==in?EOF:*p1++;
}
inline void pc(char c){ // 向缓冲区写1个字符,满则刷新
if(p3-out==SIZE) fwrite(out,1,SIZE,stdout),p3=out;
*p3++=c;
}
void read(int &x){ // 读整数(支持负数)
int c,f=1;
while((c=gc())<'0'||c>'9') if(c=='-') f=-1;
for(x=c^48;(c=gc())>='0'&&c<='9';x=(x<<3)+(x<<1)+(c^48));
x*=f;
}
void read(char &c){ // 读一个非空字符
while(isspace(c=gc()));
}
int read(char s[]){ // 读一个字符串,到空格/EOF为止,返回长度
int len=0;
char c;
while(isspace(c=gc()));
do s[len++]=c;
while(!isspace(c=gc())&&c!=EOF);
s[len]='\0'; // 补字符串结束符
return len;
}
int getline(char s[]){ // 读一行字符串,返回长度
int len=0;
char c;
while((c=gc())!='\n'&&c!=EOF) s[len++]=c;
s[len]='\0'; // 补字符串结束符
return len;
}
void write(int x){ // 写整数(支持负数)
if(x<0) pc('-'),x=-x;
x<10?pc(x|48):(write(x/10),pc(x%10|48));
}
void write(char s[],int len){ // 写字符串,指定长度
for(int i=0;i<len;++i) pc(s[i]);
}
void write(char s[]){ // 写字符串,直到'\0'为止
for(int i=0;s[i];++i) pc(s[i]);
}
#undef SIZE
}io; // 全局实例化,无需重复创建对象

0x06 关键避坑指南

  1. 混用不同 I/O 方式

    • 关流后不能混用 cinscanf,也不能混用 coutprintf
    • 自定义缓冲区快读与 getchar() 不可混用,缓冲区快写和 putchar() 也不能混用;
  2. 缓冲区快写忘记刷新:非封装版本需在程序结束前调用 fwrite(out,1,p3-out,stdout),否则缓冲区剩余数据不会输出;
  3. 整数快读快写处理边界值:处理 INT_MIN 即 \(-2147483648\) 时会溢出,若需支持要开 long long

0x07 性能对比与选型建议

I/O 方式 速度排序 优点 缺点 适用场景
缓冲区快读快写 1 速度极致,适合大数据 代码长,需封装,不支持复杂类型(如浮点数) 输入超大,高频 I/O 的题目
普通快读快写 2 代码短,速度快 不支持复杂类型(如浮点数) 输入较大,需要卡常的题目
关流 cin / cout 3 代码超短 兼容性差 一般题目
scanf / printf 4 适合输出格式串 格式控制符复杂,速度较慢 一般题目
不关流 cin / cout 5 兼容性好 速度极慢 极小数据量的题目,调试输出

注:若需支持浮点数(如 double),需扩展快读快写函数,手动解析小数点前后数字。因浮点数 I/O 场景较少,通常含有浮点数的题目数据量也不会太大,本文暂不展开,可根据需求自行扩展。

C++ I/O 终极加速指南,全网最全整理的更多相关文章

  1. WordPress SEO ☞ WordPress网站终极优化指南

    原文地址:http://www.eastdesign.net/wordpress-seo/ 最新消息,东方设计学院 WordPress SEO 系列视频教程正在持续更新中,目前为了不至于让视频传播过于 ...

  2. Python实现代码统计工具——终极加速篇

    Python实现代码统计工具--终极加速篇 声明 本文对于先前系列文章中实现的C/Python代码统计工具(CPLineCounter),通过C扩展接口重写核心算法加以优化,并与网上常见的统计工具做对 ...

  3. (zhuan) 深度学习全网最全学习资料汇总之模型介绍篇

    This blog from : http://weibo.com/ttarticle/p/show?id=2309351000224077630868614681&u=5070353058& ...

  4. 全网最全ASP.NET MVC 教程汇总

    全网最全ASP.NET MVC 教程汇总 MVC架构已深得人心,微软也不甘落后,推出了Asp.net MVC.小编特意整理博客园乃至整个网络最具价值的MVC技术原创文章,为想要学习ASP.NET MV ...

  5. 自学MVC看这里——全网最全ASP.NET MVC 教程汇总(转)

    自学MVC看这里——全网最全ASP.NET MVC 教程汇总   MVC架构已深得人心,微软也不甘落后,推出了Asp.net MVC.小编特意整理博客园乃至整个网络最具价值的MVC技术原创文章,为想要 ...

  6. 自学MVC看这里——全网最全ASP.NET MVC 教程汇总【转】

    自学MVC看这里——全网最全ASP.NET MVC 教程汇总 http://www.cnblogs.com/powertoolsteam/archive/2015/08/13/4667892.html ...

  7. Mybatis系列全解(四):全网最全!Mybatis配置文件XML全貌详解

    封面:洛小汐 作者:潘潘 做大事和做小事的难度是一样的.两者都会消耗你的时间和精力,所以如果决心做事,就要做大事,要确保你的梦想值得追求,未来的收获可以配得上你的努力. 前言 上一篇文章 <My ...

  8. 【全网最全的博客美化系列教程】08.自定义地址栏Logo

    全网最全的博客美化系列教程相关文章目录 [全网最全的博客美化系列教程]01.添加Github项目链接 [全网最全的博客美化系列教程]02.添加QQ交谈链接 [全网最全的博客美化系列教程]03.给博客添 ...

  9. 全网最全的Windows下Anaconda2 / Anaconda3里正确下载安装爬虫框架Scrapy(离线方式和在线方式)(图文详解)

    不多说,直接上干货! 参考博客 全网最全的Windows下Anaconda2 / Anaconda3里正确下载安装OpenCV(离线方式和在线方式)(图文详解) 第一步:首先,提示升级下pip 第二步 ...

  10. 全网最全的Windows下Anaconda2 / Anaconda3里正确下载安装OpenCV(离线方式和在线方式)(图文详解)

    不多说,直接上干货! 说明: Anaconda2-5.0.0-Windows-x86_64.exe安装下来,默认的Python2.7 Anaconda3-4.2.0-Windows-x86_64.ex ...

随机推荐

  1. JDBC浅应用

    1 DriverManager: 此类管理数据库驱动程序列表.使用通信协议将来自java应用程序的连接请求与适 2 当的数据库驱动程序匹配. 3 4 Driver:此接口处理与数据库服务器的通信,我们 ...

  2. 使用Oracle数据库的递归查询语句生成菜单树

    SQL 格式 SELECT * FROM TABLE WHERE [...结果过滤语句] START WITH [...递归开始条件] CONNECT BY PRIOR [...递归执行条件] 查询所 ...

  3. [车载以太网] SOME/IP 参数和数据结构的序列化

    概述:SOME/IP 参数和数据结构的序列化 大小端/字节序 每个参数(parameter)的字节顺序由接口定义进行规定. 所有的 SOME/IP Header 字段,应该以网络字节序(大端)编码. ...

  4. Windows系列操作系统,跳过开机密码直接登录

    要让 **Windows 11** 在启动时 **自动登录**,跳过开机输入密码,通常不推荐直接修改注册表来实现,但如果你希望使用注册表实现这个效果,可以按如下方式操作: --- ## 方法:修改注册 ...

  5. 以接口肢解bean factory,源码没那么神秘

    本来昨天在看 spring frame的八股, 看到了IOC部分,但是实在看不懂是什么东西,讲是讲源码部分,但又不完全讲,我想着那我要不自己看一下源码 这是我画的Bean Factory的大致关系图 ...

  6. oracleINS-13001 环境不满足最低要求

    使用windows10等系统安装oracle 11g等版本的数据库时,经常会发现开始安装时弹出[INS-13001 环境不满足最低要求]的提示,此时可以点击[是]继续安装. 也可以点击[否]结束安装, ...

  7. 在华为云服务器上测试GCC for OpenEuler的特性

    前言 操作系统课程任务 探讨 GCC for openEuler 的特性和优势 什么是 GCC for openEuler? GCC for openEuler 基于开源 GCC-10.3 版本(GC ...

  8. MRST绘制三维网格图

    MRST绘制三维网格图 Matlab储层模拟工具箱(The MATLAB Reservoir Simulation Toolbox, MRST)是一款用于储层建模的免费.开源的软件,主要由 SINTE ...

  9. C# yyyyMMddHHmmss格式转换DateTime

    https://blog.csdn.net/lilinoscar/article/details/75529821 例如14位日期:20170417101215 转换DateTime格式:    va ...

  10. windows11安装linux

    安装教程 https://blog.csdn.net/Daisy74RJ/article/details/125483629 可能遇到的问题 如果报错 则参考 WslRegisterDistribut ...