Go語言中CGO的使用實踐

       部門產品業務功能采用Golang開發,但是有些功能是用c寫的,比如說net-snmp,bfd協議等等,像這些如果使用GO語言重編的話,既有實現的復雜度也需要相當長的時間,好在GO語言提供瞭CGO機制,使得能夠在go代碼中直接調用C的庫函數,大大提高瞭效率,減少瞭重復開發工作,此外還支持在C語言中調用GO函數,這一點還是蠻強大的。

1. Go語言調用C函數例子:

package main
 
//
// 引用的C頭文件需要在註釋中聲明,緊接著註釋需要有import "C",且這一行和註釋之間不能有空格
//
 
/*
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
void myprint(char* s) {
	printf("%s\n", s);
}
*/
import "C"
 
 
import (
	"fmt"
	"unsafe"
)
 
func main() {
	//使用C.CString創建的字符串需要手動釋放。
	cs := C.CString("Hello World\n")
	C.myprint(cs)
	C.free(unsafe.Pointer(cs))
	fmt.Println("call C.sleep for 3s")
	C.sleep(3)
	return
}

運行:

2. Go語言調用C庫函數:

hello.c

#include <stdio.h>
void hello()
{
    printf("hello world\n"); 
}

hello.h

#ifndef HELLO_H
#define HELLO_H
 
void hello(void);
#endif

編譯:

gcc -c hello.c
ar -cru libhello.a hello.o
package main
 
//使用#cgo定義庫路徑
 
 
/*
#cgo CFLAGS: -I .
#cgo LDFLAGS: -L . -lhello
#include "hello.h"
*/
import "C"
 
func main() {
	C.hello()
}

運行:

3. Go語言導出函數給C語言使用:

main.go

package main
 
//
//#include <stdio.h>
//int add(int a, int b);
//
import "C"
 
import (
	"fmt"
)
 
//當使用export的時候,在同一個文件中就不能再定義其它的c函數瞭,不然會報錯。
//使用export導出函數給c語言調用。
 
//export GoAdd
func GoAdd(a, b int) int {
	return a + b
}
 
func main() {
	a := C.add(1, 2)
	fmt.Printf("C.add(1,2) return %d\n", a)
}

cfunc.go

package main
 
//
//int GoAdd(int a, int b); 
//
//int add(int a, int b)
//{
//	  return GoAdd(a,b);
//}
//
import "C"

運行:

4. Go語言導出函數指針給c語言使用:

還有一種使用方式,這種是我使用比較多的。就是傳遞函數指針,因為GO函數無法取址,因此需要寫個中間函數做個轉換操作,例子如下:

clibrary.c

#include <stdio.h>
 
#include "clibrary.h"
 
//參數是函數指針
void some_c_func(callback_fcn callback)
{
	int arg = 2;
	printf("C.some_c_func(): calling callback with arg = %d\n", arg);
	int response = callback(2);
	printf("C.some_c_func(): callback responded with %d\n", response);
}

 clibrary.h

#ifndef CLIBRARY_H
#define CLIBRARY_H
//定義函數指針
typedef int (*callback_fcn)(int);
void some_c_func(callback_fcn);
#endif

Go code:

package main
 
/*
#cgo CFLAGS: -I .
#cgo LDFLAGS: -L . -lclibrary
#include "clibrary.h"
int callOnMeGo_cgo(int in); // 聲明
*/
import "C"
 
import (
	"fmt"
	"unsafe"
)
 
//export callOnMeGo
func callOnMeGo(in int) int {
	return in + 1
}
 
func main() {
	fmt.Printf("Go.main(): calling C function with callback to us\n")
 
    //使用unsafe.Pointer轉換
	C.some_c_func((C.callback_fcn)(unsafe.Pointer(C.callOnMeGo_cgo)))
}

中間函數:

package main
 
/*
 
#include <stdio.h>
int callOnMeGo(int);
 
// The gateway function
int callOnMeGo_cgo(int in)
{
	printf("C.callOnMeGo_cgo(): called with arg = %d\n", in);
    //調用GO函數
	return callOnMeGo(in);
}
*/
import "C"

運行:

開發註意事項:

1. 在註釋和import”C”之間不能有空行

2. 使用C.CString函數轉換GoString為CString時要手動釋放該字符串。

3. CGO不支持使用變參的函數,例如printf,如果要使用的話,可以寫個包裹函數m’yprintf,使用傳參的方式調用。

4. Go支持使用//export導出函數給C使用,但是有一點需要註意就是不能在export導出的同一個文件裡定義c函數,不然會出現

multiple definition of “xxx”編譯錯誤,如果函數非常tiny的話,還有一個方法是使用static inline 來聲明該函數,如下:

package gocallback
 
import (
	"fmt"
	"sync"
)
 
/*
extern void go_callback_int(int foo, int p1);
// normally you will have to define function or variables
// in another separate C file to avoid the multiple definition
// errors, however, using "static inline" is a nice workaround
// for simple functions like this one.
static inline void CallMyFunction(int foo) {
	go_callback_int(foo, 5);
}
*/
import "C"
 

參考資料:

1. https://github.com/golang/go/wiki/cgo

到此這篇關於Go語言中CGO的使用實踐的文章就介紹到這瞭,更多相關Go語言使用CGO內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: