c/c++靜態庫之間相互調用的實戰案例

本文主要介紹c語言寫的項目如何調用c++實現的庫和c++如何調用c語言實現的庫

一、c++項目如何調用c的庫(以靜態庫為例)

⛄️示例:建立一個項目Stack_c.lib,將該項目中的棧的實現的代碼打包為一個靜態庫,再建立另一個c++的項目,並調用打包好的c實現的靜態庫裡的棧的相關函數

1.將棧的源文件和頭文件拷貝一份到Stack_c.lib這個項目的路徑下

2.打包成靜態庫

⛄️應用好之後就點擊生成解決方案 ,之後就會生成一個Stack_c.lib的靜態庫瞭 (與項目名重名瞭,問題不大 不要在意)

生成後我們可以點開項目的所在目錄下 找到debug目錄 可以看到裡面多瞭一個Stack_c.lib的靜態庫文件 表示打包成功!

⛄️至此,靜態庫就打包好瞭,當然是編譯器幫我們打包的,後續會更新如何自己親手打包一個庫(動靜態庫的知識)!

3.建立一個c++項目(TestStacklib) 引入c項目中的頭文件

🎸註意:這裡的引入頭文件的路徑可以是相對路徑 其中是代表的上級目錄 隻需找到自己創建的動態庫的頭文件的路徑並在新建的c++項目中包含即可

本文中的test.cpp是一段關於括號匹配的算法代碼 可以用建的靜態庫裡的棧的特性來解決

bool isValid(const char* s) {
    const char* cur = s;
    stack stack;
    StackInit(&stack);
    while (*cur != '\0')
    {
        if (*cur == '(' || *cur == '{' || *cur == '[')//如果是左括號就進棧
        {
            StackPush(&stack, *cur);
        }
        else
        {
            if (StackEmpty(&stack))//考慮到開始是右括號,那麼就是棧為空,就不可能有效,直接返回FALSE
            {
                return false;
            }
            char top = StackTop(&stack);
            if (*cur == ')' && top == '(' || *cur == '}' && top == '{' || *cur == ']' && top == '[')
            {
                StackPop(&stack);
            }
            else {
                return false;
            }
        }
        cur++;
    }
    if (StackEmpty(&stack))//有可能隻有一個左括號,進棧就沒瞭,有效還有判斷棧是否為空,為空才是有效括號
        return true;
    else
        return false;

}

int main()
{
    cout << isValid("{{))") << endl;
    cout << isValid("({})") << endl;
    return 0;
}

4.設置附加庫和附屬關系

將靜態庫中的debug目錄的路徑復制到附加庫目錄中

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-uwjKkcKk-1657719900201)(C:\Users\華哥\AppData\Roaming\Typora\typora-user-images\image-20220713195455905.jpg)]

到這裡附加庫目錄的操作就完成瞭 接下來就是在輸入中設置依賴項瞭

🎸到此準備工作就完成瞭 接下來有兩種方法可以實現c++項目調用c靜態庫

1.將stack_c.lib中的stack.c的後綴改成stack.cpp 即可
2.在c++項目中使用extern “C” 表示編譯的時候按照c的規則編譯鏈接(主要就是函數名修飾的規則用c的規則)因為c++是兼容c的所以c++的編譯器可以這麼幹 反過來c是不可以兼容c++的 所以反過來是行不通的。

接著往下看:

🎸第一種方法:將Stack_c.lib中的stack.c 改名為 stack.cpp(使得其編譯鏈接的時候是按照c++的規則,這樣c++項目調c++規則生成的庫就可以理所當然的調動瞭,但是這樣的方式似乎很不著調,本來是c庫,但是硬是把裡面的源文件的後綴改成瞭cpp 不太好)

然後運行那段代碼就可以成功運行瞭 證明調用靜態庫成功!

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-Og8Njbtu-1657719900202)(C:\Users\華哥\AppData\Roaming\Typora\typora-user-images\image-20220713201637750.jpg)]

🎸第二種:利用extern "C"改變c++項目的編譯鏈接過程的規則由c++的規則變成c的規則,這樣再調用c的靜態庫也就可以實現瞭,而且不用像第一種方法一樣去改源文件的後綴(強盜行為)

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-BYdV4ZeG-1657719900202)(C:\Users\華哥\AppData\Roaming\Typora\typora-user-images\image-20220713202155357.jpg)]

格式:

extern "C"
{
	#include"庫的頭文件路徑"
}

之後就可以運行成功 ,表示鏈接成功瞭!就不貼圖瞭 與上面的圖一樣

二、c項目如何調用c++的庫(靜態庫為例)

還是那句話,c++兼容c 要用c項目調用c++庫 那麼就隻能是讓c++ 的庫編譯的時候用c的規則來,那麼該咋弄呢?
還是extern “C”

1.建立c++項目(Stack_cpp) 並且將棧的源文件和頭文件包含在該目錄 將其打包為cpp的靜態庫

🎄右擊項目名稱 點擊屬性 再更改配置類型為靜態庫類型

🎄之後點擊到項目的路徑 進入debug目錄 看到生成瞭一個Stack_cpp.lib 就說明生成打包靜態庫成功瞭!

接下來就是用extern "C"結合條件編譯來使得c++項目中的代碼按照c的規則來編譯鏈接 但是c項目要包含c++靜態庫的頭文件 那麼就會在預處理的時候頭文件展開 那麼c項目中就也會有 extern “C” 這是不可以的 隻有c++才可以識別extern “C” c是識別不瞭的 會報錯!!!

那麼如何解決 ? 條件編譯這個時候就派上大用處瞭

🎄c++ 的文件中天然包含 __cplusplus 標識符 而c是沒有的 可以以此為入口點 通過條件編譯使得extern “C” 在c++項目中展開 但是在c項目中不展開 就將問題解決瞭

下面還有一個簡化版的條件編譯

//簡化版
#ifdef __cplusplus
extern "C"
{
#endif

 void StackInit(stack* pst);
 void StackDestory(stack* pst);
 void StackPush(stack* pst,STDataType x);
 void StackPop(stack* pst);
 bool StackEmpty(stack* pst);
 int StackSize(stack* pst);
 STDataType StackTop(stack* pst);

#ifdef __cplusplus
}
#endif

之後就是 建立c項目 然後 建立test.c 拷貝那段關於括號匹配的代碼到其中 包含靜態庫的頭文件 添加打包的cpp動態庫到附加庫目錄 設置依賴項 通過調用cpp靜態庫裡的棧的函數解決 運行成功就說明c項目鏈接c++的靜態庫成功

2.建立c項目

3.包含靜態庫的頭文件

4 .添加打包的cpp動態庫到附加庫目錄 設置依賴項

然後就大功告成瞭 ,如果沒有差錯就可以直接運行成功瞭。

🎄這裡需要註意的是 要記得使用靜態庫前完成瞭準備工作後一定要生成解決方案後再在來調用庫!

總結

到此這篇關於c/c++靜態庫之間相互調用的文章就介紹到這瞭,更多相關c/c++靜態庫間相互調用內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: