Golang通過(guò)Thrift框架完美實(shí)現(xiàn)跨語(yǔ)言調(diào)用

每種語(yǔ)言都有自己最擅長(zhǎng)的領(lǐng)域,Golang 最適合的領(lǐng)域就是服務(wù)器端程序。

做為服務(wù)器端程序环葵,需要考慮性能同時(shí)也要考慮與各種語(yǔ)言之間方便的通訊。采用http協(xié)議簡(jiǎn)單宝冕,但性能不高张遭。采用TCP通訊,則需要考慮封包地梨、解包菊卷、粘包等等很多因素,而且想寫個(gè)高效的TCP服務(wù)宝剖,也很難洁闰。

其實(shí),對(duì)于此類需求万细,采用RPC(Remote Procedure Call Protocol)編程最靠譜扑眉。使用 RPC 編程被認(rèn)為是在分布式環(huán)境中運(yùn)行的客戶機(jī)和服務(wù)器應(yīng)用程序之間進(jìn)行可靠通信的最強(qiáng)大、最高效的方法之一赖钞。

Golang內(nèi)置了對(duì)RPC支持腰素,但只能適用于go語(yǔ)言程序之間調(diào)用,且貌似序列化仁烹、反序列化性能不高耸弄。如果go語(yǔ)言能使用Thrift開發(fā),那么就如虎添翼了卓缰〖瞥剩可惜砰诵,thrift雖然很早就包含了golang的代碼,但一直都存在各種問題無(wú)法正確執(zhí)行捌显,以至于GitHub上有許多大牛小牛自行實(shí)現(xiàn)的Thrift代碼茁彭,但依然各種問題……直到0.9.1版本的發(fā)布!

是的扶歪,最近理肺,Apache Thrift 0.9.1正式發(fā)布了。新版的Thrift終于對(duì)Golang提供了完美的支持善镰。經(jīng)過(guò)實(shí)驗(yàn)妹萨,服務(wù)器端、客戶端已經(jīng)完美支持跨語(yǔ)言調(diào)用炫欺,且性能乎完、尤其是內(nèi)存占用上,編譯型語(yǔ)言的特點(diǎn)展現(xiàn)出來(lái)品洛,比java版的實(shí)現(xiàn)強(qiáng)了很多树姨。

下面,我們采用golang實(shí)現(xiàn)一個(gè)Thrift的Server端和Client端程序桥状。

一帽揪、開發(fā)前準(zhǔn)備

1、安裝golang的Thrift包:

go get git.apache.org/thrift.git/lib/go/thrift

2辅斟、產(chǎn)生協(xié)議庫(kù):

這是我定義的測(cè)試用IDL转晰,為檢驗(yàn)兼容性,采用了多種數(shù)據(jù)結(jié)構(gòu):

RpcService.thrift

namespace go demo.rpc
namespace java demo.rpc

// 測(cè)試服務(wù)
service RpcService {

    // 發(fā)起遠(yuǎn)程調(diào)用
    list<string> funCall(1:i64 callTime, 2:string funCode, 3:map<string, string> paramMap),

}

3砾肺、生成開發(fā)庫(kù)

下載開發(fā)庫(kù)編譯器 http://www.apache.org/dyn/closer.cgi?path=/thrift/0.9.1/thrift-0.9.1.exe

thrift-0.9.1.exe -gen go RpcService.thrift

自動(dòng)生成出的源碼結(jié)構(gòu)如下:

image

其中 constants.go挽霉、rpc_service.go、ttypes.go 是協(xié)議庫(kù)变汪,編寫程序需要用到。rpc_service-remote.go 是自動(dòng)生成的例程蚁趁,可以不用裙盾。

二、go語(yǔ)言實(shí)現(xiàn)

1他嫡、服務(wù)器端

下面番官,我們來(lái)寫服務(wù)器端程序:

package main

import (
    "demo/rpc"
    "fmt"
    "git.apache.org/thrift.git/lib/go/thrift"
    "os"
)

const (
    NetworkAddr = "127.0.0.1:19090"
)

type RpcServiceImpl struct {
}

func (this *RpcServiceImpl) FunCall(callTime int64, funCode string, paramMap map[string]string) (r []string, err error) {
    fmt.Println("-->FunCall:", callTime, funCode, paramMap)

    for k, v := range paramMap {
        r = append(r, k+v)
    }
    return
}

