golang的动态库(so)生成与使用

作者: adm 分类: go 发布时间: 2022-04-25

go语言有很多的库,在开发c语言的时候为了快速完成某些功能,可以直接通过go语言的库进行调用.那么我们将go语言开发的程序编译成so动态库文件就可以给c或者c++程序调用.

go文件编写
如果想要导出C能使用的动态库.我们需要将函数的名称编写为大写.

并且还需要添加导出注释(重要) //export HelloWorld 这个注释必须写,并且//后面不要有空格.

package main
 
/*
#include 
*/
import "C"
import "fmt"
 
//export HelloWorld
func HelloWorld(str *C.char) *C.char{
	fmt.Printf(C.GoString(str))
	return str
}
 
func main() {
 
	// fmt.Printf("Done\n")
}

so编译命令

go mod init test
go mod tidy
go build -buildmode=c-shared -o libtest.so

编译成功后会自动生成libtest.so 以及 libtest.h文件

可以通过一下命令检查是否编写编译成功.

readelf -a libtest.so| grep -i hello

C语言测试程序
main.c文件

#include "libtest.h"
void main(int argc, char* argv[]) {
    char * str =  HelloWorld("asdf");
    print(str);
}

如果libtest.h文件为自动生成,我们可以自己手动编写libtest.h文件

#ifndef __INTERFACE_H__
#define __INTERFACE_H__
 
char* HelloWorld(char *);
 
#endif
CMakeList.txt编写

 
cmake_minimum_required(VERSION 2.8)
project(demo)
# C++11 编译
set(CMAKE_BUILD_TYPE "Debug")
set(CMAKE_CXX_STANDARD 11)
# set(CMAKE_SHARED_LINKER_FLAGS " -L./ -Wl,-rpath=./")
SET(CMAKE_INCLUDE_CURRENT_DIR ON)
SET(CMAKE_CXX_STANDARD_REQUIRED ON)
 
set( demo_SOURCE
   ${CMAKE_CURRENT_SOURCE_DIR}/main.c
)
 
link_directories(${PROJECT_SOURCE_DIR})
 
SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR})
 
add_executable(demo ${demo_SOURCE})
 
target_link_libraries(demo -ltest ) 

编译C程序

mkdir build
cd build
cmake ..
make

运行

./demo

高级用法
注意:go语言中申请的内存不要直接c中使用,因为go语言的内存管理是自己决定的,因此你不知道他在什么时候自动释放,因此如果需要在go语言中使用内容请使用c.malloc来申请内容返回到C语言中。C语言注意要自己调用free释放哦。

func add() (**C.char, *C.char) {
        //a := []string{"1111111111111111111", "222", "333"}
        a, err := Parsefilelist()
        if err != nil {
                fmt.Println(err)
                return nil, C.CString(err.Error())
        }
        count := len(a)
        c_count := C.int(count)
        //使用完请释放该内存空间,防止内存泄露
        cArray := C.malloc(C.size_t(c_count) * C.size_t(unsafe.Sizeof(uintptr(0))))
        b := (*[1024]*C.char)(cArray)
        for index, value := range a {
                b[index] = C.CString(value)
        }
        return (**C.char)(&b[0]), nil
}

如果觉得我的文章对您有用,请随意赞赏。您的支持将鼓励我继续创作!