2022年

2022年发布的文章
  • C++输入输出重定向(3种方法)

    《C++输入流和输出流》一节提到,cout 和 cerr、clog 的一个区别是,cout 允许被重定向,而 cerr 和 clog 都不支持。值得一提的是,cin 也允许被重定向。

    那么,什么是重定向呢?在默认情况下,cin 只能接收从键盘输入的数据,cout 也只能将数据输出到屏幕上。但通过重定向,cin 可以将指定文件作为输入源,即接收文件中早已准备好的数据,同样 cout 可以将原本要输出到屏幕上的数据转而写到指定文件中。

    C++ 中,实现重定向的常用方式有 3 种,本节将一一做详细讲解。

    C++ freopen()函数实现重定向

    freopen() 定义在<stdio.h>头文件中,是 C 语言标准库中的函数,专门用于重定向输入流(包括 scanf()、gets() 等)和输出流(包括 printf()、puts() 等)。值得一提的是,该函数也可以对 C++ 中的 cin 和 cout 进行重定向。

    举个例子:

    #include <iostream>    //cin、cout
    #include <string>      //string
    #include <stdio.h>     //freopen
    using namespace std;
    int main()  
    {
        string name, url;
        //将标准输入流重定向到 in.txt 文件
        freopen("in.txt", "r", stdin);
        cin >> name >> url;
    
        //将标准输出重定向到 out.txt文件
        freopen("out.txt", "w", stdout); 
        cout << name << "\n" << url;
        return 0;
    }

    执行此程序之前,我们需要找到当前程序文件所在的目录,并手动创建一个 in.txt 文件,其包含的内容如下:

    C++
    http://c.biancheng.net/cplus/

    创建好 in.txt 文件之后,可以执行此程序,其执行结果为:

        <--控制台中,既不需要手动输入,也没有任何输出

    与此同时,in.txt 文件所在目录下会自动生成一个 out.txt 文件,其包含的内容和 in.txt 文件相同:

    C++
    http://c.biancheng.net/cplus/

    显然,通过 2 次调用 freopen() 函数,分别对输入流和输出流重定向,使得 cin 不再接收由键盘输入的数据,而是直接从 in.txt 文件中获取数据;同样,cout 也不再将数据输出到屏幕上,而是写入到 out.txt 文件中。

    C++ rdbuf()函数实现重定向

    rdbuf() 函数定义在<ios>头文件中,专门用于实现 C++ 输入输出流的重定向。

    值得一提的是,ios 作为 istream 和 ostream 类的基类,rdbuf() 函数也被继承,因此 cin 和 cout 可以直接调用该函数实现重定向。

    rdbuf() 函数的语法格式有 2 种,分别为:

    streambuf * rdbuf() const;
    streambuf * rdbuf(streambuf * sb);

    streambuf 是 C++ 标准库中用于表示缓冲区的类,该类的指针对象用于代指某个具体的流缓冲区。

    其中,第一种语法格式仅是返回一个指向当前流缓冲区的指针;第二种语法格式用于将 sb 指向的缓冲区设置为当前流的新缓冲区,并返回一个指向旧缓冲区的对象。

    举个例子:

    #include <iostream>
    #include <fstream>
    using namespace std;
    int main()
    {
        //打开 in.txt 文件,等待读取
        ifstream fin("in.txt");
        //打开 out.txt 文件,等待写入
        ofstream fout("out.txt");
        streambuf *oldcin;
        streambuf *oldcout;
        char a[100];
        //用 rdbuf() 重新定向,返回旧输入流缓冲区指针
        oldcin = cin.rdbuf(fin.rdbuf());
        //从input.txt文件读入
        cin >> a;
        //用 rdbuf() 重新定向,返回旧输出流缓冲区指针
        oldcout = cout.rdbuf(fout.rdbuf());
        //写入 out.txt
        cout << a << endl;
    
        //还原标准输入输出流
        cin.rdbuf(oldcin); // 恢复键盘输入
        cout.rdbuf(oldcout); //恢复屏幕输出
        //打开的文件,最终需要手动关闭
        fin.close();
        fout.close();
        return 0;
    }

    程序中涉及到的文件操作,后续章节会做详细讲解,读者只需领悟 rdbuf() 函数的用法即可。

    仍以前面创建好的 in.txt 文件为例,执行此程序后,控制台不会输出任何数据,而是会在该项目的目录下生成一个 out.txt 文件,其中就存有该程序的执行结果:

    C++
    http://c.biancheng.net/cplus/

    C++通过控制台实现重定向

    以上 2 种方法,都是从代码层面实现输入输出流的重定向。除此之外,我们还可以通过控制台实现输入输出的重定向。

    举个例子,假设有如下代码(文件名为 demo.cpp):

    #include <iostream>
    #include <string>
    using namespace std;
    int main()
    {
        string name, url;
        cin >> name >> url;
        cout << name << '\n' << url;
        return 0;
    }

    通过编译、链接后,会生成一个 demo.exe 可执行文件,该执行文件可以双击执行,也可以在控制台上执行。例如,打开控制台(Windows 系统下指的是 CMD命令行窗口,Linux 系统下指的是 Shell 终端),并输入如下指令:

    C:\Users\mengma>D:\demo.exe
    C++ http://c.biancheng.net/cplus/
    C++
    http://c.biancheng.net/cplus/

    可以看到,demo.ext 成功被执行,但程序中的 cin 和 cout 并没有被重定向,因此这里仍需要我们手动输入测试数据。

    在此基础上,继续在控制台执行如下指令:

    C:\Users\mengma>D:\demo.exe <in.txt >out.txt

    需要注意的是,执行此命令前,需保证 C:\Users\mengma 目录下有 in.txt 文件。

    执行后会发现,控制台没有任何输出。这是因为,我们使用了"<in.txt"对程序中的 cin 输入流做了重定向,同时还用 ">out.txt"对程序中的 cout 输出流做了重定向。

    如果此时读者进入 C:\Users\mengma 目录就会发现,当前目录生成了一个 out.txt 文件,其中就存储了 demo.ext 的执行结果。

    在控制台中使用 > 或者 < 实现重定向的方式,DOS、windows、Linux 以及 UNIX 都能自动识别。

更多...

加载中...