C语言和go语言之间的交互 - C语言中使用go语言,使用的go语言又使用了c语言

一、go语言中使用C语言

go代码中使用C代码,在go语言的函数块中,以注释的方式写入C代码,然后紧跟import “C” 即可在go代码中使用C函数

代码示例:

go代码:testC.go

  1 package main
  2
  3 /*
  4 #include <stdio.h>
  5 #include <stdlib.h>
  6 void c_print(char *str) {
  7     printf("%s\n", str);
  8 }
  9 */
 10 import "C"   //import “C” 必须单起一行,并且紧跟在注释行之后
 11 import "unsafe"
 12
 13 func main() {
 14     s := "Hello Cgo"
 15     cs := C.CString(s)//字符串映射
 16     C.c_print(cs)//调用C函数
 17     defer C.free(unsafe.Pointer(cs))//释放内存
 18 }

运行结果:

$ go run testC.go Hello Cgo

讲解:

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字符串

需要注意的是每次转换都会导致一次内存复制,所以字符串的内容是不可以修改的

5、17行 利用defer C.free 和unsafe.Pointer显示释放调用C.Cstring所生成的内存块

二、C语言中使用go语言

代码示例:

go代码:print.go

  1 package main
  2
  3 import "C"
  4 import "fmt"
  5
  6 //export go_print
  7 func go_print(value string) {
  8     fmt.Println(value)
  9 }
 10
 11 func main() {//main函数是必须的 有main函数才能让cgo编译器去把包编译成C的库
 12 }

讲解:

1、第11行 这里go代码中的main函数是必须的,有main函数才能让cgo编译器去把包编译成c的库

2、第3行 import “C”是必须的,如果没有import “C” 将只会build出一个.a文件,而缺少.h文件

3、第6行 //exoort go_print  这里的go_print要和下面的的go函数名一致,并且下面一行即为要导出的go函数

4、命令执行完毕后会生成两个文件 nautilus.a nautilus.h

nautilus.h中定义了go语言中的类型在C中对应的类型 和导出的go函数的函数声明

如:

typedef signed char GoInt8;//对应go代码中的int8类型
typedef struct { const char *p; GoInt n; } GoString;//对应go中的字符串
extern void go_print(GoString p0);//go中导出的函数的函数声明

C代码: c_go.c

  1 #include “nautilus.h”//引入go代码导出的生成的C头文件
  2 #include <stdio.h>
  3
  4 int main() {
  5   char cvalue[] = "Hello This is a C Application";
  6   int length = strlen(cvalue);
  7   GoString value = {cvalue, length};//go中的字符串类型在c中为GoString
  8   go_print(value);
  9   return 0;
 10 }

编译步骤

// as c-shared library
$ go build -buildmode=c-shared -o nautilus.a print.go

或者

// as c-archive 
$ go build -buildmode=c-archive -o nautilus.a print.go
$ gcc -o c_go c_go.c nautilus.a

运行结果

$ ./c_go Hello This is a C Application

讲解:

1、第1行 #include “nautilus.h"包含go代码导出生成的C头文件

2、第7行 go中字符串类型在c中为GoString 定义为typedef struct { const char *p; GoInt n; } GoString; p为字符串指针,n为长度;所以这里通过GoString value = {cavalue, length}将C中的char赋值给GoString

3、第8行 go_print调用对应函数

三、C语言中使用go语言,使用的go语言又使用了c语言

代码示例:

被go调用的C代码 hello.h

  1 #ifndef HELLO_H
  2 #define HELLO_H
  3
  4
  5 #include <stdio.h>
  6 #include <stdlib.h>7
  8 void go_print_c(char *str);
  9
 10 #endif

被go调用的C代码 hello.c

1 #include "hello.h"
  2
  3 void go_print_c(char *str) {
  4     printf("%s\n", str);
  5 }

被C调用的go代码 print.go

  1 package main
  2
  3 //#include "hello.h"
  4 import "C"
  5
  6 //export go_print
  7 func go_print(value string) {
  8     cs := C.CString(value)
  9     C.go_print_c(cs)
 10 }
 11
 12 func main() {
 13 }

讲解:

1、这里在函数前面加上了inline关键字

如果把C代码放入go代码注释块中并且没有inline关键字中,会出现重定义的错误

