C++ 程序调用 C 函数在 g++ 链接期间获取 undefined reference

标签 c++ c gcc g++

我有一个简单的 C++ 程序可以调用 C 函数。当我链接 2 个目标文件以创建 C++ 可执行文件时,我收到许多“ undefined reference ”错误和单个“对 svrport 的多个引用”错误。我有以下文件 chorusmain.cpp、chorusclient.c 和 chorusclient.h。 chorusmain.cpp 调用 chorusclient.c 中的函数。 chorusclient.c 非常大,是我 2 年前写的。我现在需要在 C++ 应用程序中使用这些函数,不想用 C++ 重新编写并重新调试代码。我将 chorusclient.c 重构为一组可调用函数。为简单起见,我将代码缩减为仅包含重要组件。

chorusclient.c

#include <...>
#include "chorusclient.h"

int my_C_Function()
{
  ...
}

chorusclient.h

#ifdef __cplusplus
extern "C" {
#endif

int my_C_Function();

#ifdef __cplusplus
}
#endif

chorusmain.cpp

#include <...>
#include "chorusclient.h"

int main(int argc, char *arvg[])
{
  ...
  my_C_Function();
  ...
}

这是 gcc/g++ 构建和链接命令,

gcc -c -o chorusclient.o chorusclient.c -I/usr/include/json-c - 
    I/usr/include/openssl -I/usr/include -I. -I. -lssl -lcrypto -ljson-c

g++ -c -o chorusmain.o chorusmain.cpp -I/usr/share/qt4/mkspecs/linux-g++ -I. - 
    I/usr/include/qt4/QtCore -I/usr/include/qt4/QtGui -I/usr/include/qt4 - 
    I/usr/include -I. -I. -lQtGui -lQtCore -lpthread

g++ -o ChorusMain chorusclient.o chorusmain.o

我在前 2 个 gcc/g++ 构建中没有发现任何错误。我在 g++ 链接期间收到以下 undefined reference 错误,

