下面我們自己在 Linux 下做一個(gè)動(dòng)態(tài)庫(.so 文件 - Shared Object),然在用 Go 來使用它。本文所用的操作系統(tǒng)為 Ubuntu18.04, 以 gcc 作為編譯器。
1.實(shí)現(xiàn)頭文件,聲明文件中函數(shù)。這里創(chuàng)建一個(gè)add.h文件。
1
2
3
4
5
6
|
#ifndef __ADD_H__ #define __ADD_H__ char * Add( char * src, int n); #endif |
2.實(shí)現(xiàn)add主體函數(shù)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.用命令生成動(dòng)態(tài)庫,在linux下文件名稱是libadd.so
1
|
gcc -fPIC -shared -o lib/libadd.so include/add.c |
會(huì)在當(dāng)前目錄下生成 libadd.so 文件, 在 Linux 下可用 nm -D libadd.so 查看其中的方法
4.編寫一個(gè)庫來測(cè)試一下
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; } |
鏈接動(dòng)態(tài)庫生成可執(zhí)行文件
1
|
gcc include/test.c -L lib/ -ladd -o test |
- -L .表示搜索要鏈接的庫文件時(shí)包含當(dāng)前目錄
- -ladd 表示要鏈接動(dòng)態(tài)庫 libadd.so (備注:默認(rèn)lib + xxx + .so ,中間的xxx就是庫名)
- -o test 生成可執(zhí)行文件 test
錯(cuò)誤:運(yùn)行出錯(cuò)的情況
# 運(yùn)行 ./test,出錯(cuò)
./test: error while loading shared libraries: libadd.so: cannot open shared object file: No such file or directory
出現(xiàn)以上的錯(cuò)誤。有可能是環(huán)境變量沒弄好導(dǎo)致的,找不到動(dòng)態(tài)庫 libadd.so, Linux 通過 ldconfig 的指示在某些目錄中(如 /lib, /user/lib) 搜索動(dòng)態(tài)庫。更簡(jiǎn)單的辦法是用 LD_LIBRARY_PATH 環(huán)境變量。解決辦法,
1
2
|
$ LD_LIBRARY_PATH=lib/ . /test giter8 |
至此,動(dòng)態(tài)庫 libadd.so 準(zhǔn)備好了,并且用 test 驗(yàn)證了它是可用的,接下來就在 Go 語言中使用該動(dòng)態(tài)庫的函數(shù)。
5.golang調(diào)用c動(dòng)態(tài)庫
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 /* // 頭文件的位置,相對(duì)于源文件是當(dāng)前目錄,所以是 .,頭文件在多個(gè)目錄時(shí)寫多個(gè) #cgo CFLAGS: ... #cgo CFLAGS: -I./include // 從哪里加載動(dòng)態(tài)庫,位置與文件名,-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 編譯器從哪里引入頭文件與加載動(dòng)態(tài)庫. 本例中 *.h 和 *.go 文件在同一個(gè)目錄的情況下, #cgo CFLAGS: -I. 可不寫。
CFLAGS: -I 和 LDFLAGS: -L 都是相對(duì)于源文件 main.go 的位置
1
2
|
./demo1 run c: go2021 |
成功調(diào)用 C 實(shí)現(xiàn)的 add 函數(shù)
下面列出一些問題
import "C" 要緊挨著 /*...*/ 注釋塊,如果寫成
1
2
3
4
5
|
/* #cgo ... */ import "C" |
出現(xiàn)下面的報(bào)錯(cuò)信息
# demo1
./main.go:15:10: could not determine kind of name for C.Add
import "C" 要獨(dú)占一行, 試圖同時(shí)引入其他的庫,如 import ("C"; "fmt") 也會(huì)報(bào)上面同樣的錯(cuò)誤
加載不到頭文件的錯(cuò)誤很明顯,#include "add.h" 時(shí)會(huì)告訴你該文件不存在,如果沒有加載到正確的頭文件調(diào)用 C.Add() 函數(shù)時(shí)就會(huì)報(bào)錯(cuò)
# demo1
./main.go:15:10: could not determine kind of name for C.Add
還有一個(gè)關(guān)鍵是能否加載到動(dòng)態(tài)庫 libadd.so, 參考了網(wǎng)上一些例子,如果把第五行改為
1
|
cgo LDFLAGS: -L./lib -ladd |
編譯不會(huì)報(bào)錯(cuò),執(zhí)行時(shí)會(huì)出錯(cuò)。
./demo1: error while loading shared libraries: libadd.so: cannot open shared object file: No such file or directory
但如果設(shè)置了環(huán)境變量 LD_LIBRARY_PATH=/home/vagrant/testgo/lib 也能讓它跑起來
1
|
LD_LIBRARY_PATH=lib/ ./demo1 |
到此這篇關(guān)于golang 調(diào)用c語言動(dòng)態(tài)庫方式實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)golang 調(diào)用c語言動(dòng)態(tài)庫內(nèi)容請(qǐng)搜索服務(wù)器之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持服務(wù)器之家!
原文鏈接:https://juejin.cn/post/7047405294107754533