C++ 将静态库添加到 CMake 项目导致未定义的引用

C++ adding static library to CMake project results in undefined references

我已经设置了一个简单的 C++ 项目,我正在尝试使用 CMake 进行编译,并且我正在尝试对其静态 link libcurl。在我的 main.cpp 中,我有标准的 curl_easy_init 示例代码,我只是想发送一个简单的请求来确保库正常工作。但是,我在每次 cURL 函数调用时都会收到未定义的引用错误。在我的 CMakeLists.txt 中,我正在检查当前平台,然后 link 从那个特定的 if 语句中获取正确的库。我的错误日志 main.cpp 和 CMakeLists.txt 如下。 我唯一的问题是如何将我的 libcurl 正确地设置为 link,因为我认为现在不是这样,因为我的二进制文件只有 72KB,而我实际上可以通过编译删除 cURL具体代码但保留头文件。 除了明显的,我尝试使用的一件事是 --enable-stdcall-fixup linker 标志,它根本不起作用。

错误日志:

C:\Users\bfsco\AppData\Local\JetBrains\Toolbox\apps\CLion\ch-03.13906.4\bin\cmake\bin\cmake.exe --build D:\Programming\CPP\cppackage\cmake-build-debug --target all -- -j 4
[ 50%] Linking CXX executable cppackage.exe
CMakeFiles\cppackage.dir/objects.a(main.cpp.obj): In function `main':
D:/Programming/CPP/cppackage/main.cpp:6: undefined reference to `curl_easy_init'
D:/Programming/CPP/cppackage/main.cpp:10: undefined reference to `curl_easy_setopt'
D:/Programming/CPP/cppackage/main.cpp:11: undefined reference to `curl_easy_perform'
D:/Programming/CPP/cppackage/main.cpp:13: undefined reference to `curl_easy_cleanup'
collect2.exe: error: ld returned 1 exit status
mingw32-make.exe[2]: *** [cppackage.exe] Error 1
CMakeFiles\cppackage.dir\build.make:98: recipe for target 'cppackage.exe' failed
mingw32-make.exe[1]: *** [CMakeFiles/cppackage.dir/all] Error 2
CMakeFiles\Makefile2:66: recipe for target 'CMakeFiles/cppackage.dir/all' failed
mingw32-make.exe: *** [all] Error 2
Makefile:82: recipe for target 'all' failed

main.cpp:

#include <iostream>

#include "include/curl/curl.h"

int main() {
    CURL *curl = curl_easy_init();

    if(curl) {
        CURLcode res;
        curl_easy_setopt(curl, CURLOPT_URL, "http://example.com");
        res = curl_easy_perform(curl);
        std::cout << res << std::endl;
        curl_easy_cleanup(curl);
    }

    std::cout << "Hello, World!" << std::endl;
    return 0;
}

CMakeLists.txt:

cmake_minimum_required(VERSION 3.6)
project(cppackage)

set(CMAKE_CXX_STANDARD 14)
set(SOURCE_FILES main.cpp)

include_directories(${CMAKE_SOURCE_DIR}/include/curl)

add_executable(${PROJECT_NAME} ${SOURCE_FILES})

if(WIN32)
    add_definitions(-DCURL_STATICLIB)
    target_link_libraries(cppackage ${CMAKE_SOURCE_DIR}/thirdparty/openssl/openssl.lib ${CMAKE_SOURCE_DIR}/thirdparty/curl/libcurl.lib)
else()
    target_link_libraries(cppackage ${CMAKE_SOURCE_DIR}/thirdparty/curl/libcurl.a)
endif()

就我个人而言,我总是对我传递相对路径的库使用命令 "set",然后将该变量放入 target_link_libraries。

您似乎正在尝试 link 从 Visual Studio 构建的库到使用 MinGW 构建的应用程序。 This is normally not possible。您应该尝试使用 MinGW 从源代码构建 libcurl。


原回答:

尝试将 2 个库版本分开放在 2 个不同的目录 /thirdparty/curl_win32 和 /thirdparty/curl_2 中,然后将您的代码更改为:

if(WIN32)
    add_definitions(-DCURL_STATICLIB)
    link_directories("${CMAKE_SOURCE_DIR}/thirdparty/openssl/" "${CMAKE_SOURCE_DIR}/thirdparty/"curl_win32)
else()
    link_directories("${CMAKE_SOURCE_DIR}/thirdparty/curl2")
endif()

add_executable(${PROJECT_NAME} ${SOURCE_FILES})

if(WIN32)
    target_link_libraries(${PROJECT_NAME} openssl curl)
else()
    target_link_libraries(${PROJECT_NAME} curl)
endif()

请注意,显然 link_directories 命令必须位于 add_executable 命令之前。参见