下面我們自己在 Linux 下做一個動態庫(.so 文件 - Shared Object),然在用 Go 來使用它。本文所用的操作系統為 Ubuntu18.04, 以 gcc 作為編譯器。
1.實現頭文件,聲明文件中函數。這里創建一個add.h文件。
1
2
3
4
5
6
|
#ifndef __ADD_H__ #define __ADD_H__ char * Add( char * src, int n); #endif |
2.實現add主體函數add.c
1
2
3
4
5
6
7
8
9
10
11
12
13
|
#include <string.h> #include <stdio.h> #include <stdlib.h> char * Add( char * src, int n) { char str[20]; sprintf (str, "%d" , n); char *result = malloc ( strlen (src)+ strlen (str)+1); strcpy (result, src); strcat (result, str); return result; } |
3.用命令生成動態庫,在linux下文件名稱是libadd.so
1
|
gcc -fPIC -shared -o lib/libadd.so include/add.c |
會在當前目錄下生成 libadd.so 文件, 在 Linux 下可用 nm -D libadd.so 查看其中的方法
4.編寫一個庫來測試一下
1
2
3
4
5
6
7
8
9
|
#include <stdio.h> #include "add.h" int main( int argc, char *argv[]) { char * aa = "giter" ; printf ( "%s\n" , Add(aa, 8)); return 0; } |
鏈接動態庫生成可執行文件
1
|
gcc include/test.c -L lib/ -ladd -o test |
- -L .表示搜索要鏈接的庫文件時包含當前目錄
- -ladd 表示要鏈接動態庫 libadd.so (備注:默認lib + xxx + .so ,中間的xxx就是庫名)
- -o test 生成可執行文件 test
錯誤:運行出錯的情況
# 運行 ./test,出錯
./test: error while loading shared libraries: libadd.so: cannot open shared object file: No such file or directory
出現以上的錯誤。有可能是環境變量沒弄好導致的,找不到動態庫 libadd.so, Linux 通過 ldconfig 的指示在某些目錄中(如 /lib, /user/lib) 搜索動態庫。更簡單的辦法是用 LD_LIBRARY_PATH 環境變量。解決辦法,
1
2
|
$ LD_LIBRARY_PATH=lib/ . /test giter8 |
至此,動態庫 libadd.so 準備好了,并且用 test 驗證了它是可用的,接下來就在 Go 語言中使用該動態庫的函數。
5.golang調用c動態庫
1
2
3
4
5
6
7
8
|
demo1 ├── include │ └── add.c │ └── add.h │ └── test.c ├── lib │ └── libadd.so └── main.go |
main.go 的代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
package main /* // 頭文件的位置,相對于源文件是當前目錄,所以是 .,頭文件在多個目錄時寫多個 #cgo CFLAGS: ... #cgo CFLAGS: -I./include // 從哪里加載動態庫,位置與文件名,-ladd 加載 libadd.so 文件 #cgo LDFLAGS: -L./lib -ladd -Wl,-rpath,lib #include "add.h" */ import "C" import "fmt" func main() { val := C.Add(C.CString("go"), 2021) fmt.Println("run c: ", C.GoString(val)) } |
通過注釋代碼來告訴 Go 編譯器從哪里引入頭文件與加載動態庫. 本例中 *.h 和 *.go 文件在同一個目錄的情況下, #cgo CFLAGS: -I. 可不寫。
CFLAGS: -I 和 LDFLAGS: -L 都是相對于源文件 main.go 的位置
1
2
|
./demo1 run c: go2021 |
成功調用 C 實現的 add 函數
下面列出一些問題
import "C" 要緊挨著 /*...*/ 注釋塊,如果寫成
1
2
3
4
5
|
/* #cgo ... */ import "C" |
出現下面的報錯信息
# demo1
./main.go:15:10: could not determine kind of name for C.Add
import "C" 要獨占一行, 試圖同時引入其他的庫,如 import ("C"; "fmt") 也會報上面同樣的錯誤
加載不到頭文件的錯誤很明顯,#include "add.h" 時會告訴你該文件不存在,如果沒有加載到正確的頭文件調用 C.Add() 函數時就會報錯
# demo1
./main.go:15:10: could not determine kind of name for C.Add
還有一個關鍵是能否加載到動態庫 libadd.so, 參考了網上一些例子,如果把第五行改為
1
|
cgo LDFLAGS: -L./lib -ladd |
編譯不會報錯,執行時會出錯。
./demo1: error while loading shared libraries: libadd.so: cannot open shared object file: No such file or directory
但如果設置了環境變量 LD_LIBRARY_PATH=/home/vagrant/testgo/lib 也能讓它跑起來
1
|
LD_LIBRARY_PATH=lib/ ./demo1 |
到此這篇關于golang 調用c語言動態庫方式實現的文章就介紹到這了,更多相關golang 調用c語言動態庫內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家!
原文鏈接:https://juejin.cn/post/7047405294107754533