我正在尝试部署我的应用程序,它使用Boost Log(Boost 1.58)。它是一个简单的控制台应用程序,在Windows 7中运行。日志在我的个人桌面上运行得非常好。
但是,当我将应用程序部署到Win7虚拟机时,它会在我的第一个日志语句中崩溃:
boost::log::sources::severity_logger<SeverityLevel> slg;
BOOST_LOG_SEV(slg, SeverityLevel::Notification) << L"Application loaded"; // <-- Crash here
创建了日志目录,但永远不会创建日志文件并且应用程序崩溃。
我在我的%APPDATA%目录中尝试了一个日志文件目录,也在我的My Documents目录中。
奇怪的是:当我以管理员身份运行应用程序时,它可以正常运行!
所以这必须是权限,但我对这些文件夹有权限,所以......
有什么想法吗?
* MORE *
以下是设置记录器的代码:
wstring appData = GetMyAppDataPath();
boost::shared_ptr< boost::log::sinks::synchronous_sink< boost::log::sinks::text_file_backend > > textFileSink(new boost::log::sinks::synchronous_sink< boost::log::sinks::text_file_backend >(to store rotated files
boost::log::keywords::file_name = "log_%7N.log", // file name pattern
boost::log::keywords::rotation_size = 16384 // rotation size, in characters
));
// Set up where the rotated files will be stored
textFileSink->locked_backend()->set_file_collector(boost::log::sinks::file::make_collector(
boost::log::keywords::target = appData + L"\\logs", // where to store rotated files
boost::log::keywords::max_size = 64 * 1024 * 1024, // maximum total size of the stored files, in bytes (64 MiB)
boost::log::keywords::min_free_space = 100 * 1024 * 1024 // minimum free space on the drive, in bytes
));
// Upon restart, scan the target directory for files matching the file_name pattern
textFileSink->locked_backend()->scan_for_files();
// Set up the format for output to the text file.
textFileSink->set_formatter(boost::log::expressions::stream
<< "[" << boost::log::expressions::attr< boost::posix_time::ptime >("TimeStamp")
<< " " << boost::log::expressions::attr< SeverityLevel, severity_tag >("Severity") << "] "
<< boost::log::expressions::message
);
// Add it to the core
boost::log::core::get()->add_sink(textFileSink);
主要取自此处:http://www.boost.org/doc/libs/1_58_0/libs/log/example/rotating_file/main.cpp。
* ALSO *
我通过添加:
为Boost Logger添加了一个异常处理程序boost::log::core::get()->set_exception_handler(boost::log::make_exception_handler<
std::runtime_error,
std::logic_error,
std::exception
>(pbb_boost_log_exception_handler()));
然后添加处理程序。当我运行时,我能够在崩溃之前捕获以下异常:
std::exception: Failed to open file for writing: Input/output error: "C:\Program Files\My App\log_0000000.log"
WTF?我肯定将日志文件位置设置为appData
值,我已验证该值是正确的。此外,如果我以管理员身份运行此应用程序,则日志文件最终会出现在我期望的位置(appdata文件夹)。所以它必须只是在可执行文件的位置创建一个临时文件。这是正常的行为吗?我无法想象它是......所以我做了什么?
答案 0 :(得分:4)
好的,我明白了。
问题在于此处,设置文本后端:
boost::shared_ptr< boost::log::sinks::synchronous_sink< boost::log::sinks::text_file_backend > > textFileSink(new boost::log::sinks::synchronous_sink< boost::log::sinks::text_file_backend >
boost::log::keywords::file_name = "log_%7N.log", // file name pattern
boost::log::keywords::rotation_size = 16384 // rotation size, in characters
));
肯定会有一个临时日志文件写入本地目录(Program Files目录,这是坏的)。基于documentation here,我看到是写入的临时文件,然后传递给文件收集器。
因此,对于解决方案,我将代码更改为:
boost::shared_ptr< boost::log::sinks::synchronous_sink< boost::log::sinks::text_file_backend > > textFileSink(new boost::log::sinks::synchronous_sink< boost::log::sinks::text_file_backend >(
boost::log::keywords::file_name = appData + L"\\log_%7N.log", // file name pattern
boost::log::keywords::rotation_size = 16384 // rotation size, in characters
));
请注意,我现在将file_name
指定为位于AppData目录中。
这解决了问题。
我很难相信我是第一个遇到这个问题的人,但我无法在网络上找到它。对于Windows开发人员来说,这将是一个经常出现的问题,所以希望这对其他人有帮助。