C++ Cmake的構(gòu)建靜態(tài)庫和動態(tài)庫詳解
靜態(tài)庫和動態(tài)庫的區(qū)別
1、靜態(tài)庫的擴展名一般為".a"或者".lib";動態(tài)庫的擴展名一般為".so"或者".dll"。
2、靜態(tài)庫在編譯時會直接整合到目標程序中,編譯成功的可執(zhí)行文件可以獨立運行
3、動態(tài)庫在編譯時不會放到連接的目標程序中,即可執(zhí)行文件無法單獨運行。
構(gòu)建示例
創(chuàng)建新項目cmake02,包含目錄如下
?hello.h中的內(nèi)容
#ifndef HELLO_H #define HELLO_H void HelloFunc(); #endif
hello.cpp中的內(nèi)容
#include "hello.h" #include <iostream> using namespace std; void HelloFunc() { cout<<"Hello World"<<endl; }
項目中CMakeLists.txt內(nèi)容
PROJECT(HELLO) ADD_SUBDIRECTORY(lib bin)
lib中CMakeLists.txt內(nèi)容
SET(LIBHELLO_SRC hello.cpp) ADD_LIBRARY(hello SHARED ${LIBHELLO_SRC})
ADD_LIBRARY
ADD_LIBRARY(hello SHARED ${LIBHELLO_SRC})
hello:就是正常的庫名,生成的名字前面會加上lib,最終產(chǎn)生的文件是libhello.so
SHARED,動態(tài)庫STATIC,靜態(tài)庫
${LIBHELLO_SRC}:源文件
同時構(gòu)建靜態(tài)和動態(tài)庫
//如果用這種方式,只會構(gòu)建一個動態(tài)庫,不會構(gòu)建出靜態(tài)庫,雖然靜態(tài)庫的后綴是.a
ADD_LIBRARY(hello SHARED ${LIBHELLO_SRC})
ADD_LIBRARY(hello STATIC ${LIBHELLO_SRC})
//修改靜態(tài)庫的名字,這樣是可以的,但是我們往往希望他們的名字是相同的,只是后綴不同而已
ADD_LIBRARY(hello SHARED ${LIBHELLO_SRC})
ADD_LIBRARY(hello_static STATIC ${LIBHELLO_SRC})
SET_TARGET_PROPERTIES
這條指令可以用來設(shè)置輸出的名稱,對于動態(tài)庫,還可以用來指定動態(tài)庫版本和API版本
同時構(gòu)建靜態(tài)和動態(tài)庫示例
SET(LIBHELLO_SRC hello.cpp) ADD_LIBRARY(hello_static STATIC ${LIBHELLO_SRC}) SET_TARGET_PROPERTIES(hello_static PROPERTIES OUTPUT_NAME "hello") SET_TARGET_PROPERTIES(hello_static PROPERTIES CLEAN_DIRECT_OUTPUT$ ADD_LIBRARY(hello SHARED ${LIBHELLO_SRC}) SET_TARGET_PROPERTIES(hello PROPERTIES OUTPUT_NAME "hello") SET_TARGET_PROPERTIES(hello PROPERTIES CLEAN_DIRECT_OUTPUT 1)
動態(tài)庫的版本號
一般動態(tài)庫都會有一個版本號的關(guān)聯(lián)
libhello.so.1.2
libhello.so->libhello.so.1
libhello.so.1->libhello.so.1.2
CMakeList.txt插入如下
SET_TARGET_PROPERTIES(hello PROPERTIES VERSION 1.2 SOVERSION 1)
VERSION指代動態(tài)庫版本,SOVERSION指代API版本
安裝共享庫和頭文件
本例中我們將hello的共享庫安裝到<prefix>/lib目錄
將hello.h安裝到<prefix>/include/hello目錄
//文件放到該目錄下 INSTALL(FILES hello.h DESTINATION include/hello) //二進制,靜態(tài)庫,動態(tài)庫安裝都是用TARGETS //ARCHIVE特指靜態(tài)庫,LIBRATY特指動態(tài)庫,RUNTIME特指可執(zhí)行目標二進制 INSTALL(TARGETS hello hello_static LIBRARY DESTINATION lib ARCHIVE DESTINATION lib)
注意:安裝的時候,指定一下路徑,放到系統(tǒng)下
cmake -DCMAKE_INSTALL_PERFIX=/usr ..
使用外部共享庫和頭文件
新建cmake03目錄來使用外部共享庫和頭文件
main.cpp內(nèi)容
#include <hello.h> int main() { HelloFunc(); }
解決 :make后頭文件找不到的問題
PS:include
這樣是可以的
關(guān)鍵字:INCLUDE_DIRECTORIES
這條指令可以用來向工程添加多個特定的頭文件搜索路徑,路徑之間用空格分割
在CMakeLists.txt中加入頭文件搜索路徑
INCLUDE_DIRECTORIES(/usr/local/include/hello)
解決:找到引用的函數(shù)問題
關(guān)鍵字:LINK_DIRCTORIES
添加非標準的共享庫搜索路徑
指定第三方庫鎖在路徑,LINK_DIRECTORIES(/home/myproject/libs)
關(guān)鍵字:TARGET_LINK_LIBRARIES
添加需要鏈接的共享庫
TARGET_LINK_LIBRARIES
的時候,只需要給出動態(tài)鏈接庫的名字就行了。
在CMakeLists.txt
中插入連接共享庫,主要要插在executable的后面
TARGET_LINK_LIBRARIES(hello libhello.so)
cmake后make
執(zhí)行bin目錄下的./hello報錯
解決:mv /usr/local/lib/libhello.so /usr/lib64/
查看main的鏈接情況
鏈接靜態(tài)庫
TARGET_LINK_LIBRARIES(hello libhello.a)
特殊的環(huán)境變量CMAKE_INCLUDE_PATH和CMAKE_LIBRARY_PATH
注意:這兩個是環(huán)境變量而不是cmake變量,可以在linux的bash中進行設(shè)置
在上面例子中使用了絕對路徑INCLUDE_DIRECTORIES(/usr/include/hello)
來指明include路徑的位置
我們還可以使用環(huán)境變量export CMAKE_INCLUDE_PATH=/usr/local/include/hello
生產(chǎn)debug版本的方法:
cmake .. -DCMAKE_BUILD_TYPE=debug
總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
c++連接mysql數(shù)據(jù)庫的兩種方法(ADO連接和mysql api連接)
現(xiàn)在正做一個接口,通過不同的連接字符串操作不同的數(shù)據(jù)庫。要用到mysql數(shù)據(jù)庫,C++連接mysql有2種方法:利用ADO連接、利用mysql自己的api函數(shù)進行連接,下面看看如何用吧2013-12-12C++實現(xiàn)LeetCode(121.買賣股票的最佳時間)
這篇文章主要介紹了C++實現(xiàn)LeetCode(121.買賣股票的最佳時間),本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下2021-07-07Qt QChart 創(chuàng)建圖表的實現(xiàn)方法
這篇文章主要介紹了Qt QChart 創(chuàng)建圖表的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-12-12C++計算任意權(quán)值的單源最短路徑(Bellman-Ford)
這篇文章主要為大家詳細介紹了C++計算任意權(quán)值的單源最短路徑,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-04-04