func main() {
    transportFactory := thrift.NewTFramedTransportFactory(thrift.NewTTransportFactory())
    protocolFactory := thrift.NewTBinaryProtocolFactoryDefault()
    //protocolFactory := thrift.NewTCompactProtocolFactory()

    serverTransport, err := thrift.NewTServerSocket(NetworkAddr)
    if err != nil {
        fmt.Println("Error!", err)
        os.Exit(1)
    }

    handler := &RpcServiceImpl{}
    processor := rpc.NewRpcServiceProcessor(handler)

    server := thrift.NewTSimpleServer4(processor, serverTransport, transportFactory, protocolFactory)
    fmt.Println("thrift server in", NetworkAddr)
    server.Serve()
}

加空行也不過(guò)才43行,怎么樣簡(jiǎn)單吧钢属。

2徘熔、客戶端程序

package main

import (
    "demo/rpc"
    "fmt"
    "git.apache.org/thrift.git/lib/go/thrift"
    "net"
    "os"
    "time"
)

func main() {
    startTime := currentTimeMillis()
    transportFactory := thrift.NewTFramedTransportFactory(thrift.NewTTransportFactory())
    protocolFactory := thrift.NewTBinaryProtocolFactoryDefault()

    transport, err := thrift.NewTSocket(net.JoinHostPort("127.0.0.1", "19090"))
    if err != nil {
        fmt.Fprintln(os.Stderr, "error resolving address:", err)
        os.Exit(1)
    }

    useTransport := transportFactory.GetTransport(transport)
    client := rpc.NewRpcServiceClientFactory(useTransport, protocolFactory)
    if err := transport.Open(); err != nil {
        fmt.Fprintln(os.Stderr, "Error opening socket to 127.0.0.1:19090", " ", err)
        os.Exit(1)
    }
    defer transport.Close()

    for i := 0; i < 1000; i++ {
        paramMap := make(map[string]string)
        paramMap["name"] = "qinerg"
        paramMap["passwd"] = "123456"
        r1, e1 := client.FunCall(currentTimeMillis(), "login", paramMap)
        fmt.Println(i, "Call->", r1, e1)
    }

    endTime := currentTimeMillis()
    fmt.Println("Program exit. time->", endTime, startTime, (endTime - startTime))
}

// 轉(zhuǎn)換成毫秒
func currentTimeMillis() int64 {
    return time.Now().UnixNano() / 1000000
}

分別編譯,先啟動(dòng)服務(wù)器端淆党,然后在執(zhí)行客戶端程序酷师⊙攘梗可以看到控制臺(tái)正確打印出了信息,說(shuō)明調(diào)用通過(guò)山孔。

-->FunCall: 1380446325199 login map[name:qinerg passwd:123456]

三懂讯、Java版實(shí)現(xiàn)

為了驗(yàn)證跨語(yǔ)言調(diào)用,下面我們分別再用java實(shí)現(xiàn)一下服務(wù)器端和客戶端:

生成Java版開發(fā)庫(kù):

thrift-0.9.1.exe -gen java RpcService.thrift

1台颠、Java服務(wù)器版

package demo.rpc;

import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

import org.apache.thrift.TException;
import org.apache.thrift.protocol.TBinaryProtocol;
import org.apache.thrift.protocol.TBinaryProtocol.Factory;
import org.apache.thrift.server.TNonblockingServer;
import org.apache.thrift.server.TServer;
import org.apache.thrift.transport.TNonblockingServerSocket;
import org.apache.thrift.transport.TNonblockingServerTransport;
import org.apache.thrift.transport.TTransportException;

/**
 * Thrift測(cè)試服務(wù)器
 */
public class Server implements RpcService.Iface {

    public static void main(String[] as) {
        TNonblockingServerTransport serverTransport = null;
        try {
            serverTransport = new TNonblockingServerSocket(19090);
        } catch (TTransportException e) {
            e.printStackTrace();
        }

        RpcService.Processor<RpcService.Iface> processor = new RpcService.Processor<RpcService.Iface>(
                new Server());

        Factory protFactory = new TBinaryProtocol.Factory(true, true);
        //TCompactProtocol.Factory protFactory = new TCompactProtocol.Factory();

        TNonblockingServer.Args args = new TNonblockingServer.Args(
                serverTransport);
        args.processor(processor);
        args.protocolFactory(protFactory);
        TServer server = new TNonblockingServer(args);
        System.out.println("Start server on port 19090 ...");
        server.serve();
    }

