centos_7_x86_x64,gcc_4.8.5
yum install -y pcre pcre-tools pcre-devel
yum install -y byacc
tar -xvzf swig-rel-3.0.12.tar.gz cd swig-rel-3.0.12 ./autogen.sh
./configure
make && make install
swig -version
这里说一下为什么要编译swig源码来进行安装,在yum上安装swig的版本比较低,而较低版本的swig不支持-cgo参数,具体见官方文档。
yum install -y go
go version
#ifndef _TEST_CPP_H_ #define _TEST_CPP_H_ #include#include /// 回调类 class ICallback { public: virtual void notify(const std::string& s) = 0; }; /// 测试类 class TestCall { public: static TestCall* Create() { return new TestCall(); } void SetCallback(ICallback* callback) { callback_ = callback; } int32_t Test(const std::string& s); private: TestCall() : callback_(NULL) {} ICallback* callback_; }; #endif
#include#include "test_cpp.h" int32_t TestCall::Test(const std::string & s) { std::cout << "TestCall::Test(" << s << ")" << std::endl; if (callback_) { callback_->notify(s); } return 0; }
cmake_minimum_required(VERSION 2.8) project(test_cpp C CXX) set(SRC_LISTS test_cpp.cpp) add_library(test_cpp SHARED ${SRC_LISTS})
mkdir cmake cd cmake cmake .. make
由于在golang中不能直接使用C++函数,所以我们稍后会使用swig工具,生成C函数提供给golang使用。
/* Copyright 2011 The Go Authors. All rights reserved. Use of this source code is governed by a BSD-style license that can be found in the LICENSE file. */ /* An example of writing a C++ virtual function in Go. */ %module(directors="1") go_test_cpp %init %{ //printf("Initialization rms done.\n"); %} %typemap(gotype) (char **ppInstrumentID, int nCount) "[]string" %typemap(in) (char **ppInstrumentID, int nCount) %{ { int i; _gostring_* a; $2 = $input.len; a = (_gostring_*) $input.array; $1 = (char **) malloc (($2 + 1) * sizeof (char *)); for (i = 0; i < $2; i++) { /* Not work */ //_gostring_ *ps = &a[i]; //$1[i] = (char *) ps->p; //$1[i][ps->n] = '\0'; /*Work well*/ _gostring_ *ps = &a[i]; $1[i] = (char*) malloc(ps->n + 1); memcpy($1[i], ps->p, ps->n); $1[i][ps->n] = '\0'; } $1[i] = NULL; } %} %typemap(argout) (char **ppInstrumentID, int nCount) "" /* override char *[] default */ %typemap(freearg) (char **ppInstrumentID, int nCount) %{ { int i; for (i = 0; i < $2; i++) { free ($1[i]); } free($1); } %} /* 在复杂的企业环境中,可能有一些 C/C++ 头文件定义了您希望向脚本框架公开的全局变量和常量。 * 在接口文件中使用 %include和 %{ #include %},可解决在头文件中重复所有元素的声明的问题。 */ /* Includes the header files in the wrapper code */ %header %{ #if defined(_WIN32) || defined(__WIN32__) || defined(__CYGWIN__) || defined(WIN32) #include "test_cpp.h" #else #include "test_cpp.h" #endif %} /* Parse the header files to generate wrappers */ %include "std_string.i" %feature("director") ICallback; #if defined(_WIN32) || defined(__WIN32__) || defined(__CYGWIN__) || defined(WIN32) %include "./../cpp/test_cpp.h" #else %include "./../cpp/test_cpp.h" #endif
有几个点我们需要注意:
1.1 指定生成go源文件中的包名
%module(directors="1") go_test_cpp
1.2 指定生成C++源文件中的include代码
%header %{ #if defined(_WIN32) || defined(__WIN32__) || defined(__CYGWIN__) || defined(WIN32) #include "test_cpp.h" #else #include "test_cpp.h" #endif %}
1.3 可以让go支持继承某个C++类
%feature("director") ICallback;
1.4 指定需要解析C++头文件,生成go和C++的包装代码
#if defined(_WIN32) || defined(__WIN32__) || defined(__CYGWIN__) || defined(WIN32) %include "./../cpp/test_cpp.h" #else %include "./../cpp/test_cpp.h" #endif
swig -go -cgo -intgosize 64 -c++ ./go_test_cpp.swigcxx
总共生成了3个文件:go_test_cpp.go、go_test_cpp_wrap.h和go_test_cpp_wrap.cxx
cmake_minimum_required(VERSION 2.8) project(go_test_cpp C CXX) include_directories(${CMAKE_CURRENT_SOURCE_DIR}/../cpp) set(SRC_LISTS go_test_cpp_wrap.cxx) add_library(go_test_cpp STATIC ${SRC_LISTS})
mkdir cmake cd cmake cmake .. make
最后编译的静态库是给go使用的,大致调用流程是:go <=> libgo_test_cpp.a <=> libtest_cpp.so
// Copyright 2012 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. package go_test_cpp //#cgo linux LDFLAGS: -fPIC -L${SRCDIR}/../../cpp/cmake -L${SRCDIR}/../../swig/cmake -Wl,-rpath=${SRCDIR}/../../cpp/cmake -ltest_cpp -lgo_test_cpp -lstdc++ //#cgo linux CPPFLAGS: -fPIC -I. import "C"
链接libtest_cpp.so和libgo_test_cpp.a模块;应该将libgo_test_cpp.go文件保存在go_test_cpp目录下,它和go_test_cpp.go都属于同一个包。
package main import( "fmt" "unsafe" "time" "./go_test_cpp" ) type my_callback struct { go_test_cpp.ICallback } func (this my_callback) Notify(arg2 string) { fmt.Printf("c++ Notify:%s\n", arg2) } func main() { cb := go_test_cpp.NewDirectorICallback(my_callback{}) test := go_test_cpp.TestCallCreate() test.SetCallback(cb) res_ptr := test.Test("Hello World!").Swigcptr() res := *(*int32)(unsafe.Pointer(res_ptr)) if res == 0 { fmt.Println("Test success!") } else { fmt.Println("init failed[", res, "]!") } go_test_cpp.Swig_free(res_ptr) time.Sleep(time.Second*1) fmt.Println("end") }
go run main.go