贝利信息

c++中如何实现简单的日志系统_c++输出日志到文件的方法【实例】

日期:2026-01-19 00:00 / 作者:尼克
推荐程序启动时以std::ios::app模式打开std::ofstream并保持开启,需显式.flush()或用std::endl刷新缓冲区;路径用/或\\;务必检查is_open()防静默失败。

std::ofstream 实现基础日志写入

最直接的方式是每次记录日志时打开文件、追加内容、再关闭。但频繁开闭文件效率低,且多线程下容易出错。推荐在程序启动时打开一个 std::ofstream 对象并保持打开状态,用 std::ios::app 模式确保追加写入。

std::ofstream log_file("app.log", std::ios::app);
if (!log_file.is_open()) {
    std::cerr << "Failed to open log file\n";
    return;
}
log_file << "[INFO] Application started at " << std::time(nullptr) << "\n";
log_file.flush(); // 关键:确保立即写入磁盘

封装成线程安全的简易日志类

裸用 std::ofstream 在多线程场景下会因竞态导致日志错乱(如两行内容混在同一行)。加 std::mutex 是最低成本方案,但注意不要在锁内做耗时操作(比如格式化时间)。

class SimpleLogger {
    std::ofstream file_;
    mutable std::mutex mtx_;
public:
    explicit SimpleLogger(const std::string& path) : file_(path, std::ios::app) {}
    void log(const std::string& level, const std::string& msg) const {
        auto now = std::chrono::system_clock::now();
        auto time_t = std::chrono::system_clock::to_time_t(now);
        std::ostringstream oss;
        oss << "[" << level << "] " << std::put_time(std::localtime(&time_t), "%Y-%m-%d %H:%M:%S") << " - " << msg << "\n";
        std::lock_guard lock(mtx_);
        file_ << oss.str();
        file_.flush();
    }
};

避免 std::endl 导致的性能陷阱

很多人习惯用 换行,但它等价于 。高频日志场景下,每次写都刷盘会严重拖慢性能。实际只需在关键节点(如进程退出前、每 100 条日志)手动 flush 即可。

Windows 下中文日志乱码问题怎么解

在 Windows 控制台或记事本中打开日志文件显示乱码,大概率是编码问题。VS 默认用 GBK 编译源码,但 ofstream 写出的是 UTF-8 字节流(无 BOM),记事本会误判为 ANSI。

真正难处理的是日志内容来自用户输入或第三方 API 的宽字符串——这时候绕不开编码转换,别图省事用 WideCharToMultiByte(CP_ACP),它依赖系统区域设置,部署到其他机器可能崩。