我有一个 Logger 类,它以下列方式期望一个 std::string:
class Logger
{
public:
void Log(const std::string message);
};
另一个类称它为:
class MyClass
{
public:
void MyMethod()
{
Logger logger;
logger.Log("Hello World!"); // This line is causing linker error.
// std::string myString("Hello World!"); If this and the line below are
// logger.Log(myString); uncommented, I get no errors.
}
};
错误信息是:
Logger.cpp: undefined reference to `Logger::Log(std::basic_string<char, std::char_traits<char>, std::allocator<char> >)`
显然链接器知道 Log 方法,因为正如我上面所展示的,注释掉的行将允许它正确链接。
有人知道发生了什么事吗?如果需要更多信息,请告诉我。
这是在Linux下使用gcc 4.4.3(使用Eclipse)
编辑:
据我了解,您始终可以将 C 风格的字符串传递给需要 std::string 的方法,因为 std::string 有一个适当的复制构造函数来处理它。
编辑x2:
捂脸时刻。我发现了错误。感谢大家让我更仔细地审视我的代码。请参阅下面的完整解决方案。
这段代码应该添加到问题的这里:
// cpp file
// This should have been in the header, not the cpp file.
inline void Logger::Log(const std::string message)
{
/* do sutff */
}
最佳答案
你的问题是你没有定义Log
(又名Logger::Log
)方法。您已经声明了它,但没有在任何地方定义它。
另一个小问题是你真的应该这样声明它:
void Log(const ::std::string &message);
没有特别好的理由强制调用者复制他们传递给您的字符串。
有几种可能性可以解释为什么您可能认为它已定义,即使它没有定义。
首先,如果定义了函数,您可能没有链接到与 .cpp
文件对应的适当的 .o
文件中。这意味着您已经定义了它,但链接器不知道您的定义。
另一种可能性是您定义了一个具有相似名称的函数,但与正在调用的函数不同。
另一种可能性是您已将函数定义为内联函数。如果该函数是内联定义的,则编译器可能尚未为其生成代码。如果定义在引用时可用,许多编译器只会为内联函数发出代码,并且引用方式要求函数在某处具有实际代码(例如获取其地址)。
如果你有一个定义为内联的函数,该函数通常应该在头文件中,否则 inline
关键字是无用的,甚至可能会给你带来问题,因为你的所有其他部分code 将假设有实际代码驻留在支持您的函数声明的特定地址,除了它出现的一个 .cpp
文件,它可能不会以强制代码的方式引用该函数被发射。
关于c++ - 为什么在将 C 样式数组传递给需要 std::string 的方法时出现链接器问题?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5517786/