C语言和go语言之间的交互操作方法

一、go代码中使用C代码

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

代码示例:

go代码:testC.go

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))//释放内存
}

运行结果:

$ 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

package main
import "C"
import "fmt"
//export go_print
func go_print(value string) {
  fmt.Println(value)
}
func main() {//main函数是必须的 有main函数才能让cgo编译器去把包编译成C的库
}

讲解:

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

#include “nautilus.h”//引入go代码导出的生成的C头文件
#include <stdio.h>

int main() {
 char cvalue[] = "Hello This is a C Application";
 int length = strlen(cvalue);
 GoString value = {cvalue, length};//go中的字符串类型在c中为GoString
 go_print(value);
 return 0;
}

编译步骤

// 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

#ifndef HELLO_H
#define HELLO_H

#include <stdio.h>
#include <stdlib.h>7
void go_print_c(char *str);

#endif

被go调用的C代码 hello.c

#include "hello.h"

void go_print_c(char *str) {
  printf("%s\n", str);
}

被C调用的go代码 print.go

package main

//#include "hello.h"
import "C"

//export go_print
func go_print(value string) {
 cs := C.CString(value)
 C.go_print_c(cs)
}

func main() {
}

讲解:

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

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

p.go

package main

/*
#include <stdio.h>
#include <stdlib.h>
void go_print_c(char *str) {
 printf("%s\n", str);
}
*/
import "C"
import "unsafe"

//export go_print
func go_print(value string) {
 cs := C.CString(value)
 C.go_print_c(cs)
}
...

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

#include "nautilus.h"
#include3
int main() {
 printf("This is a C Application.\n");
 char cvalue[] = "hello world";
 int length = strlen(cvalue);
 GoString value = {cvalue, length};
 go_print(value);
 return 0;
}

编译步骤:

// 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

