12

我正在尝试使用 freopen() 打印到文本文件和屏幕,但我只是实现打印到文件。

我想知道是否可以轻松地将程序输出保存到文件并将其打印到屏幕上?因为我以另一种方式工作,但我最终不得不将每条语句打印两次。一个用于文件,另一个仅用于输出。

注意:我是 C++ 的新手,我正在尝试为下学期的课程学习它,所以需要直接的答案,因为我已经在网上查找过,除了这个解决方案之外找不到任何简单的答案。

这是我到目前为止所拥有的:

#include<iostream>
#include<time.h>
#include<stdlib.h>
#include<fstream>

using namespace std; 

void menu(){
    cout << "\t********************************************************\n"
         << "\t* Welcome to slot machine.                             *\n"
         << "\t* Would you like to play? (1 to play, 2 not to play)   *\n"
         << "\t********************************************************\n\n";
    return;
}

void update(int arr[], int &token) {
    if (arr[0]==arr[1] && arr[1]==arr[2]) {
        token+=4;
        cout << "You win\n\n";
    } else if (arr[0]==arr[1] || arr[1]==arr[2] || arr[0]==arr[2]) {
        token+=1;
        cout << "You got two out of three\n\n";
    } else {
        token-=1;
        cout << "You lose\n\n";
    }
}

int main() {
    freopen("file.txt", "w", stdout);
    int x, arr[3], token=4;
    srand(time(0));
    menu();
    cin >> x;
    while(token!=0) {
        cout << "You have " << token << " tokens\n\n"
             << "Pull? (1 to pull, 2 not to pull)\n\n";
        cin>>x;
        if(x==1) {
            for(int i=0; i<3; i++) {
                arr[i]=1+rand()%10;
            }
            cout << "\t\t";
            for(int j=0; j<3; j++) {
                cout << arr[j] << " ";
            }
            cout << "\n\n";
            update(arr,token);
        }
        else{  
            cout << "OK\n";
        }
    }
    cin.get();
    return 0;
}
4

2 回答 2

6

我不知道实现这一目标的简单方法,但我设法以某种方式解决了这个问题。

使用 fstreams,您可以像写入控制台一样输出到文件。

#include <fstream>

int main()
{
     std::ofstream f("file.txt");
     f << "something";
}

现在有一点我们可以开始了:有没有一种方法可以同时输出到控制台和文件?

我最近编写了流解复用器来解决这个问题:

#include <vector>
#include <ostream>
class stream_demultiplexer
{
private:
    typedef std::vector<std::ostream*> str_cont;
    str_cont d;
public:
    stream_demultiplexer& put(std::ostream::char_type ch)
    {
        for(str_cont::iterator it = d.begin(); it != d.end(); ++it)
            (*it)->put(ch);
        return *this;
    }

    stream_demultiplexer& write(const std::ostream::char_type* s, std::streamsize count)
    {
        for(str_cont::iterator it = d.begin(); it != d.end(); ++it)
            (*it)->write(s, count);
        return *this;
    }

    stream_demultiplexer& flush()
    {
        for(str_cont::iterator it = d.begin(); it != d.end(); ++it)
            (*it)->flush();
        return *this;
    }


    template<typename T>
    stream_demultiplexer& operator<<( const T& obj )
    {
        for(str_cont::iterator it = d.begin(); it != d.end(); ++it)
            (**it) << obj;
        return *this;
    }

    stream_demultiplexer& operator<<(std::ios_base& (*func)(std::ios_base&))
    {
        for(str_cont::iterator it = d.begin(); it != d.end(); ++it)
            (**it) << func;
        return *this;
    }

    template<typename CharT, typename Traits>
    stream_demultiplexer& operator<<(std::basic_ios<CharT,Traits>& (*func)(std::basic_ios<CharT,Traits>&) )
    {
        for(str_cont::iterator it = d.begin(); it != d.end(); ++it)
            (**it) << func;
        return *this;
    }

    stream_demultiplexer& operator<<(std::ostream& (*func)(std::ostream&) )
    {
        for(str_cont::iterator it = d.begin(); it != d.end(); ++it)
            (**it) << func;
        return *this;
    }

    void add_stream(std::ostream& ss)
    {
        d.push_back(&ss);
    }
};

你可以像这样使用它:

stream_demultiplexer spl;
std::ofstream f("file.txt");
spl.add_stream(f);
spl.add_stream(std::cout);
spl << 55 << " HELLO WORLD";

我的方法的优点是操纵器和未格式化的输出可以正常工作:

spl << 76 << " " << std::hex << 76 << std::endl;
spl.put('a');
spl.write("ABCDE", 5);
于 2012-12-20T18:59:09.267 回答
4

在类 UNIX 环境中,最简单的方法是使用 shell 命令tee

$ my-program | tee output.txt

将 stdout 复制到终端,也复制到文件output.txt.


如果您必须在代码中执行此操作,您可以使用自己的输出流而不是cout,它将每一个转发operator<<到两个(或更多)ostreams。(对我来说)这感觉比在 C++ ostream 底层的 C 输出文件中乱七八糟更好cout

#include <ostream>

class Tee {
    std::ostream &first, &second;

    template<typename T> friend Tee& operator<< (Tee&, T);

public:
    Tee(std::ostream &f, std::ostream &s) : first(f), second(s) {}
};

template <typename T>
Tee& operator<< (Tee &t, T val)
{
    t.first << val;
    t.second << val;
    return t;
}

然后,如果您将 freopen 行替换为:

std::ofstream outfile("file.txt");
Tee tee(std::cout, outfile);

你可以只使用tee <<而不是cout <<.

请注意,您要么需要传递tee给您的函数,要么将其设为全局函数才能正常工作。

于 2012-12-20T18:58:33.027 回答