Попытка извлечь из wfilebuf (filebuf) для ведения журнала - PullRequest
0 голосов
/ 13 ноября 2010

Я в основном пытаюсь извлечь из wfilebuf, чтобы я мог как выводить в файл, так и перехватывать вывод, чтобы распечатать его в окне консоли / отладки, а также показано здесь: http://savingyoutime.wordpress.com/2009/04/21/ и / или здесь:http://savingyoutime.wordpress.com/2009/04/22/40/

(древние поддерживающие идеи здесь: http://www.horstmann.com/cpp/streams.txt)

Я почти получил это, но я не могу как написать в основной файл, так и посмотреть наinput.

Я переопределил функцию sync (), как во втором примере, но кажется, что pbase () и pptr () всегда равны NULL, если я не установил буфер с помощью setp (...), но этоКажется, что это нарушает вывод файла. Файл всегда пуст!

Моя грубая попытка сделать это ниже:

class LoggerBuffer : public wfilebuf {
// Functions
    public:
        LoggerBuffer();
        ~LoggerBuffer();
        void open(const wchar_t loggerFile[]);
        void close();
        int sync();
        int_type overflow(int_type c = EOF);
        void setState(int newState);
// Variables
    private:
        int currentState;
        static const int BUFFER_SIZE = 10;
        wchar_t buffer[BUFFER_SIZE];  
};

class LoggerStream : public wostream {
// Functions
    public:
         LoggerStream();
         ~LoggerStream();
         void open(const wchar_t loggerFile[] = 0);
         void close();
         void setState(int newState);
};

LoggerBuffer::LoggerBuffer() {
    wfilebuf::open("NUL", wios::out); currentState = 1;
}
LoggerBuffer::~LoggerBuffer() {
    wcout << "Destruction of LoggerBuffer" << endl;
}
void LoggerBuffer::open(const wchar_t loggerFile[]) {
    wcout << "LoggerBuffer Opening " << loggerFile << endl;
    close();
    wfilebuf* temp = wfilebuf::open(loggerFile, wios::out); //ios::out | ios::app | ios::trunc
    setp (buffer, buffer+(BUFFER_SIZE-1));
}
void LoggerBuffer::close() {
    wfilebuf::close();
}

int LoggerBuffer::sync() {
    wcout << "  Syncing ";
    int out_waiting = pptr() - pbase();
    wcout << out_waiting << " characters!";
    wcout << endl;
    wcout << "pptr(): " << (unsigned int)pptr() << endl;
    return wfilebuf::sync();
}
LoggerBuffer::int_type LoggerBuffer::overflow(int_type c) {
    wcout << "overflow! (" << (wchar_t)c << ")" << endl;
    if (c == EOF)
        return EOF;
    if (sync() == EOF)
        return EOF;
    return wfilebuf::overflow(c);
}
void LoggerBuffer::setState(int newState) {
    wcout << "New buffer state = " << newState << endl;
    currentState = newState;
}

LoggerStream::LoggerStream() : wostream(new LoggerBuffer), wios(0) {
}
LoggerStream::~LoggerStream() {
    delete rdbuf();
}
void LoggerStream::open(const wchar_t loggerFile[]) {
    wcout << "LoggerStream Opening " << loggerFile << endl;
    ((LoggerBuffer*)rdbuf())->open(loggerFile);
}
void LoggerStream::close() {
    ((LoggerBuffer*)rdbuf())->close();
}
void LoggerStream::setState(int newState) {
    wcout << "New stream state = " << newState << endl;
    ((LoggerBuffer*)rdbuf())->setState(newState);
}

Полное раскрытие: я задавал вопрос относительно чего-то подобного ранее: Простой класс регистрации Wostream (с пользовательскими потоковыми манипуляторами)

Я думаю, что я решил эту проблему, хотя.

Любая помощь очень ценится! Спасибо!

1 Ответ

0 голосов
/ 13 ноября 2010

Я бы использовал фильтрующий потоковый буф , который не выполняет свою собственную буферизацию, вместо этого передавая данные в реальный потоковый буф (т. Е. Тот, который выполняет настоящую буферизацию) для каждого из пунктов назначения.Это должно немного упростить ваш код и позволить вам сконцентрироваться на тех частях, которые вам действительно нужны.

...