c语言调用go封装的动态库步骤及减小体积包的方法
嫌go编译后的动态库或静态库供c代码或嵌入式终端使用,体积太大?upx工具解决这一问题。
upx真是一个神器,再也不用担心和抱怨go在嵌入式linux应用上体积包大了。
1.go build添加 -ldflags="-w -s" 会去除 DWARF调试信息、符号信息
``` go build -ldflags="-w -s" ota_main.go ```
``` go build -buildmode=c-shared -o test.so ``` -buildmode=c-shared requires exactly one main package
注意:生成C可调用的so时,Go源代码需要以下几个注意。
必须导入 “C” 包
必须在可外部调用的函数前加上 【//export 函数名】的注释
必须是main包,切含有main函数,main函数可以什么都不干
2.优化方案 第二步:压缩优化
执行命令: ``` upx.exe -9 *.exe ``` upx-3.96-amd64_linux.tar.xz
打开文件夹
upx工具 解压后放到 /usr/bin目录下就可以直接使用了
编译为c动态库用的什么指令?
增加一个属性:-buildmode=c-archive
-buildmode=c-shared 这个是动态库
举例:一个go文件编译初始值:7M
去调试信息编译:5.2M
upx处理后:1.9M
go调用c代码一例:
package main
/*
#include <stdio.h>
#include <stdlib.h>
void c_print(char *str) {
printf("%sn", str);
}
*/
import "C" //import “C” 必须单起一行,并且紧跟在注释行之后
import "unsafe"
func main() {
s := "Hello Cgo"
cs := C.CString(s)//字符串映射
C.c_print(cs)//调用C函数
defer C.free(unsafe.Pointer(cs))//释放内存
}
1、go代码中的C代码,需要用注释包裹,块注释和行注释均可。
其次import “C”是必须的,并且和上面的C代码之间不能用空行分割,必须紧密相连
如果执行go run **时出现
# command-line-arguments could not determine kind of name for xxx
那么就需要考虑 是不是improt “C”和上面的C代码没有紧挨着导致了
2、import “C” 并没有导入一个名为C的包,这里的import “C”类似于告诉Cgo将之前注释块中的C代码生成一段具有包装性质的Go代码
3、访问C语言中的函数需要在前面加上C.前缀,如C.Cstring C.go_print C.free
4、对于C语中的原生类型,Cgo都有对应的Go语言中的类型 如go代码中C.int,C.char对应于c语言中的int,signed char,而C语言中void*指针在Go语言中用特殊的unsafe.Pointer(cs)来对应。
而Go语言中的string类型,在C语言中用字符数组来表示,二者的转换需要通过go提供的一系列函数来完成:
C.Cstring : 转换go的字符串为C字符串,C中的字符串是使用malloc分配的,所以需要调用C.free来释放内存。
C.Gostring : 转换C字符串为go字符串。
C.GoStringN : 转换一定长度的C字符串为go字符串。
typedef signed char GoInt8;//对应go代码中的int8类型
typedef struct { const char *p; GoInt n; } GoString;//对应go中的字符串
需要注意的是每次转换都会导致一次内存复制,所以字符串的内容是不可以修改的。
5、利用defer C.free 和unsafe.Pointer显示释放调用C.Cstring所生成的内存块。
go代码中的main函数是必须的,有main函数才能让cgo编译器去把包编译成c的库。
import “C”是必须的,如果没有import “C” 将只会build出一个.a文件,而缺少.h文件。
//exoort 函数名 这里的函数名要和下面的的go函数名一致,并且下面一行即为要导出的go函数。
举例:
package main
import "fmt"
import "C"
//export test
func test() {
fmt.Printf("hello go test0n")
}
//export test1
func test1(a, b int) int {
fmt.Printf("hello go test1n")
return 0
}
//export test2
func test2(a, b string) (int, error) {
fmt.Printf("hello go test2,a=%s,b=%sn", a, b)
return 0, nil
}
//export test3
func test3(a, b []byte) {
fmt.Printf("hello go test3n")
fmt.Printf("%#v,%#vn", a, b)
}
func main() {
fmt.Printf("hello go,this is mainn")
}
go build -buildmode=c-shared -o test.so
c语言中调用:
#include <stdio.h>
#include <string.h>
#include "test.h"
int main()
{
printf("c test,use go so libn");
test();
test1(1,2);
struct test2_return r;
char cvalue[] = "Hello This is a C Application";
int length = strlen(cvalue);
GoString value = {cvalue, length};//go中的字符串类型在c中为GoString
r = test2(value,value);
GoSlice p1,p2;
char a1[5],a2[5];
p1.data = a1;
p1.len =5;
p1.cap = 5;
p2.data = a2;
p2.len =5;
p2.cap = 5;
test3(p1,p2);
return 0;
}
test.h头文件如下:
/* Code generated by cmd/cgo; DO NOT EDIT. */
/* package test */
#line 1 "cgo-builtin-prolog"
#include <stddef.h> /* for ptrdiff_t below */
#ifndef GO_CGO_EXPORT_PROLOGUE_H
#define GO_CGO_EXPORT_PROLOGUE_H
typedef struct { const char *p; ptrdiff_t n; } _GoString_;
#endif
/* Start of preamble from import "C" comments. */
/* End of preamble from import "C" comments. */
/* Start of boilerplate cgo prologue. */
#line 1 "cgo-gcc-export-header-prolog"
#ifndef GO_CGO_PROLOGUE_H
#define GO_CGO_PROLOGUE_H
typedef signed char GoInt8;
typedef unsigned char GoUint8;
typedef short GoInt16;
typedef unsigned short GoUint16;
typedef int GoInt32;
typedef unsigned int GoUint32;
typedef long long GoInt64;
typedef unsigned long long GoUint64;
typedef GoInt64 GoInt;
typedef GoUint64 GoUint;
typedef __SIZE_TYPE__ GoUintptr;
typedef float GoFloat32;
typedef double GoFloat64;
typedef float _Complex GoComplex64;
typedef double _Complex GoComplex128;
/*
static assertion to make sure the file is being used on architecture
at least with matching size of GoInt.
*/
typedef char _check_for_64_bit_pointer_matching_GoInt[sizeof(void*)==64/8 ? 1:-1];
typedef _GoString_ GoString;
typedef void *GoMap;
typedef void *GoChan;
typedef struct { void *t; void *v; } GoInterface;
typedef struct { void *data; GoInt len; GoInt cap; } GoSlice;
#endif
/* End of boilerplate cgo prologue. */
#ifdef __cplusplus
extern "C" {
#endif
extern void test();
extern GoInt test1(GoInt p0, GoInt p1);
/* Return type for test2 */
struct test2_return {
GoInt r0;
GoInterface r1;
};
extern struct test2_return test2(GoString p0, GoString p1);
extern void test3(GoSlice p0, GoSlice p1);
#ifdef __cplusplus
}
#endif
gcc test.c test.so 生成了a.out
./a.out运行。
Go导出dll如何返回struct等复杂结构?
Go type not supported in export: struct
因为export里不支持Go语言的struct。
正确的写法如:
package main
/*
struct Info {
int x;
};
*/
import "C"
import (
"log"
)
//export Show
func Show() C.struct_Info{
log.Printf("Return: %d", 110)
return C.struct_Info{x:0}
}
func main() {
Show();
}
- No.003 Longest Substring Without Repeating Characters
- 【Spark研究】极简 Spark 入门笔记——安装和第一个回归程序
- 通常Java开发人员如何进行数据排序?
- 消息服务框架使用案例之--大文件上传(断点续传)功能
- Java中三种Set类型用法、性能大比拼
- Android基础总结(5)——数据存储,持久化技术
- 如何突破Windows环境限制打开“命令提示符”
- 【Spark研究】Spark之工作原理
- Java中泛型使用的必要性
- “一切都是消息”--MSF(消息服务框架)之【发布-订阅】模式
- Android基础总结(4)——广播接收器
- “一切都是消息”--MSF(消息服务框架)之【请求-响应】模式(点对点)
- “老坛泡新菜”:SOD MVVM框架,让WinForms焕发新春
- 解决服务器SID引起虚拟机不能加入AD域用户,无法远程登录的问题
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- 数据防泄漏 | 禁止PrintScreen键
- LeetCode | 实现strStr()
- LeetCode | 机器人能否返回原点
- LeetCode | 2 的幂
- JeeSite | Excel 导入导出
- 贪心法--哈夫曼编码
- 基于maven+ssm的增删改查之使用mybatis逆向工程生成相关文件
- Arrays 的二分查找
- (四十四)golang--协程(goroutine)和管道(channel)相结合实例
- 基于maven+ssm的增删改查之测试相关ssm环境是否成功
- 基于maven+ssm的增删改查之带分页的显示员工相关信息(基于bootstrap)
- (四十五)golang--反射
- 基于maven+ssm的增删改查之前后端之间使用json进行交互(显示员工信息)
- 动态规划--矿工挖矿
- 基于maven+ssm的增删改查之添加员工实现