62

我一直在想,堵塞有什么意义?据我所知, clog 与 cerr 相同,但具有缓冲功能,因此效率更高。通常stderr和stdout一样,所以clog和cout一样。这对我来说似乎很蹩脚,所以我想我一定是误解了它。如果我有日志消息发送到同一个地方,我有错误消息发送到(可能是 /var/log/messages 中的某些内容),那么我可能不会写太多(所以使用 non 不会丢失太多-缓冲 cerr)。根据我的经验,我希望我的日志消息是最新的(不是缓冲的),所以我可以帮助找到崩溃(所以我不想使用缓冲的阻塞)。显然我应该一直使用 cerr。

我希望能够在我的程序中重定向堵塞。重定向 cerr 会很有用,这样当我调用库例程时,我可以控制 cerr 和 clog 的去向。一些编译器可以支持这个吗?我刚刚检查了 DJGPP,stdout 被定义为 FILE 结构的地址,因此执行“stdout = freopen(...)”之类的操作是非法的。

  • 是否可以重定向 clog、cerr、cout、stdin、stdout 和/或 stderr?
  • clog 和 cerr 之间的唯一区别是缓冲吗?
  • 我应该如何实现(或找到)更强大的日志记录工具(请链接)?
4

6 回答 6

42

是否可以重定向 clog、cerr、cout、stdin、stdout 和/或 stderr?

是的。你想要这个rdbuf功能。

ofstream ofs("logfile");
cout.rdbuf(ofs.rdbuf());
cout << "Goes to file." << endl;

clog 和 cerr 之间的唯一区别是缓冲吗?

据我所知,是的。

于 2008-09-09T17:13:03.497 回答
15

如果您在 posix shell 环境中(我真的在考虑 bash),您可以将任何文件描述符重定向到任何其他文件描述符,因此要重定向,您可以:

$ myprogram 2>&5 

将 stderr 重定向到 fd=5 表示的文件。

编辑:再想一想,我更喜欢@Konrad Rudolph 关于重定向的回答。rdbuf() 是一种更连贯且可移植的方法。

至于日志记录,嗯......我从 Boost 库开始,用于所有不在 std 库中的 C++。看哪:Boost Logging v2

编辑:Boost Logging不是Boost 库的一部分;它已经过审核,但没有被接受。

编辑:2 年后,早在 2010 年 5 月,Boost 确实接受了一个日志库,现在称为Boost.Log

当然,还有其他选择:

  • Log4Cpp(用于 C++ 的 log4j 风格的 API)
  • Log4Cxx(Apache 赞助的 log4j 风格的 API)
  • Pantheios(已失效?上次我尝试无法在最近的编译器上构建它)
  • 谷歌的 GLog(帽子提示@SuperElectric)

还有 Windows 事件记录器。

还有几篇可能有用的文章:

于 2008-09-09T17:16:52.350 回答
4

由于这里有几个关于重定向的答案,我将添加我最近偶然发现的关于重定向的这个不错的 gem :

#include <fstream>
#include <iostream>

class redirecter
{
public:
    redirecter(std::ostream & dst, std::ostream & src)
        : src(src), sbuf(src.rdbuf(dst.rdbuf())) {}
    ~redirecter() { src.rdbuf(sbuf); }
private:
    std::ostream & src;
    std::streambuf * const sbuf;
};

void hello_world()
{
    std::cout << "Hello, world!\n";
}

int main()
{
    std::ofstream log("hello-world.log");
    redirecter redirect(log, std::cout);
    hello_world();
    return 0;
}

它基本上是一个重定向类,允许您重定向任意两个流,并在完成后恢复它。

于 2019-05-25T20:00:44.537 回答
4

重定向

关于如何重定向std::clogstd::wclog),康拉德鲁道夫(Konrad Rudolph)的回答很好。

其他答案告诉您各种可能性,例如使用命令行重定向,例如2>output.log. 使用 Unix,您还可以创建一个文件并将另一个输出添加到您的命令中,例如3>output.log. 在您的程序中,您必须使用fd数字 3 来打印日志。您可以继续stdout正常打印stderr。Visual Studio IDE 的命令具有类似的功能CDebug,可将其输出发送到 IDE 输出窗口。

stderr是一样的stdout吗?

这通常是正确的,但在 Unix 下,您可以设置它stderr/dev/console这意味着它会转到另一个tty(又名终端)。这些天很少使用它。我在 IRIX 上就是这样。我会打开一个单独的 X-Window 并查看其中的错误。

也有很多人向/dev/null. 在你写的命令行上:

command ...args... 2>/dev/null

系统日志

没有提到的一件事,在 Unix 下,你也有syslog().

Linux 下的最新版本(可能还有 Mac OS/X)的功能比以前要多得多。特别是,它可以使用身份和一些其他参数将日志重定向到特定文件(即mail.log)。syslog 机制可以在计算机之间使用,因此计算机 A 的日志可以发送到计算机 B。当然,您可以通过各种方式过滤日志,尤其是按严重性。

使用syslog()起来也非常简单:

syslog(LOG_ERR, "message #%d", count++);

它提供了 8 个级别(或严重性)、一个 la 格式printf()以及该格式的参数列表。

openlog()以编程方式,如果您第一次调用该函数,您可能会调整一些东西。您必须在第一次调用之前调用它syslog()