    @Override
    public List<String> funCall(long callTime, String funCode,
            Map<String, String> paramMap) throws TException {
        System.out.println("-->FunCall:" + callTime + " " + funCode + " "
                + paramMap);
        List<String> retList = new ArrayList<>();

        for (Entry<String, String> entry : paramMap.entrySet()) {
            retList.add(entry.getKey() + entry.getValue());
        }

        return retList;
    }
}

2褐望、Java客戶端版

package demo.rpc;

import java.util.HashMap;
import java.util.Map;

import org.apache.thrift.TException;
import org.apache.thrift.protocol.TBinaryProtocol;
import org.apache.thrift.transport.TFramedTransport;
import org.apache.thrift.transport.TSocket;
import org.apache.thrift.transport.TTransport;

/**
 * Thrift測(cè)試客戶端
 */
public class Client {

    public static void main(String[] args) {

        long startTime = System.currentTimeMillis();
        try {
            TTransport transport = new TFramedTransport(new TSocket("localhost", 19090));

            TBinaryProtocol protocol = new TBinaryProtocol(transport);
            //TCompactProtocol protocol = new TCompactProtocol(transport);

            RpcService.Client client = new RpcService.Client(protocol);
            transport.open();

            Map<String, String> param = new HashMap<String, String>();
            param.put("name", "qinerg");
            param.put("passwd", "123456");

            for (int i = 0; i < 1000; i++) {
                System.out.println(client.funCall(System.currentTimeMillis() , "login", param));
            }

            transport.close();
        } catch (TException x) {
            x.printStackTrace();
        }
        long endTime = System.currentTimeMillis();
        System.out.println(" 本次調(diào)用完成時(shí)間:" + endTime + "   " + startTime + "  " + (endTime - startTime));
    }
}

好了,現(xiàn)在啟動(dòng)java版服務(wù)器端串前,用golang版客戶端調(diào)用瘫里,完全沒有問題。啟動(dòng)golang版服務(wù)器端程序荡碾,用java版客戶端調(diào)用谨读,同樣OK。

完美實(shí)現(xiàn)了跨語(yǔ)言調(diào)用玩荠。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末漆腌,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子阶冈,更是在濱河造成了極大的恐慌闷尿,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,816評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件女坑,死亡現(xiàn)場(chǎng)離奇詭異填具,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)匆骗,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,729評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門劳景,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人碉就,你說(shuō)我怎么就攤上這事盟广。” “怎么了瓮钥?”我有些...
    開封第一講書人閱讀 158,300評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵筋量,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我碉熄,道長(zhǎng)桨武,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,780評(píng)論 1 285
  • 正文 為了忘掉前任锈津,我火速辦了婚禮呀酸,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘琼梆。我一直安慰自己性誉,他們只是感情好窿吩,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,890評(píng)論 6 385
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著爆存,像睡著了一般。 火紅的嫁衣襯著肌膚如雪蝗砾。 梳的紋絲不亂的頭發(fā)上先较,一...
    開封第一講書人閱讀 50,084評(píng)論 1 291
  • 那天,我揣著相機(jī)與錄音悼粮,去河邊找鬼闲勺。 笑死,一個(gè)胖子當(dāng)著我的面吹牛扣猫,可吹牛的內(nèi)容都是我干的菜循。 我是一名探鬼主播,決...
    沈念sama閱讀 39,151評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼申尤,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼癌幕!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起昧穿,我...
    開封第一講書人閱讀 37,912評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤勺远,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后时鸵,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體胶逢,經(jīng)...
    沈念sama閱讀 44,355評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,666評(píng)論 2 327
  • 正文 我和宋清朗相戀三年饰潜,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了初坠。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,809評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡彭雾,死狀恐怖碟刺,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情薯酝,我是刑警寧澤南誊,帶...
    沈念sama閱讀 34,504評(píng)論 4 334
  • 正文 年R本政府宣布,位于F島的核電站蜜托,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏霉赡。R本人自食惡果不足惜橄务,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,150評(píng)論 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望穴亏。 院中可真熱鬧蜂挪,春花似錦重挑、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,882評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至严肪,卻和暖如春史煎,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背驳糯。 一陣腳步聲響...
    開封第一講書人閱讀 32,121評(píng)論 1 267
  • 我被黑心中介騙來(lái)泰國(guó)打工篇梭, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人酝枢。 一個(gè)月前我還...
    沈念sama閱讀 46,628評(píng)論 2 362
  • 正文 我出身青樓恬偷,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親帘睦。 傳聞我的和親對(duì)象是個(gè)殘疾皇子袍患,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,724評(píng)論 2 351

推薦閱讀更多精彩內(nèi)容