以上这篇C语言和go语言之间的交互操作方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • Go语言中嵌入C语言的方法

    本文实例讲述了Go语言中嵌入C语言的方法.分享给大家供大家参考.具体分析如下: Go语言官方带了一个工具叫cgo,可以很方便的在Go语言代码中内嵌C代码或做C和Go代码的集成.下面是一段简单的在Go中内嵌C的实验代码: 复制代码 代码如下: package main /* #include <stdio.h> #include <stdlib.h> void say_hello() {         printf("Hello World!\n"); } */

  • C语言和go语言之间的交互操作方法

    一.go代码中使用C代码 go代码中使用C代码,在go语言的函数块中,以注释的方式写入C代码,然后紧跟import "C" 即可在go代码中使用C函数 代码示例: go代码:testC.go package main /* #include <stdio.h> #include <stdlib.h> void c_print(char *str) { printf("%s\n", str); } */ import "C"

  • 易语言中Com对象和Office之间的交互使用方法

    Com对象和office之间的交互使用 在所有属于office支持库的控件中都可以调用com对象来完善功能,那在这些控件中如何调用com对象呢? 首先新建"易语言"我们以word支持库为例,在窗体上放入word程序集和word文档集 我们写入如下代码,把两个组件联系起来 这时,我们查看word文档集的命令中有"取某某对象"的命令,这些命令的返回值就是一个对象,使用这些命令的时候必须先创建一个对象类型的变量来保存这些命令的返回值. 这时一个对象就产生了,我们可以用对象

  • 实例说明js脚本语言和php脚本语言的区别

    js脚本语言和php脚本语言的区别是什么? 一句话: js是客户端脚本, 由浏览器执行. php是服务端脚本, 由php服务执行, php脚本跟shell脚本(bash执行)颇为类似. 来看看js脚本, 写一个test.html文件, 其中内容为: <script> alert("this is js"); </script> 直接双击本地的test.html文件, 结果就有一个弹框了, 因为浏览器执行了上述js脚本. 再看看看php脚本, 写一个test.ph

  • 关于C语言和命令行之间的交互问题

    在Windows操作系统中,后缀为.exe的文件都是可执行文件..exe是英文单词executable的缩写,意思是可执行的.凡是可执行的文件都是二进制的文件,计算机也只能识别二进制的文件. 后缀为.exe的文件是由C语言的源文件.c或C++的源文件.cpp编译而来. 在Windows操作系统中,凡是.exe的文件,都可以作为一条命令,在命令行中来执行.例如:在我的电脑D盘cPro文件夹中就有很多编译好的.exe可执行文件: Win + R 键输入cmd,进入命令行: 然后输入命令进入到D:\c

  • PHP与Go语言之间的通信详解

    前言 最近工作中遇到的一个场景,php项目中需要使用一个第三方的功能,而恰好有一个用Golang写好的类库.那么问题就来了,要如何实现不同语言之间的通信呢?下面就来一起看看吧. 常规的方案 1. 用Golang写一个http/TCP服务,php通过http/TCP与Golang通信 2.将Golang经过较多封装,做为php扩展. 3.PHP通过系统命令,调取Golang的可执行文件 存在的问题 1.http请求,网络I/O将会消耗大量时间 2.需要封装大量代码 3.PHP每调取一次Golang

  • C++、C语言和JAVA开发的区别

    1.面向对象没有java彻底. 由于C++要兼容C的内容,而C是面向过程的,所以C++不可避免地出现过程影子,并不算是完全的面向对象的程序设计语言.例如总得要有main或winmain之类的过程吧. 2.C++的移植能力没有java好. 由于C++的事实标准的存在,即各个编译器总存在差异,所以或多或少存 在不兼容.而且各个软件平台的C++启动代码和硬件指令不同,编译后的C++程序一般是不能跨平台的.而java从娘胎里出来就是为了跨平台执行的,不采 用二进制机器码作为最终代码,所以在移植方面较好.

  • C语言和C++的6点区别

    C语言和C++的区别 (1)面向过程语言和面向对象语言 C语言是面向过程语言,即先分析出解决问题的步骤然后再将这些步骤一一实现 C++是面向对象语言,即把问题分成若干个对象,目的是为了描述某个事物在解决整个问题的步骤中的行为 (2)关键字不同 C语言中有32个关键字,而C++有63个关键字.另外在C语言中struct关键字定义的变量不能有函数,而在C++中可以有函数 (3)文件后缀名不同 C语言中源文件的后缀名是.c,C++源文件后缀名是.cpp (4)函数返回值不同 C语言中如果一个函数没有指

  • c语言和c++语言中const修饰的变量区别浅析

    目录 c: 修饰全局变量: 修饰局部变量: c++: 修饰全局变量: 修饰局部变量: 总结: 在c语言中: 在c++语言中: 总结 c: 修饰全局变量: 用const修饰的全局变量是没有办法直接修改的,间接的修改也是不成功的(语法可以通过,但是编译运行的时候会报错.)(const只要是修饰全局变量,那么就会储存到常量区中,收到常量区的保护.) 修饰局部变量: 但是如果用const修饰局部变量,同样的也是没有办法直接修改的,但是是可以间接修改的. int main() { const int b

  • AngularJS指令与控制器之间的交互功能示例

    本文实例讲述了AngularJS指令与控制器之间的交互功能.分享给大家供大家参考,具体如下: 本节我们来看控制器与指令之间的交互 1.首先来看最简单的,在指令中调用父控制器的方法: <div ng-controller="myController1"> </div> app.controller('myController1',['$scope',function($scope){ $scope.load=function(){ console.log('正在加

  • AngularJS指令与指令之间的交互功能示例

    本文实例讲述了AngularJS指令与指令之间的交互功能.分享给大家供大家参考,具体如下: 前面一篇文章<AngularJS指令与控制器之间的交互功能示例>我们了解了指令与控制器之间的交互,接下来看看指令与指令之间是如何进行交互的. 1.首先来了解一下什么是独立scope 为了更好的理解独立scope,我们来看一段代码: <div ng-controller="myController1"> <hello></hello> <hel

随机推荐