p.go

  1 package main
  2
  3 /*
  4  #include <stdio.h>
  5  #include <stdlib.h>
  6 void go_print_c(char *str) {
  7     printf("%s\n", str);
  8 }
  9 */
 10 import "C"
 11 import "unsafe"
 12
 13 //export go_print
 14 func go_print(value string) {
 15     cs := C.CString(value)
 16     C.go_print_c(cs)
 17 }
 18 ... 

 go build -buildmode=c-shared -o nautilus.a print.go执行失败

duplicate symbol _go_print_c in: $WORK/_/Users/baidu/go_code/t/_obj/_cgo_export.o $WORK/_/Users/baidu/go_code/t/_obj/p.cgo2.o ld: 1 duplicate symbol for architecture x86_64 clang: error: linker command failed with exit code 1 (use -v to see invocation)

解决办法是给函数加上inline或者static关键字将函数改成内部链接,或者是像上面那样include头文件

C代码 _c_go.c

  1 #include "nautilus.h"
  2 #include3
  4 int main() {
  5   printf("This is a C Application.\n");
  6   char cvalue[] = "hello world";
  7   int length = strlen(cvalue);
  8   GoString value = {cvalue, length};
  9   go_print(value);
 10   return 0;
 11 }

编译步骤:

// as c-shared library
$ go build -buildmode=c-shared -o nautilus.a 

或者

// as c-archive 
$ go build -buildmode=c-archive -o nautilus.a 
$ gcc -o c_go_c c_go.c nautilus.a

运行结果

$ ./c_go_c.o This is a C Application. hello world

四、参考文献

http://www.cnblogs.com/sevenyuan/p/4544294.html               Go与C语言的互操作

http://blog.ralch.com/tutorial/golang-sharing-libraries/              Sharing Golang packages to C and Go

https://groups.google.com/forum/#!topic/golang-china/vUd4Civs_Bs         google go论坛

http://blog.csdn.net/u014633283/article/details/52225274          GO中调用C代码(CGO)中的坑

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏阿凯的Excel

Python读书笔记17(while与列表、字典)

今天分享利用while函数处理列表和字典,顺便温习一下历史知识 一、论如何将一个列表折腾至另外一个列表!(两个列表是独立的) 论折腾列表有几种方法! 先分...

36950
来自专栏郑科的专栏

PHP7 新特性简介(一)

PHP7是PHP编程语言全新的一个版本,在性能方面获得了极大的提升。官方的文档显示,PHP7可以达到PHP5.x版本两倍的性能。同时还提供了很多其他语言流行的语...

68700
来自专栏我是攻城师

在Scala里面如何使用元组

37840
来自专栏Deep learning进阶路

C++随记(八)---存储持续性、作用域和链接性

版权声明:本篇文章是阅读《C++primer plus (第6版)中文版》第9章之后所作的笔记。部分文字和图表摘自于这本书。 C++随记(八)---存储持续性、...

20200
来自专栏大眼瞪小眼

PHP HashTable总结

本篇文章主要是对 PHP HashTable 总结,下面的参考链接是很好的学习资料。学习“散列”这个数据结构—推荐《数据结构与算法分析 C语言描述》

17610
来自专栏java学习

Java基础总结大全(1)

一、基础知识: 1、JVM、JRE和JDK的区别: JVM(Java Virtual Machine):java虚拟机,用于保证java的跨平台的特性。 ...

37950
来自专栏Brian

C语言心得一

C语言学习 ---- C语言是一个非常灵活且高效的语言,在学习的过程中总会有很多坑。最近有一个项目是混合编程,高性能部分采用C/C++来完成,数据处理和分析采用...

39460
来自专栏大闲人柴毛毛

三分钟理解“模板方法模式”——设计模式轻松掌握

模板方法模式的官方定义: 在模板方法模式中,只定义一个算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重新定义该算法的某些特定...

375100
来自专栏程序员互动联盟

【答疑释惑】c/c++有办法加快程序进入main()前的速度吗

一、答案是Yes。 因为在执行main函数之前,程序会进行很多工作,包括调用系统API,初始化环境。 你能触及到的就是全局变量和静态变量,这些初始化工作也都在m...

32250
来自专栏Linyb极客之路

并发编程之Synchronized关键字

一、Synchronized的基本使用 Synchronized是Java中解决并发问题的一种最常用的方法,也是最简单的一种方法。Synchronized的作...

30460

扫码关注云+社区

领取腾讯云代金券