正如 unixman83 所提到的,您可能想要使用宏来代替。这样,您可以在消息中包含一些参数,而不必一遍又一遍地重复它们。也许是这样的(参见Variadic Macro):

// (not tested... requires msg to be a string literal)
#define LOG(lvl, msg, ...) \
     syslog(lvl, msg " (in " __FILE__ ":%d)", __VA_ARGS__, __LINE__)

您可能还会发现__func__有用。

重定向、过滤等是通过创建配置文件来完成的。这是我的 snapwebsites 项目中的一个示例:

mail.err /var/log/mail/mail.err
mail.* /var/log/mail/mail.log
& stop

我在下面安装文件/etc/rsyslog.d/并运行:

invoke-rc.d rsyslog restart

因此syslog服务器会处理该更改并将任何与邮件相关的日志保存到这些文件夹中。

注意:我还必须创建/var/log/mail文件夹和文件夹内的文件以确保一切正常(因为否则邮件守护程序可能没有足够的权限。)

snaplogger(一个小插件)

我使用过log4cplus,从 1.2.x 版开始,它非常好。不过,我对此有三个缺点:

  1. 如果我想以某种方式调用它需要我完全清除所有内容,fork();它不能fork();正常地在调用中存活......(至少在我使用线程的版本中)
  2. .properties在我喜欢管理员在不修改原始文件的情况下进行更改的环境中,配置文件 ( ) 不容易管理
  3. 它使用 C++03,我们现在是 2019 年……我希望至少有 C++11

正因为如此,尤其是因为第 (1) 点,我编写了自己的版本snaplogger。不过,这并不完全是一个独立的项目。我使用 snapcpp 环境中的许多其他项目(只需获取 snapcpp 并运行bin/build-snap脚本或从 launchpad 获取二进制文件要容易得多。)

使用 snaplogger 或 log4cplus 等记录器的优势在于,您通常可以定义任意数量的目标和许多其他参数(例如 提供的严重性级别syslog())。log4cplus 能够将其输出发送到许多不同的地方:文件、系统日志、MS-Windows 日志系统、控制台、服务器等。查看这两个项目中的附加程序以了解可能性列表。这里有趣的因素是任何日志都可以发送到所有目的地。这对于有一个名为 all.log 的文件很有用,所有服务都在其中发送它们的日志。这允许理解某些在并行运行许多服务时使用单独的日志文件并不容易的错误。

这是 snaplogger 配置文件中的一个简单示例:

[all]
type=file
lock=true
filename=/var/log/snapwebsites/all.log

[file]
lock=false
filename=/var/log/snapwebsites/firewall.log

请注意,对于该all.log文件,我需要一个锁,因此多个编写者不会破坏彼此之间的日志。这[file]部分没有必要,因为我只有一个进程(没有线程)。

两者都为您提供了一种添加自己的附加程序的方法。因此,例如,如果您有一个带有输出窗口的 Qt 应用程序,您可以编写一个附加程序来将SNAP_LOG_ERROR()调用的输出发送到该窗口。

snaplogger还为您提供了一种方法来扩展消息中的变量支持(也称为格式)。例如,我可以使用${date}变量插入日期。然后我可以用一个参数来调整它。为了只输出年份,我使用${date:year}. 这种可变参数支持也是可扩展的。

snaplogger可以按严重性(如系统日志)、正则表达式和组件过滤输出。我们有一个normal和一个secure组件,默认是normal. 我希望将发送到secure组件的日志写入安全文件。这意味着在一个子目录中,它比大多数管理员可以查看的普通日志受到更多保护。当我运行我的 HTTP 服务时,有时我会发送信用卡的最后 3 位数字等信息。我更喜欢将它们保存在安全日志中。也可能是与密码相关的错误。任何我认为在日志中存在安全风险的东西,真的。同样,组件是可扩展的,因此您可以拥有自己的组件。

于 2019-08-01T17:53:50.967 回答
2

关于重定向器类的一点。它需要被正确地销毁,而且只需要一次。析构函数将确保在声明它的函数实际返回时会发生这种情况,并且对象本身永远不会被复制。

为确保它不能被复制,请提供私有复制和赋值运算符:

class redirecter
{
public:
    redirecter(std::ostream & src, std::ostream & dst)
        : src_(src), sbuf(src.rdbuf(dst.rdbuf())) {}
    ~redirecter() { src.rdbuf(sbuf); }
private:
    std::ostream & src_;
    std::streambuf * const sbuf_;
    // Prevent copying.                        
    redirecter( const redirecter& );
    redirecter& operator=( const redirecter& );
};

我通过将 std::clog 重定向到我的 main() 中的日志文件来使用这种技术。为了确保 main() 实际返回,我将 main() 的内容放在 try/catch 块中。然后在我的程序的其他地方,我可能会调用 exit(),我会抛出一个异常。这会将控制权返回给 main(),然后它可以执行 return 语句。

于 2020-07-02T05:02:03.183 回答
1

基本记录器

#define myerr(e) {CriticalSectionLocker crit; std::cerr << e << std::endl;}

用作myerr("ERR: " << message);myerr("WARN: " << message << code << etc);

非常有效。

然后做:

./programname.exe 2> ./stderr.log
perl parsestderr.pl stderr.log

或者只是手动解析 stderr.log

我承认这不适用于性能极其关键的代码。但无论如何,谁写的。

于 2010-12-18T01:39:02.320 回答