在Golang中使用C语言代码实例

作者:junjie 时间:2024-05-25 15:15:46 

cgo 使得在 Golang 中可以使用 C 代码。

Hello World

为了有一个较为直观的了解,我们来看一个简单的例子,创建文件 main.go:


package main
 
/*
#include <stdio.h>
 
void sayHi() {
    printf("Hi");
}
*/
import "C"
 
func main() {
    C.sayHi()
}


执行程序:


go run main.go


程序执行并输出 hi(更多的范例可以见 $GOROOT/misc/cgo)。

Windows 下的准备工作

如果想要在 Windows 上使用 cgo,那么需要安装 gcc 编译器,这里我使用 mingw-w64。

设置编译和链接标志

我们使用 import “C” 导入的是一个伪包(pseudo-package),我们通过其来使用 C 代码。在 import “C” 之前,紧跟着 import “C” 的注释可以包括:

1.编译器和链接器标志
2.C 代码

我们可以通过 #cgo 指令来设置编译器和链接器标志,例如:


// #cgo CFLAGS: -DPNG_DEBUG=1
// #cgo amd64 386 CFLAGS: -DX86=1
// #cgo LDFLAGS: -lpng
// #include <png.h>
import "C"

附带提及一点的是,这些指令中可以包含构建约束(build constraint),详细内容见:http://golang.org/pkg/go/build/#hdr-Build_Constraints。

常用的 #cgo 指令有:

1.CPPFLAGS、CFLAGS 指令被用于编译当前包中的 C 文件(任何的 .c、.s、.S 文件)
2.CPPFLAGS、CXXFLAGS 指令被用于编译当前包中的 C++ 文件(任何的 .cpp、.cc、.cxx 文件)
3.LDFLAGS 指令用于指定链接器标志
4.pkg-config 指令用于通过 pkg-config 工具获取编译器和链接器标志(例如:#cgo pkg-config: png cairo)

Golang 引用 C

结构体上需要注意的点:

1.C 结构体的域名称如果为 Golang 的关键字时,访问时需要在域名称前面加上 _。比如说,C 中有一个结构体变量 x,此变量对应的结构体中有一个域 type,那么在 Golang 中需要通过 x._type 来访问 type 域
2.结构体的位域、非对齐数据等无法在 Golang 中表示时会被忽略
3.Golang 结构体中不能使用 C 类型的域

标准的 C 数值类型对应:

1.C.char
2.C.schar(signed char)
3.C.uchar(unsigned char)
4.C.short
5.C.ushort(unsigned short)
6.C.int
7.C.uint(unsigned int)
8.C.long
9.C.ulong(unsigned long)
10.C.longlong(long long)
11.C.ulonglong(unsigned long long)
12.C.float
13.C.double

任何的 C 函数(包括 void 函数)都可以返回一个返回值和 C 的 errno 变量(作为错误):


n, err := C.sqrt(-1)
_, err := C.voidFunc()

直接调用 C 函数指针目前还无法支持。

有一些特殊的函数可以用于 C 类型和 Golang 类型之间转换(通过数据拷贝的方式),伪定义如下:


// Golang 的字符串转为 C 字符串
// C 的字符串是使用 malloc 分配的,因此,此函数的调用者
// 需要调用 C.free 来释放内存
func C.CString(string) *C.char
 
// 转换 C 字符串到 Golang 字符串
func C.GoString(*C.char) string
 
// 转换一定长度的 C 字符串到 Golang 字符串
func C.GoStringN(*C.char, C.int) string
 
// 转换一块 C 内存区域到 Golang 的字节数组中去
func C.GoBytes(unsafe.Pointer, C.int) []byte

其他需要注意的点:

1.C 语言中的 void* 对应 unsafe.Pointer
2.C 语言中的结构、联合、枚举类型(而非变量),在 Golang 中需要加上 struct_、union_、enum_ 前缀访问。由于 Golang 中没有联合这种数据类型,因此 C 的联合在 Golang 中被表示为字节数组
3.和 C 语言等价的那些类型是不可以导出的

标签:Golang,C语言
0
投稿

猜你喜欢

  • 使用Python读取大文件的方法

    2022-02-18 00:43:35
  • MySQL深度分页(千万级数据量如何快速分页)

    2024-01-26 12:33:54
  • Python装饰器如何实现修复过程解析

    2023-11-22 04:27:43
  • Python多线程原理与用法实例剖析

    2023-07-22 13:40:37
  • Python networkx包的实现

    2023-07-29 23:58:03
  • Mysql查询时间区间日期列表实例代码

    2024-01-17 16:17:32
  • 对python中arange()和linspace()的区别说明

    2023-08-23 05:32:53
  • Go语言压缩和解压缩tar.gz文件的方法

    2024-05-21 10:21:46
  • Vue Element前端应用开发之常规Element界面组件

    2024-04-26 17:39:32
  • 解决Tensorflow使用pip安装后没有model目录的问题

    2023-08-09 22:58:05
  • SQL 存储过程基础语法之一

    2024-01-26 19:58:05
  • JS实现根据数组对象的某一属性排序操作示例

    2024-04-19 10:14:50
  • Python 如何给图像分类(图像识别模型构建)

    2022-07-24 20:34:08
  • mysql update case 更新字段值不固定的操作

    2024-01-15 06:25:41
  • 基于python实现获取网页图片过程解析

    2023-10-08 21:57:59
  • python使用SQLAlchemy操作MySQL

    2024-01-16 19:11:11
  • 交互设计模式——分页

    2009-07-30 13:00:00
  • Oracle 数据显示 横表转纵表

    2024-01-22 23:35:05
  • windows开发记事本程序纪实(二)逻辑篇1

    2022-05-15 10:52:51
  • MySQL Create Database 创建数据库

    2011-09-10 16:04:51
  • asp之家 网络编程 m.aspxhome.com