這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。
服務端的某個項目,要考慮到效能問題,之前使用基於公司架構的php,效能有瓶頸,經過調研,決定使用go。因為之前有一個關鍵模組已經使用c++實現,所以計劃使用go做HTTP容器,讓go和c++模組結合起來。
在網上調研結合方案,go內建的cgo只支援c語言,swig工具可以支援多種語言結合c++,所以決定使用swig,這裡簡要敘述下swig的使用和一些發現的要點。
1. 我們解壓了swig後,首先要根目錄運行configure和make,這樣會產生各個子目錄的makefile,並在目前的目錄產生preinst-swig.in和swig可執行檔,這些在後續會擁有。
2. 我們進入Example的go裡面,這裡有一些值得參考的例子
3. 研究那個class例子,有這幾個檔案:
class.cxx和example.h是我們希望轉換的C++代碼。
example.i是C++和go的橋樑介面檔案,主要是配置。
Makefile會做build工作,其核心其實是調用了Example根目錄下那個Makefile。
runme.go是go代碼,其中調用了C++代碼。
4. 通過閱讀Makefile和分析,瞭解了swig的基本流程:
對C++代碼進行編譯,產生 .o;
對C++代碼根據介面設定檔進行轉化,產生.go和go的打包.a檔案;
將runme.go和產生的C++的.go檔案進行結合,產生可執行檔。
5. 如果要預定義一些map和vector,可以在設定檔裡配置一下就可以,這些類型會和C++中的結構、類等均被轉化為go中的類型並在C++的.go和.a中儲存。一個介面設定檔例子:
/* File : example.i */%module tipcpp%{#include "example.h"%}%include "std_string.i"%include "std_vector.i"%include "std_map.i"namespace std { %template(AppInfoVector) vector<sAppInfo>; %template(JobDict) map<string, string>; %template(RetMap) map<string, bool>;}/* Let's just grab the original header file here */%include "example.h"
6.我們通常會希望將C++代碼以go pkg方式呈現,所以可以將產生的.go和.a放到GOPATH中對應地方就好。但是如果要調用這個庫產生可執行檔,不能簡單的使用go build,而是要使用類似下面的編譯方式,這個是從Makefile裡面摘出來的:
#! /bin/bashgo tool 6g -I ${GOPATH}/pkg/linux_amd64/ -pack ${1}.gogo tool 6l -L ${GOPATH}/pkg/linux_amd64/ -linkmode external -extld "g++" -extldflags "-I/usr/include " -o $1 ${1}.6
上面如果不設定GOPATH的路徑,會去go的安裝目錄下面的lib/pkg中找安裝包,這不是我們希望的,我們希望所有的包控制在自己的GOPATH裡面。