引用网上对thrift的介绍:
1 2 3 4 5 6 7 | Thrift源于大名鼎鼎的facebook之手,在2007年facebook提交Apache基金会将Thrift作为一个开源项目, 对于当时的facebook来说创造thrift是为了解决facebook系统中各系统间大数据量的传 输通信以及系统之间语言环境不同需要跨平台的特性。 所以thrift可以支持多种程序语言,例如: C++, C#, Cocoa, Erlang, Haskell, Java, Ocami, Perl, PHP, Python, Ruby, Smalltalk. 在多种不同的语言之间通信thrift可以作为二进制的高性能的通讯中间件,支持数据(对象)序列化和多种类型的RPC服务。 Thrift适用于程序对程 序静态的数据交换,需要先确定好他的数据结构,他是完全静态化的,当数据结构发生变化时,必须重新编辑IDL文件, 代码生成,再编译载入的流程,跟其他IDL工具相比较可以视为是Thrift的弱项,Thrift适用于搭建大型数据交换及存储的通用工具, 对于大型系统中的内部数据传输相对于JSON和xml无论在性能、传输大小上有明显的优势。 |
主要根据这里的教程:
这里面的有几处错误,可能是老版本的thrift。
首先下载安装thrift最新版本。编译安装步骤:
1 2 3 4 5 6 7 | sudo apt-get install libboost-dev libboost- test -dev libboost-program-options-dev libevent-dev automake libtool flex bison pkg-config g++ libssl-dev wget -c https: //dist .apache.org /repos/dist/release/thrift/0 .8.0 /thrift-0 .8.0. tar .gz tar -zvxf thrift-0.8.0. tar .gz cd thrift-0.8.0 . /configure make sudo make install |
安装好后,安装相应的python库,不然在运行python实例的时候会提示“ImportError: No module named Thrift”错误。
1 2 | cd thrift-0.8.0 /lib/py sudo python setup.py install |
ok,现在我们来生成一个thrift文件,定义一个数据类型与方法,在thrift-0.8.0文件夹下面创建hello.thrift文件:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 | namespace php hello enum SexType { MALE = 1, FEMALE = 2 } struct User { 1: string firstname, 2: string lastname, 3: i32 user_id, 4: SexType sex, 5: bool active = false , 6: optional string description } exception InvalidValueException { 1: i32 error_code, 2: string error_msg } service UserManager { void ping(), i32 add_user(1:User u) throws (1: InvalidValueException e), User get_user(1:i32 uid) throws (1: InvalidValueException e), oneway void clear_list() } |
首先我们定义了一个在php代码下的命名空间,namespace php hello。这样在生成相应的php代码后,php类名前加有hello_的前缀。
然后定义了一个性别的枚举类型SexType。
再下面就是结构体User,在定义字段的时候,每个字段前必须加一个序号,后面根数据类型。可以给字段设置默认值,如bool active = false。默认情况下所有字段都为必需,要使其为可选,可加optional修饰符。
exception InvalidValueException 定义了一个异常,当传递的参数不合法时,就抛出此异常。
接下来定义了一个服务类 UserManager,下面是一些方法。格式根我们编程定义的类似,很容易看懂。
最后一个方法加了oneway,表名在客户端调用此方法时,不用等待服务端返回就立马结束。
下面我们用thrift命名来生成相应的代码:
1 2 | thrift -r --gen php hello.thrift thrift -r --gen py hello.thrift |
这时候,在目录下面出现两个目录,gen-py和gen-php,分别为php与python的代码。
我们用python来写服务端,新建python_server.py:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 | #!/usr/bin/env python import sys sys.path.append( './gen-py' ) from hello import UserManager from hello.ttypes import * from thrift.transport import TSocket from thrift.transport import TTransport from thrift.protocol import TBinaryProtocol from thrift.server import TServer users = [] class UserManagerHandler: def __init__( self ): pass #self.log = {} def ping( self ): print 'ping()' def add_user( self , user): if user.firstname = = None : raise InvalidValueException( 1 , 'no firstname exception' ) if user.lastname = = None : raise InvalidValueException( 2 , 'no lastname exception' ) if user.user_id < = 0 : raise InvalidValueException( 3 , 'wrong user_id' ) if user.sex ! = SexType.MALE and user.sex ! = SexType.FEMALE: raise InvalidValueException( 4 , 'wrong sex id' ) print 'Processing user ' + user.firstname + ' ' + user.lastname users.append(user) print users return True def get_user( self , user_id): if user_id < 0 : raise InvalidValueException( 5 , 'wrong id' ) return users[user_id] def clear_list( self ): print 'Clearing list' print users del users [:] print users handler = UserManagerHandler() processor = UserManager.Processor(handler) transport = TSocket.TServerSocket(port = 9090 ) tfactory = TTransport.TBufferedTransportFactory() pfactory = TBinaryProtocol.TBinaryProtocolFactory() server = TServer.TSimpleServer(processor, transport, tfactory, pfactory) # You could do one of these for a multithreaded server #server = TServer.TThreadedServer(processor, transport, tfactory, pfactory) #server = TServer.TThreadPoolServer(processor, transport, tfactory, pfactory) print 'Starting the server...' server.serve() print 'done.' |
sys.path.append(‘./gen-py’) 这句话是把我们之前用thrift生成的python类代码引用进来。
然后我们运行此服务端:
1 2 | chmod +x . /python_server .py . /python_server .py |
接下来我们用php来实现客户端,当执行php页面后,php去连接python的服务端,调用相应的方法等。
在apache网站根目录下面,我们创建一个文件夹来放示例php代码。
1 2 3 4 5 6 7 8 | cd ~ /www/ mkdir thrift-php cd thrift-php cp -r ~ /thrift-0 .8.0 /lib/php/src/ . #要用到thrift的php类库 cd src mkdir packages cd packages cp -r ~ /thrift-0 .8.0 /gen-php/hello/ . #将我们之前生成的hello的php部分代码复制过来 |
然后我们在thrift-php目录下面新建hello.php文件:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 | <?php $GLOBALS[ 'THRIFT_ROOT' ] = 'src' ; require_once $GLOBALS[ 'THRIFT_ROOT' ]. '/Thrift.php' ; require_once $GLOBALS[ 'THRIFT_ROOT' ]. '/protocol/TBinaryProtocol.php' ; require_once $GLOBALS[ 'THRIFT_ROOT' ]. '/transport/TSocket.php' ; require_once $GLOBALS[ 'THRIFT_ROOT' ]. '/transport/THttpClient.php' ; require_once $GLOBALS[ 'THRIFT_ROOT' ]. '/transport/TBufferedTransport.php' ; require_once $GLOBALS[ 'THRIFT_ROOT' ]. '/packages/hello/UserManager.php' ; try { $socket = new TSocket( 'localhost' , 9090); $transport = new TBufferedTransport($socket, 1024, 1024); $protocol = new TBinaryProtocol($transport); $client = new hello_UserManagerClient($protocol); $transport-> open (); $client-> ping (); $u = new hello_User(); $u->user_id = 1; $u->firstname = 'John' ; $u->lastname = 'Smith' ; $u->sex = SexType::MALE; if ($client->add_user($u)) { echo 'user added succesfully</br>' ; } var_dump($client->get_user(0)); $client->clear_list(); $u2 = new hello_User(); $client->add_user($u2); } catch (hello_InvalidValueException $e) { echo $e->error_msg. '<br/>' ; } ?> |
现在就可以在浏览器中运行我们的php客户端了,打开 http://127.0.0.1/thrift-php/hello.php,显示:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | user added succesfully</br>object(hello_User)#7 (6) { ["firstname"]=> string(4) "John" ["lastname"]=> string(5) "Smith" ["user_id"]=> int(1) ["sex"]=> int(1) ["active"]=> bool(false) ["description"]=> NULL } no firstname exception |
在python的服务器输出:
1 2 3 4 5 6 7 | Starting the server... ping() Processing user John Smith [User(user_id=1, description=None, firstname='John', lastname='Smith', sex=1, active=False)] Clearing list [User(user_id=1, description=None, firstname='John', lastname='Smith', sex=1, active=False)] [] |
ok,最基本的hello例子就完成了。
在这里还能找到更加丰富的例子: