Golang减小体积包的方法和c语言调用go封装的动态库步骤

简介: Golang减小体积包的方法和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最新版下载地址:


https://www.52pojie.cn/thread-1094430-1-1.html



Changes in 3.96 (23 Jan 2020):


 * bug fixes - see https://github.com/upx/upx/milestone/5


下载地址:


Releases · upx/upx · GitHub


https://github.com/upx/upx/relea ... /upx-3.96-win32.zip


https://github.com/upx/upx/relea ... /upx-3.96-win64.zip


打开文件夹


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("%s\n", 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 test0\n")
}
//export test1
func test1(a, b int) int {
  fmt.Printf("hello go test1\n")
  return 0
}
//export test2
func test2(a, b string) (int, error) {
  fmt.Printf("hello go test2,a=%s,b=%s\n", a, b)
  return 0, nil
}
//export test3
func test3(a, b []byte) {
  fmt.Printf("hello go test3\n")
  fmt.Printf("%#v,%#v\n", a, b)
}
func main() {
  fmt.Printf("hello go,this is main\n")
}


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 lib\n");
   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();
}


相关文章
|
5月前
|
Go 开发者
Go语言包的组织与导入 -《Go语言实战指南》
本章详细介绍了Go语言中的包(Package)概念及其使用方法。包是实现代码模块化、复用性和可维护性的核心单位,内容涵盖包的基本定义、命名规则、组织结构以及导入方式。通过示例说明了如何创建和调用包,并深入讲解了`go.mod`文件对包路径的管理。此外,还提供了多种导入技巧,如别名导入、匿名导入等,帮助开发者优化代码结构与可读性。最后以表格形式总结了关键点,便于快速回顾和应用。
235 61
|
30天前
|
Java 编译器 Go
【Golang】(1)Go的运行流程步骤与包的概念
初次上手Go语言!先来了解它的运行流程吧! 在Go中对包的概念又有怎样不同的见解呢?
85 4
|
30天前
|
Java 编译器 Go
【Golang】(5)Go基础的进阶知识!带你认识迭代器与类型以及声明并使用接口与泛型!
好烦好烦好烦!你是否还在为弄不懂Go中的泛型和接口而烦恼?是否还在苦恼思考迭代器的运行方式和意义?本篇文章将带你了解Go的接口与泛型,还有迭代器的使用,附送类型断言的解释
109 3
|
30天前
|
存储 安全 Java
【Golang】(4)Go里面的指针如何?函数与方法怎么不一样?带你了解Go不同于其他高级语言的语法
结构体可以存储一组不同类型的数据,是一种符合类型。Go抛弃了类与继承,同时也抛弃了构造方法,刻意弱化了面向对象的功能,Go并非是一个传统OOP的语言,但是Go依旧有着OOP的影子,通过结构体和方法也可以模拟出一个类。
87 1
|
2月前
|
监控 前端开发 数据可视化
Github 12.3kstar, 3分钟起步做中后台?Go+Vue 脚手架,把权限、代码生成、RBAC 都封装好了
Go-admin 是基于 Gin + Vue 的中后台脚手架,集成 Casbin RBAC 权限、JWT 鉴权、GORM 数据库操作与 Swagger 文档,内置用户、角色、菜单等管理模块。提供代码生成器与表单构建器,支持多租户与多前端框架(Element UI/Arco/Ant Design),3 分钟快速搭建企业级后台,助力高效交付。
156 4
|
4月前
|
JSON 中间件 Go
Go语言实战指南 —— Go中的反射机制:reflect 包使用
Go语言中的反射机制通过`reflect`包实现,允许程序在运行时动态检查变量类型、获取或设置值、调用方法等。它适用于初中级开发者深入理解Go的动态能力,帮助构建通用工具、中间件和ORM系统等。
286 63
|
4月前
|
设计模式 Kubernetes Go
​​什么是Golang项目的“主包精简,逻辑外置”?​
“主包精简,逻辑外置”是Go语言项目的一种设计原则,强调将程序入口保持简单,核心逻辑拆分至其他包,以提升代码可维护性、可测试性及扩展性,适用于CLI工具、Web服务等场景。
102 7
|
4月前
|
人工智能 测试技术 持续交付
Golang深入浅出之-Go语言中的持续集成与持续部署(CI/CD)
持续集成与持续部署(CI/CD)是现代软件开发的关键实践,尤其适用于Go语言项目。本文探讨了Go项目中常见的CI/CD问题,如测试覆盖不足、版本不一致和构建时间过长,并提供解决方案及GitHub Actions示例代码,帮助开发者优化流程,提升交付效率和质量。
124 5
|
3月前
|
缓存 监控 安全
告别缓存击穿!Go 语言中的防并发神器:singleflight 包深度解析
在高并发场景中,多个请求同时访问同一资源易导致缓存击穿、数据库压力过大。Go 语言提供的 `singleflight` 包可将相同 key 的请求合并,仅执行一次实际操作,其余请求共享结果,有效降低系统负载。本文详解其原理、实现及典型应用场景,并附示例代码,助你掌握高并发优化技巧。
270 0
|
6月前
|
Go 持续交付 开发者
Go语言包与模块(module)的基本使用-《Go语言实战指南》
本章深入讲解Go语言中的包(Package)和模块(Module)概念。包是代码组织的最小单位,每个`.go`文件属于一个包,通过`import`实现复用;主程序包需命名为`main`。模块是Go 1.11引入的依赖管理机制,支持自动版本管理和私有/远程仓库,无需依赖GOPATH。通过实际示例,如自定义包`mathutil`和第三方模块`gin`的引入,展示其使用方法。常用命令包括`go mod init`、`go mod tidy`等,帮助开发者高效管理项目依赖。最后总结,包负责功能划分,模块实现现代化依赖管理,提升团队协作效率。
261 15

热门文章

最新文章

推荐镜像

更多