debian@beaglebone:~/Projects/CloudClient$ ls
chorusclient.c  chorusclient.h  chorusclient.o  chorusmain.cpp  chorusmain.o
debian@beaglebone:~/Projects/CloudClient$ g++ -o ChorusMain chorusmain.o chorusclient.o
chorusclient.o:(.data+0x0): multiple definition of `svrport'
chorusmain.o:(.data+0x0): first defined here
chorusclient.o: In function `chorusServices':
chorusclient.c:(.text+0x408): undefined reference to `OPENSSL_init_crypto'
chorusclient.c:(.text+0x416): undefined reference to `OPENSSL_init_crypto'
chorusclient.c:(.text+0x41c): undefined reference to `OPENSSL_config'
chorusclient.o: In function `httpSend':
chorusclient.c:(.text+0x1c2c): undefined reference to `OPENSSL_init_ssl'
chorusclient.c:(.text+0x1c3a): undefined reference to `OPENSSL_init_ssl'
chorusclient.c:(.text+0x1c4c): undefined reference to `OPENSSL_init_ssl'
chorusclient.c:(.text+0x1c50): undefined reference to `TLSv1_2_client_method'
chorusclient.c:(.text+0x1c58): undefined reference to `SSL_CTX_new'
chorusclient.c:(.text+0x1c62): undefined reference to `SSL_CTX_set_options'
chorusclient.c:(.text+0x1c7e): undefined reference to `ERR_print_errors_fp'
chorusclient.c:(.text+0x1ca0): undefined reference to `SSL_new'
chorusclient.c:(.text+0x1caa): undefined reference to `SSL_set_fd'
chorusclient.c:(.text+0x1cb0): undefined reference to `SSL_connect'
chorusclient.c:(.text+0x1ce4): undefined reference to `SSL_get_peer_certificate'
chorusclient.c:(.text+0x1cec): undefined reference to `X509_get_subject_name'
chorusclient.c:(.text+0x1cfa): undefined reference to `X509_NAME_oneline'
chorusclient.c:(.text+0x1d1a): undefined reference to `X509_get_issuer_name'
chorusclient.c:(.text+0x1d28): undefined reference to `X509_NAME_oneline'
chorusclient.c:(.text+0x1d3e): undefined reference to `X509_free'
chorusclient.c:(.text+0x1d4a): undefined reference to `SSL_get_current_cipher'
chorusclient.c:(.text+0x1d52): undefined reference to `SSL_CIPHER_get_name'
chorusclient.c:(.text+0x1d80): undefined reference to `SSL_write'
chorusclient.c:(.text+0x1d90): undefined reference to `SSL_read'
chorusclient.c:(.text+0x1df2): undefined reference to `SSL_read'
chorusclient.c:(.text+0x1e8e): undefined reference to `SSL_free'
chorusclient.c:(.text+0x1e9a): undefined reference to `SSL_CTX_free'
chorusclient.o: In function `jsonBatteryData':
chorusclient.c:(.text+0x21a8): undefined reference to `json_object_new_object'
chorusclient.c:(.text+0x21b2): undefined reference to `json_object_new_string'
chorusclient.c:(.text+0x21be): undefined reference to `json_object_new_string'
chorusclient.c:(.text+0x21ca): undefined reference to `json_object_new_string'
chorusclient.c:(.text+0x21d8): undefined reference to `json_object_new_double'
chorusclient.c:(.text+0x21e6): undefined reference to `json_object_new_double'
chorusclient.c:(.text+0x21f4): undefined reference to `json_object_new_double'
chorusclient.c:(.text+0x2202): undefined reference to `json_object_new_double'
chorusclient.c:(.text+0x2210): undefined reference to `json_object_new_double'
chorusclient.c:(.text+0x2220): undefined reference to `json_object_object_add'
chorusclient.c:(.text+0x222e): undefined reference to `json_object_object_add'
chorusclient.c:(.text+0x223c): undefined reference to `json_object_object_add'
chorusclient.c:(.text+0x224a): undefined reference to `json_object_object_add'
chorusclient.c:(.text+0x2258): undefined reference to `json_object_object_add'
chorusclient.o:chorusclient.c:(.text+0x2266): more undefined references to `json_object_object_add' follow
chorusclient.o: In function `jsonBatteryData':
chorusclient.c:(.text+0x2292): undefined reference to `json_object_to_json_string'
chorusclient.o: In function `blockencrypt':
chorusclient.c:(.text+0x22de): undefined reference to `EVP_CIPHER_CTX_new'
chorusclient.c:(.text+0x22ee): undefined reference to `EVP_aes_256_cbc'
chorusclient.c:(.text+0x22fe): undefined reference to `EVP_EncryptInit_ex'
chorusclient.c:(.text+0x231a): undefined reference to `EVP_EncryptUpdate'
chorusclient.c:(.text+0x233c): undefined reference to `EVP_EncryptFinal_ex'
chorusclient.c:(.text+0x2354): undefined reference to `EVP_CIPHER_CTX_free'
chorusclient.o: In function `decrypt':
chorusclient.c:(.text+0x2370): undefined reference to `EVP_CIPHER_CTX_new'
chorusclient.c:(.text+0x2380): undefined reference to `EVP_aes_256_cbc'
chorusclient.c:(.text+0x2390): undefined reference to `EVP_DecryptInit_ex'
chorusclient.c:(.text+0x23ac): undefined reference to `EVP_DecryptUpdate'
chorusclient.c:(.text+0x23ce): undefined reference to `EVP_DecryptFinal_ex'
chorusclient.c:(.text+0x23e6): undefined reference to `EVP_CIPHER_CTX_free'
chorusclient.o: In function `handleErrors':
chorusclient.c:(.text+0x2404): undefined reference to `ERR_print_errors_fp'
collect2: error: ld returned 1 exit status
debian@beaglebone:~/Projects/CloudClient$

当我创建一个调用 chorusclient.c 的简单 C 程序并构建一个链接时,我没有收到任何 undefined reference 错误,并且该程序可以正常运行。有什么想法吗?

最佳答案

你说你有:

gcc -c -o chorusclient.o chorusclient.c -I/usr/include/json-c \
    -I/usr/include/openssl -I/usr/include -I. -I. -lssl -lcrypto -ljson-c
g++ -c -o chorusmain.o chorusmain.cpp -I/usr/share/qt4/mkspecs/linux-g++ -I. \
    -I/usr/include/qt4/QtCore -I/usr/include/qt4/QtGui -I/usr/include/qt4 \
    -I/usr/include -I. -I. -lQtGui -lQtCore -lpthread
g++ -o ChorusMain chorusclient.o chorusmain.o

编译创建目标文件时列出库;尝试链接可执行文件时不列出库。这些库与创建目标文件无关——它们在链接可执行文件时至关重要。

因此你需要更多类似的东西:

gcc -Wall -Werror -c -o chorusclient.o chorusclient.c -I/usr/include/json-c \
    -I/usr/include/openssl -I/usr/include -I. 
g++ -Wall -Werror -c -o chorusmain.o chorusmain.cpp -I/usr/share/qt4/mkspecs/linux-g++ \
    -I/usr/include/qt4/QtCore -I/usr/include/qt4/QtGui -I/usr/include/qt4 \
    -I/usr/include -I.
g++ -o ChorusMain chorusclient.o chorusmain.o -lQtGui -lQtCore -lpthread -lssl -lcrypto -ljson-c

我不相信你需要 -I/usr/include , 你通常不需要 -I. (除非你坚持使用 #include <chorusclient.h> 而不是 #include "chorusclient.h" ),而且你绝对不需要 -I.重复。

您有可能需要对链接中的库列表重新排序——最好将 -lpthread接近尾声。

我添加了 -Wall -Werror选项,因为你不应该尝试运行你的代码,除非它至少在那些选项下干净地编译(我会在我的代码中添加更多 - 这足以说明问题)。

我有点惊讶你需要 -I/usr/include/openssl线;你应该使用 #include <openssl/ssl.h>等(对于 OpenSSL header )这样您就不需要 -I/usr/include/openssl命令行选项?同样,也许,对于 Qt 和 JSON-C header ?

另一个问题是你有一个对象——可能是一个全局变量——叫做svrport。在 chorusclient.o 中定义和 chorusmain.o .这意味着通过某种机制,您可以在两个源文件中定义(而不是声明)相同的对象。这可能是因为它是在 header 中定义的而不是声明的(但代码中未显示)。或者可能是因为您在两个源文件中都定义了它。如果它需要是一个全局变量,它应该在头文件中声明并在其中一个源文件中定义(可能是 chorusclient.c 因为 chorusclient.h 头文件正在声明它)而不是在另一个源文件中定义。

关于C++ 程序调用 C 函数在 g++ 链接期间获取 undefined reference ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49470444/

相关文章:

c++ - 修复此编译错误 : identifier "and" is a special operator name in C++ [-Werror=c++-compat]?

gcc - 在 openshift 上使用 pip 安装 lxml 失败

将十六进制字符串转换为十六进制字节,无需在没有库的c中使用

创建/分配 p_threads 数组

c - 如何将带有可变参数的函数声明为 stdcall?

c++ - 尝试在 C++ 中返回字符串数组时遇到错误

c - 理解数组和函数

c++ - 如何在 Windows 中获取用于特定显示器的显示适配器?

c++ - 如何在 ACE 中获取本地时间而不是 UTC 时间?

c++ - 在 C++ 中替换字符串中的特定字符