Эта заметка была написана в моем втором по счету блоге на сайте blogspot два года назад. Сейчас я уже пришел к пониманию, что такая система логов мне не нравится. Она довольно навороченная, но все эти шаблоны меня теперь почему-то пугают. Дабы это добро не сгинуло вместе с тем дневником, публикую здесь. Прочитав данную статью, возможно, вы сможете подцепить тёлочку блеснуть умом в узких кругах. :)
Полгода назад мы создавали крутой логгер, использование которого выглядело так:
logger::info() << "Тестовое сообщение номер " << 1; |
Самая главная проблема была в том, чтобы воткнуть ‘\n’ в конце сообщения. Для этого в info() был примерно следующий код:
InfoStream info() { static InfoStream stream; // std::endl пишет в ostream '\n' и делает flush() stream << std::endl; return stream; } |
Последняя строка завершалась только при закрытии файла (в деструкторе класса InfoStream). Первый flush игнорировался. Это все кажется простым, если мы пишем лог в обычный текстовый файл.
Когда лог стал в формате html, начались пляски с бубном.
Во-первых, в html надо записать заголовок, во-вторых, красиво оформить начало строки, в-третьих, красиво закончить строку, в-четвертых, вставить теги в конце файла. Это было реализовано, но совместимость между использованием текстового лога и html в некотором смысле пропала (базовый класс для обоих типов выглядел монстром).
Совсем недавно пришлось копаться во внутренностях. Задача стояла — вынести этот логгер в отдельный набор заголовков а-ля boost. Посмотрев на этот код, я плюнул и решил написать логгер заново. :)
Вот как это было реализовано:
class logger { public: class internal_stream { public: template <class T> internal_stream & operator <<(const T & value) { logger << value; return *this; } ~internal_stream() {} private: // Может использоваться только классом logger internal_stream(logger & stream) : mStream(stream) {}; logger & mStream; }; internal_stream info() { return internal_stream(*this); } }; |
Как это все работает:
- в конструкторе logger — подготавливаем файл для записи, в текстовом ничего не делаем, в html — записываем заголовок
- в деструкторе logger — закрываем файл, в html формате закрываем теги body, html и т.д.
- в конструкторе internal_stream — начало строки, в html формате пишем tr, td и все что нравится
- в деструкторе internal_stream — окончание строки, в текстовом это просто ‘\n’, в html закрываем теги tr и т.д.
Если сделать logger шаблонным классом, то можно реализовать разные буферы вывода. Начиная от текстового, html и xml и заканчивая сервером для приема и клиента для отправки сообщений по сети.