由std::string继承引起的LNK 2005错误

本文介绍了解决Visual Studio 2012中出现的LNK2005错误的方法,该错误通常出现在DLL或EXE的链接阶段。文章分析了错误产生的原因,并提供了一种有效的解决方案——将派生类改为组合方式。

    最近在用VS2012构建项目时,发现在编译DLL或EXE的链接阶段,会提示LNK2005错误:

error LNK2005: "public: __thiscall std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >::~basic_string<char,struct std::char_traits<char>,class std::allocator<char> >(void)" (??1?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@XZ) already defined in xyz.obj

     经过排查,确保没有重复类型定义,并且运行库格式一致(都为/MTd或/MDd),没有时间仔细考虑这个问题,只是通过加一个/force选项勉强通过编译。

     今天在网上搜索,发现其他人也遇到过类似问题,原因是由于导入的动态链接库中,定义了从std::string派生的类。参见:http://social.msdn.microsoft.com/Forums/vstudio/en-US/9ec13f84-0396-4078-be2c-2448dc4eef8b/lnk-2005-error-with-dll-in-msvc-10-class-derived-from-stdstring?forum=vcgeneral

     而我们的引用的库中,也确实有一个从std::basic_string<>派生的类。

     对于这一问题,微软工程师给出的建议是,将继承改为组合,却没有说明引发链接错误的根本原因。

     按照上面的建议,将那个类改为组合basic_string实现,果然问题消失了。

    

#include "itek_log.h" ItkLog::ItkLog():m_init(false){}; ItkLog::~ItkLog() { if (m_init) { this->unInit(); } } bool ItkLog::addConsole(const char* logger_name, const OutLevel level) { auto console_sink = std::make_shared<spdlog::sinks::stdout_color_sink_mt>(); console_sink->set_level((spdlog::level::level_enum)level); updateSinkCollection(logger_name, console_sink); return true; } bool ItkLog::addRotatingFile(const char* logger_name, const char* file_name, const int max_file_size, const int max_file_num, const OutLevel level) { auto file_sink = std::make_shared<spdlog::sinks::rotating_file_sink_mt>(file_name, max_file_size, max_file_num); file_sink->set_level((spdlog::level::level_enum)level); updateSinkCollection(logger_name, file_sink); return true; } bool ItkLog::addDailyFile(const char* logger_name, const char* file_name, const int hour, const int minute, const OutLevel level) { auto daily_sink = std::make_shared<spdlog::sinks::daily_file_sink_mt>(file_name, hour, minute); daily_sink->set_level((spdlog::level::level_enum)level); updateSinkCollection(logger_name, daily_sink); return true; } bool ItkLog::init(const OutMode mode, const std::string format) { if (m_init) { return false; } m_init = true; if (mode == ASYNC) { for (auto element : log_sinks) { std::string t_log_name = element.first; std::vector<std::shared_ptr<spdlog::sinks::sink>> t_sink(element.second); auto t_thread_pool = std::make_shared<spdlog::details::thread_pool>(1024000, 1); auto t_logger = std::make_shared<spdlog::async_logger>(t_log_name, std::begin(t_sink), std::end(t_sink), t_thread_pool, spdlog::async_overflow_policy::block); /*updateThreadPool(t_log_name, t_thread_pool);*/ t_logger->set_level(spdlog::level::trace); t_logger->flush_on(spdlog::level::warn); spdlog::register_logger(t_logger); } } else if (mode == SYNC) { for (auto element : log_sinks) { std::string t_log_name = element.first; std::vector<std::shared_ptr<spdlog::sinks::sink>> t_sink(element.second); auto t_logger = std::make_shared<spdlog::logger>(t_log_name, std::begin(t_sink), std::end(t_sink)); t_logger->set_level(spdlog::level::trace); t_logger->flush_on(spdlog::level::warn); spdlog::register_logger(t_logger); } } spdlog::flush_every(std::chrono::seconds(3)); spdlog::set_pattern(format); return true; } void ItkLog::unInit() { spdlog::shutdown(); //spdlog::drop_all(); //spdlog::shutdown(); } void ItkLog::updateSinkCollection(const std::string logger_name, std::shared_ptr<spdlog::sinks::sink> sink) { auto iter = log_sinks.find(logger_name); if (iter != log_sinks.end()) { iter->second.push_back(sink); } else { std::vector<spdlog::sink_ptr> t_sink; t_sink.push_back(sink); log_sinks[logger_name] = t_sink; } } // void updateThreadPool(const std::string logger_name, std::shared_ptr<spdlog::details::thread_pool> thread_pool) // { // auto iter = m_mapAsyncThreadPool.find(logger_name); // if (iter != m_mapAsyncThreadPool.end()) // { // iter->second = (async_thread_pool); // } // else // { // m_mapAsyncThreadPool[logger_name] = async_thread_pool; // } // }析构函数总是报错
05-28
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值