c++ - 为什么在将 C 样式数组传递给需要 std::string 的方法时出现链接器问题?

标签 c++ string gcc linker g++

我有一个 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/

相关文章:

c++ - 如何将重载的new/delete运算符的范围限制为仅翻译单元(当不能将它们设为静态时)

C++ 在编译时将整数转换为字符串

python - 在 CentOS 6.7 版(最终版)上导入 lxml 失败

c++ - 以结构作为 boost::unordered_map 中的键的 boost 变体

c++ - 为什么不允许具有不同数量模板参数的类/结构声明?

python - 将秒转换为分秒

c - 在c中标记字符串的快速方法

c++ - std::string 不是 std::vector<char> 吗?

c++ - 编译对象后链接共享库时出错

c - 交换C中多维数组的子数组