grpc套路(四)php通过grpc调用golang的grpc接口服务
如果看到这篇文章你还搞不明白为什么不直接使用 http, 而要搞 rpc?那么请看我的另一篇博客:
什么是rpc? 你会彻底搞明白为啥这个?为啥那个?如果还不懂那我也没办法了!
我们开发过程当中可能不仅涉及到一门语言,比如php java golang 在同一个项目当中都运用了,当然这可能是个非常大的项目,但是考虑到项目的可维护性三者同时出现的可能性还是比较低的,但是涉及到大型项目比如涉及到微服务架构,难免会有多门语言共存的情况发生,我们考虑是php+golang的混编模式,因为毕竟php有php的短板golang有golang的长处当然java更有java的特色!所以今天空穴来潮临时打了鸡血一样的写了这么篇稿子!
首先提个醒,grpc章节的文章都是连贯性的,不可能你看这一篇文章就能搞明白什么是rpc grpc,所以建议从本章节的第一篇文章读起!
现在php流行的框架很多,我们就选择国内使用人数最多的thinkphp5.0框架来演示
第一步:
把你在golang里面的proto后缀的文件原模原样的拿过来 放到extend目录下 至于为什么放到extend目录下下边我们会解释的
解释一下: package pdfiles.user 那么就会在User.proto文件所在的位置生成Pdfiles/User目录里面存放存根文件!
第二步:
composer.json文件当中的require当中放入:
"require": {
"grpc/grpc": "v1.30.0",
"google/protobuf": "^v3.13.0"
},
"autoload":{
"psr-4":{
"GPBMetadata\\":"GPBMetadata/",
"Hello\\":"Hello/"
}
},
需要说明的是 “google/protobuf”: “^v3.13.0”不是必须的,可以去掉,这个是为了你在没有安装php的protobuf扩展情况下,也能正常运行,这种运行方式相对效率较低。
然后执行
composer install
或者
composer update
第三步:
a.安装php的grpc扩展
用pecl方式安装
pecl install grpc
找到php.ini 加上
extension = "grpc.so"
cli模式下以及fpm模式下两种 找对php.ini php-m以及phpinfo()都能看到grpc才算成功哈!之前的博文当中讲过如何安装php扩展这里不再废话!
b.安装 protobuf 及其 php 扩展(上边我们通过composer安装了已经 这里再废话一下如何在linux下安装)
执行命令:
pecl install protobuf
找到php.ini文件添加上
extension=protobuf.so
重启服务即可 直接重启nginx就完事 不放心那就重启一遍php-fpm进程
第四步:
安装 grpc_php_plugin 插件
这一步非必须哈!
并且非常耗时 我就没搞!
第五步:
因为前边博文当中讲过windows下安装protoc编译工具 并且配置到了环境变量当中 所以这里你就可以直接执行命令:
进入到 extend目录 找到User.proto文件 然后执行:
protoc --php_out=plugins=grpc:. User.proto
这个时候你就会发现
红线标注的就是新生成的类文件!
我们第四部没有安装grpc_php_plugin 插件所以生成的pdfiles/user里面没有UserClient文件所以撒我们自己得新创建一个
第六步:
<?php
/**
* Created by PhpStorm.
* User: 胡军
* Date: 2020/8/22
* Time: 13:43
*/
//命名空间一定要对 保持和RequestUser.php里面的一致即可!
namespace Pdfiles\User;
//必须继承 \Grpc\BaseStub
class UserClient extends \Grpc\BaseStub{
//构造函数必须有哦
public function __construct($hostname, array $opts, $channel = null)
{
parent::__construct($hostname, $opts, $channel);
}
//为什么是userList() 还记得你的proto文件里面的service吗?
//service UserService{
// rpc UserList(RequestUser) returns (ResponseUser){};
//}
//如果rpc GetUsers(RequestUser1) returns (ResponseUser1){};
//那么这里你也要实现GetUsers()方法
//第一个参数是请求体 依赖注入进去就行啊 第二个是参数 第三个是其他参数 一把不传入
public function userList(\Pdfiles\User\RequestUser $argument,$metadata=[],$options=[]){
//简单模式
return $this->_simpleRequest(
//这里找到golang那边生成的.pd.go文件里面的FullMethod的值原模原样填写过来即可!
'/pdfiles.user.UserService/UserList',
//传入的请求体
$argument,
//第一个表示用什么解析 之类是 ResponseUser类来解析 命名空间路径要写对啊! decode就是解析喽
['\Pdfiles\User\ResponseUser','decode'],
//参数
$metadata,
//参数
$options
);
}
}
链接文件就创建完成了!
第七步:
调用就完事了:
public function index2(){
//第二个参数固定写法 是一种认证
$userClient = new \Pdfiles\User\UserClient("10.10.16.52:8084",[
'credentials' => \Grpc\ChannelCredentials::createInsecure()
]);
//请求体
$requestUser = new \Pdfiles\User\RequestUser();
$requestUser->setName("huxiaobai");//设置要传递的参数的值
$requestUser->setMobile("15275411187");//设置要传递的参数的值
//调用userList()方法请求体过去并等待响应
$request = $userClient->userList($requestUser)->wait();
//返回体和状态
//状态: code = 0表示链接成功了
//stdClass Object ( [metadata] => Array ( ) [code] => 0 [details] => )
list($response,$status) = $request;
//用返回体调取getUser()函数获取返回值
$ret = $response->getUser();
foreach ($ret as $k=>$user){
echo $user->getName();
}
}
整个过程就完事了!
还是想说一句并不是所有的系统和系统之间数据的读取用grpc就好 http也有http协议的好处 前边我们都讲过 是不是用还是自己考虑好哦!
源码:git@codehub.devcloud.huaweicloud.com:golang00004/php_go_grpc.git 付费!
本作品采用《CC 协议》,转载必须注明作者和本文链接
推荐文章: