C和Go相互调用

aiat5431 6年前
   <p>C可以调用Go,并且Go可以调用C, 如果更进一步呢, C-->Go-->C 或者 Go-->C-->Go 的调用如何实现?</p>    <p>本文通过两个简单的例子帮助你了解这两种复杂的调用关系。本文不涉及两者之间的复杂的数据转换,官方文章 <a href="/misc/goto?guid=4959730284455884938" rel="nofollow,noindex">C? Go? Cgo!</a> 、 <a href="/misc/goto?guid=4959730284536450207" rel="nofollow,noindex">wiki/cgo</a> 和 <a href="/misc/goto?guid=4959758074574008490" rel="nofollow,noindex">cmd/cgo</a> 有一些介绍。</p>    <h2>Go-->C-->Go</h2>    <p>Go程序调用C实现的函数,然后C实现的函数又调用Go实现的函数。</p>    <p>1、首先,我们新建一个 hello.go 的文件:</p>    <p>hello.go</p>    <table>     <tbody>      <tr>       <td> <pre>  <code class="language-go">package main    import "C"  import "fmt"    //export HelloFromGo  func HelloFromGo() {   fmt.Printf("Hello from Go!\n")  }  </code></pre> </td>      </tr>     </tbody>    </table>    <p>它定义了一个 HelloFromGo 函数,注意这个函数是一个纯的Go函数,我们定义它的输出符号为 HelloFromGo 。</p>    <p>2、接着我们新建一个 hello.c 的文件:</p>    <p>hello.c</p>    <table>     <tbody>      <tr>       <td> <pre>  <code class="language-go">#include <stdio.h>  #include "_cgo_export.h"    int helloFromC() {   printf("Hi from C\n");   //call Go function   HelloFromGo();   return 0;  }  </code></pre> </td>      </tr>     </tbody>    </table>    <p>这个c文件定义了一个C函数 helloFromC ,内部它会调用我们刚才定义的 HelloFromGo 函数。</p>    <p>这样,我们实现了 C 调用 Go : C-->Go ,下面我们再实现Go调用C。</p>    <p>3、最后新建一个 main.go 文件:</p>    <p>main.go</p>    <table>     <tbody>      <tr>       <td> <pre>  <code class="language-go">package main    /*  extern int helloFromC();  */  import "C"    func main() {   //call c function   C.helloFromC()  }  </code></pre> </td>      </tr>     </tbody>    </table>    <p>它调用第二步实现的C函数 helloFromC 。</p>    <p>运行测试一下:</p>    <pre>  <code class="language-go">$ go run .  Hi from C  Hello from Go!  </code></pre>    <p>可以看到,期望的函数调用正常的运行。第一行是C函数的输出,第二行是Go函数的输出。</p>    <h2>C-->Go-->C</h2>    <p>第二个例子演示了C程序调用Go实现的函数,然后Go实现的函数又调用C实现的函数。</p>    <p>1、首先新建一个 hello.c 文件:</p>    <p>hello.c</p>    <table>     <tbody>      <tr>       <td> <pre>  <code class="language-go">#include <stdio.h>    int helloFromC() {   printf("Hi from C\n");   return 0;  }  </code></pre> </td>      </tr>     </tbody>    </table>    <p>它定义了一个纯C实现的函数。</p>    <p>2、接着新建一个 hello.go 文件:</p>    <pre>  <code class="language-go">// go build -o hello.so -buildmode=c-shared .  package main    /*  extern int helloFromC();  */  import "C"    import "fmt"    //export HelloFromGo  func HelloFromGo() {   fmt.Printf("Hello from Go!\n")   C.helloFromC()  }    func main() {    }  </code></pre>    <p>它实现了一个Go函数 HelloFromGo ,内部实现调用了C实现的函数 helloFromC ,这样我们就实现了 Go-->C 。</p>    <p>注意包名设置为 package main ,并且增加一个空的 main 函数。</p>    <p>运行 go build -o hello.so -buildmode=c-shared . 生成一个C可以调用的库,这调命令执行完后会生成 hello.so 文件和 hello.h 文件。</p>    <p>3、最后新建一个文件夹,随便起个名字,比如 main</p>    <p>将刚才生成的 hello.so 文件和 hello.h 文件复制到 main 文件夹,并在 main 文件夹中新建一个文件 main.c :</p>    <p>main.c</p>    <table>     <tbody>      <tr>       <td> <pre>  <code class="language-go">#include <stdio.h>  #include "hello.h"    int main() {   printf("use hello lib from C:\n");      HelloFromGo();      return 0;  }  </code></pre> </td>      </tr>     </tbody>    </table>    <p>运行 gcc -o main main.c hello.so 生成可执行文件 main , 运行 main :</p>    <pre>  <code class="language-go">$ ./main  use hello lib from C:  Hello from Go!  Hi from C  </code></pre>    <p>第一行输出来自 main.c ,第二行来自Go函数,第三行来自 hello.c 中的C函数,这样我们就实现了 C-->Go--C 的复杂调用。</p>    <h3>参考文档</h3>    <ol>     <li><a href="/misc/goto?guid=4959758074661073659" rel="nofollow,noindex">https://medium.com/using-go-in-mobile-apps/using-go-in-mobile-apps-part-1-calling-go-functions-from-c-be1ecf7dfbc6</a></li>     <li><a href="/misc/goto?guid=4959758074756250968" rel="nofollow,noindex">https://github.com/vladimirvivien/go-cshared-examples</a></li>     <li><a href="/misc/goto?guid=4959758074574008490" rel="nofollow,noindex">http://golang.org/cmd/cgo</a></li>     <li><a href="/misc/goto?guid=4959758074859304153" rel="nofollow,noindex">https://gist.github.com/zchee/b9c99695463d8902cd33</a></li>     <li><a href="/misc/goto?guid=4959758074940181206" rel="nofollow,noindex">https://medium.com/@liamkelly17/working-with-packed-c-structs-in-cgo-224a0a3b708b</a></li>    </ol>    <p> </p>    <p>来自:https://colobu.com/2018/08/28/c-and-go-calling-interaction/</p>    <p> </p>