[Go语言]cgo用法演示
经历了数十年发展的C语言,各种各样的现成的库已经非常丰富。通过cgo,可以在Go语言中使用C语言代码,充分利用好现有的“轮子”。
本文所有代码,在下述环境中调试通过:
- Windows 8.1 64-bit
- Go 1.3.3 64-bit
- GCC 4.8.1 64-bit
要想使用cgo,要导入C“包”:
import "C"
这行代码的上方要紧挨着连续的若干行的注释,在这些注释中编写C代码。例如:
/*
int PlusOne(int n)
{
return n + 1;
}
*/
import "C"
我们知道,如果要引用一个包中的符号,需要用“包名.符号名”的方式,C“包”也是这样,例如:C.int、C.GetWindowLongPtr。
下面介绍使用C语言变量、函数、结构体、联合体、回调函数和动态链接库(Dynamic Link Library,dll)的方法。
- 变量
- 函数
- 结构体
- 联合体
- 回调函数
- dll
1. 变量
使用C的变量很简单,比方说,要使用int,只要在Go代码中写C.int就可以了。
package main import (
"fmt"
) import "C" func main() {
var n C.int
n = 5
fmt.Println(n) // 5 var m1 int
// Go不认为C.int与int、int32等类型相同
// 所以必须进行转换
m1 = int(n + 3)
fmt.Println(m1) // 8 var m2 int32
m2 = int32(n + 20)
fmt.Println(m2) // 25
}
2. 函数
在Go中调用C的函数也不困难。
package main import (
"fmt"
) /*
int PlusOne(int n)
{
return n + 1;
}
*/
import "C" func main() {
var n int = 10
var m int = int(C.PlusOne(C.int(n))) // 类型要转换
fmt.Println(m) // 11
}
3. 结构体
package main import (
"fmt"
) /*
typedef struct _POINT
{
double x;
double y;
}POINT;
*/
import "C" func main() {
var p C.POINT
p.x = 9.45
p.y = 23.12
fmt.Println(p) // {9.45 23.12}
}
4. 联合体
Go中使用C的联合体是比较少见而奇怪的事情,而且稍显麻烦,因为Go将C的联合体视为字节数组。比方说,下面的联合体LARGE_INTEGER被视为[8]byte。
typedef long LONG;
typedef unsigned long DWORD;
typedef long long LONGLONG; typedef union _LARGE_INTEGER {
struct {
DWORD LowPart;
LONG HighPart;
};
struct {
DWORD LowPart;
LONG HighPart;
} u;
LONGLONG QuadPart;
} LARGE_INTEGER, *PLARGE_INTEGER;
所以,如果一个C的函数的某个参数的类型为LARGE_INTEGER,我们可以给它一个[8]byte类型的实参,反之亦然。
package main import (
"fmt"
) /*
typedef long LONG;
typedef unsigned long DWORD;
typedef long long LONGLONG; typedef union _LARGE_INTEGER {
struct {
DWORD LowPart;
LONG HighPart;
};
struct {
DWORD LowPart;
LONG HighPart;
} u;
LONGLONG QuadPart;
} LARGE_INTEGER, *PLARGE_INTEGER; void Show(LARGE_INTEGER li)
{
li.u.LowPart = 1;
li.u.HighPart = 4;
}
*/
import "C" func main() {
var li C.LARGE_INTEGER // 等价于: var li [8]byte
var b [8]byte = li // 正确,因为[8]byte和C.LARGE_INTEGER相同
C.Show(b) // 参数类型为LARGE_INTEGER,可以接收[8]byte
li[0] = 75
fmt.Println(li) // [75 0 0 0 0 0 0 0]
li[4] = 23
Test(li) // 参数类型为[8]byte,可以接收C.LARGE_INTEGER
} func Test(b [8]byte) {
fmt.Println(b)
}
5. 回调函数
有些C函数的参数是回调函数,比方说:
typedef UINT_PTR(__stdcall* GIRL_PROC)(int);
typedef UINT_PTR(__cdecl* GIRL_PROC_CDECL)(int); UINT_PTR Func1(int n, GIRL_PROC gp)
{
if (gp == NULL)
{
return 0;
}
return (*gp)(n);
} UINT_PTR Func2(int n, GIRL_PROC_CDECL gp)
{
if (gp == NULL)
{
return 0;
}
return (*gp)(n);
}
syscall包中有如下两个函数:
syscall.NewCallback
syacall.NewCallbackCDecl
其中,第一个函数接收一个Go函数(这个Go函数的返回值必须只有一个,而且类型为uintptr),并生成一个__stdcall调用约定的C函数,并将生成的函数的地址以uintptr的形式返回;第二个函数的作用与之类似,但生成的函数的调用约定是__cdecl。
一个值得注意的问题是:C的指向函数的指针在Go中被视为*[0]byte,所以要转换一下才能用。这里演示一下__stdcall调用约定的函数的用法,__cdecl类似。
package main import (
"fmt"
"syscall"
"unsafe"
) /*
#define WIN32_LEAN_AND_MEAN
#include <windows.h> typedef UINT_PTR(__stdcall* GIRL_PROC)(int);
typedef UINT_PTR(__cdecl* GIRL_PROC_CDECL)(int); UINT_PTR Func1(int n, GIRL_PROC gp)
{
if (gp == NULL)
{
return 0;
}
return (*gp)(n);
} UINT_PTR Func2(int n, GIRL_PROC_CDECL gp)
{
if (gp == NULL)
{
return 0;
}
return (*gp)(n);
}
*/
import "C" func GirlProc(n int32) uintptr {
return uintptr(n + 97)
} func main() {
gp := syscall.NewCallback(GirlProc)
fmt.Println(gp)
gop := (*[0]byte)(unsafe.Pointer(gp))
var t C.UINT_PTR = C.Func1(C.int(29), gop)
fmt.Println(t) // 126
}
6. dll
以后再写。
[Go语言]cgo用法演示的更多相关文章
- 标准SQL语言的用法
原文链接:http://www.ifyao.com/2015/05/18/%E6%A0%87%E5%87%86%E7%9A%84sql%E8%AF%AD%E8%A8%80%E4%BD%BF%E7%94 ...
- HTML5 Canvas阴影用法演示
HTML5 Canvas阴影用法演示 HTML5 Canvas中提供了设置阴影的四个属性值分别为: context.shadowColor = “red” 表示设置阴影颜色为红色 context.sh ...
- 【转】话说C语言const用法
原文:话说C语言const用法 const在C语言中算是一个比较新的描述符,我们称之为常量修饰符,意即其所修饰的对象为常量(immutable). 我们来分情况看语法上它该如何被使用. 1.函数体内修 ...
- C#中的yield return用法演示源码
下边代码段是关于C#中的yield return用法演示的代码. using System;using System.Collections;using System.Collections.Gene ...
- 【三支火把】---C语言const用法总结
C语言关键字const相信对于不少C语言新手是既陌生又熟悉的,好像经常见,但是却不知道为何用,怎么用?学习至此,总结一下const的用法,使用程序来帮助你理解该关键字,希望能帮到像我一样的新手. 我看 ...
- java struts2入门学习--OGNL语言基本用法
一.知识点学习 1.struts2中包含以下6种对象,requestMap,sessionMap,applicationMap,paramtersMap,attr,valueStack; 1)requ ...
- 快速掌握mongoDB(四)—— C#驱动MongoDB用法演示
前边我们已经使用mongo shell进行增删查改和聚合操作,这一篇简单介绍如何使用C#驱动MongoDB.C#驱动MongoDB的本质是将C#的操作代码转换为mongo shell,驱动的API也比 ...
- .net Core MongoDB用法演示
C#驱动MongoDB的本质是将C#的操作代码转换为mongo shell,驱动的API也比较简单明了,方法名和js shell的方法名基本都保持一致,熟悉mongo shell后学习MongoDB的 ...
- 【C_Language】---C语言const用法总结
C语言关键字const相信对于不少C语言新手是既陌生又熟悉的,好像经常见,但是却不知道为何用,怎么用?学习至此,总结一下const的用法,使用程序来帮助你理解该关键字,希望能帮到像我一样的新手. 我看 ...
随机推荐
- 【数据库】一篇文章搞掂:MySQL数据库
一.安装 使用版本:5.7(2018/08/03 阿里云的云数据库最高支持5.7,所以这里考虑用5.7) 下载版本:MySQL Community Server 5.7.23 下载地址:https:/ ...
- ubuntu安装完成后需要做的事情
1.删除libreoffice libreoffice虽然是开源的,但是Java写出来的office执行效率实在不敢恭维,装完系统后果断删掉 [html] view plain copy sudo a ...
- upc组队赛5 Ground Defense【枚举】
Ground Defense 题目描述 You are a denizen of Linetopia, whose n major cities happen to be equally spaced ...
- css点击高亮
.btn-default:active:focus, .btn-default:active:hover { color: #333; background-color:lightseagreen; ...
- 3.Jmeter 快速入门教程(三-1) --添加响应断言(即loadrunner中所指的检查点)
上一节课,我们创建了一个测试场景,并进行了少量vuser的负载测试. 有时候我们执行了测试,但是发现并不是所有事务都执行成功了. 那是因为我们只是发起了测试,但并没有对每次请求测试的返回作校验. 所以 ...
- 安全服务——CVE中CVSS相关指标介绍
目录 CVSS相关指标 一.CVSS是什么 二.指标内容 1.Base指标 2.Temporal指标 3.Environmental指标 三.Base, Temporal, Environmental ...
- GNU Linux 64汇编学习
函数调用传参: 第一个参数:rdi, 第二个参数:rsi 函数调用栈结构: 返回值 第一个参数 第二个参数 +----------+ rsp-24 | a | +----------+ rsp-16 ...
- 清除浮动最有效的css写法
说起来呢,基本上只要你给容器div加了float的属性,就需要清除浮动来使页面显示正常,那么,到底有什么方法可以让浮动轻松清除呢? 可以用display:flex;替代,但是它对IE8,9支持不是很好 ...
- React-Native 指定模拟器RUN-IOS
react-native run-ios --simulator "iPhone 7”
- 面试经典:链表中倒数第k个结点?如何从大量数据中找出高频词?
记录两道面试题: 题目描述: 输入一个链表,输出该链表中倒数第k个结点.(单向链表) 拿到这个问题的时候自然而然会想到让链表从末尾开始next K-1 次不就是第K-1个节点了么,但是必须要注意一 ...