百摩网
当前位置: 首页 生活百科

grpc耗时长(gRPC如何节省您的开发时间)

时间:2023-07-07 作者: 小编 阅读量: 1 栏目名: 生活百科

此时,您应该已经听说过"gRPC"。在本文中,我将着重介绍采用gRPC作为微服务之间的通信介质的好处。开发人员之间需要非常严格的协议。RPC代表远程过程调用。而且,如果不同的服务也需要使用服务B,则开发人员B将不得不花费大量时间来使用不同的语言来制作RPC客户端,并且必须对其进行维护。gRPC使用protobuf作为定义数据结构和服务的语言。·集中式代码库·自动生成代码·服务用户(客户)可以通过软件包管理器使用生成的代码,例如。

此时,您应该已经听说过" gRPC"(标题中至少一次)。 在本文中,我将着重介绍采用gRPC作为微服务之间的通信介质的好处。

首先,我将尝试简要介绍一下架构演变的历史。 其次,我将重点介绍使用REST(作为媒介)和可能出现的问题。 第三,gRPC启动。最后,我将以我的开发工作流程为例。

架构发展简史

本节将列出并讨论每种体系结构的优缺点(着重于基于Web的应用程序)

整体式

一切都在一个包中。

优点:

· 容易上手

· 单一代码库可满足所有需求

缺点:

· 难以扩展(部分)

· 加载服务器(服务器端渲染)

· 不良的用户体验(加载时间长)

· 难以扩展的开发团队

Monolith architecture

Inside monolith architecture

Monolith v2(前端-后端)

前端逻辑和后端逻辑之间的清晰分隔。 后端仍然庞大。

优点:

· 可以将团队分为前端和后端

· 更好的用户体验(客户端的前端逻辑(应用程序))

缺点:

· [仍然]难以扩展(部分)

· [仍然]难以扩展的开发团队

Frontend-Backend architecture

微服务

每一件事物一项服务(包)。 使用网络在每个软件包之间进行通信。

优点:

· 可扩展的组件

· 可扩展团队

· 灵活的语言选择(如果使用标准通讯方式)

· 独立部署/修复每个软件包

缺点:

· 介绍网络问题(通信之间的等待时间)

· 服务之间进行通信所需的文档,协议

· 如果使用共享数据库,则难以识别错误

Micro-service architecture with shared database

Micro-service architecture with standalone database per service

REST(作为媒介)和可能出现的问题

REST(基于HTTP的JSON)由于易于使用,是当前服务之间通信的最流行方式。 使用REST使您可以灵活地为每种服务使用任何语言。

Typical REST call

但是,灵活性会带来一些陷阱。 开发人员之间需要非常严格的协议。 下面的草图展示了一个非常常见的场景,通常在开发过程中发生。

Developer A want Developer B to make a service

Bad request

Expectation vs Actual

问题:

· 依靠人类的同意

· 依赖文档(需要维护/更新)

· 从协议到协议(这两种服务)都需要大量的"格式化,解析"

· 大多数开发时间都花在了协议和格式化上,而不是业务逻辑上

gRPC启动

gRPC是可以在任何环境中运行的现代开源高性能RPC框架。

什么是RPC? RPC代表远程过程调用。 它是一种协议,一个程序可用于从网络上另一台计算机上的程序请求服务,而无需了解网络的详细信息。

Remote Procedure Call

以REST为媒介的RPC

使用服务创建者提供的RPC客户端/库将确保调用服务时的正确性。 如果我们要使用RPC和REST作为媒介,则开发人员B必须编写客户端代码供开发人员A使用。 如果两个开发人员都使用不同的选择语言,那么这对开发人员B来说是一个主要问题,因为他需要用他不习惯的另一种语言来编写PRC客户。 而且,如果不同的服务也需要使用服务B,则开发人员B将不得不花费大量时间来使用不同的语言来制作RPC客户端,并且必须对其进行维护。

原虫?

协议缓冲区是Google的语言无关,平台无关的可扩展机制,用于序列化结构化数据。 gRPC使用protobuf作为定义数据结构和服务的语言。 您可以将其与REST服务的严格文档进行比较。 Protobuf语法非常严格,因此机器可以进行编译。

下面的代码块是一个简单的原始文件,描述了一个简单的待办事项服务以及用于通信的数据结构。

用于定义数据结构的" message"关键字

用于定义服务的" service"关键字

" rpc"关键字,用于定义服务功能

syntax = "proto3";package gogrpcspec;message Employee {string name = 1;}message Task {Employee employee = 1;string name = 2;string status = 3;}message Summary {int32 todoTasks = 1;int32 doingTasks = 2;int32 doneTasks = 3;}message SpecificSummary {Employee employee = 1;Summary summary = 2;}service TaskManager {rpc GetSummary(Employee) returns (SpecificSummary) {}rpc AddTask(Task) returns (SpecificSummary) {}rpc AddTasks(stream Task) returns(Summary) {}rpc GetTasks(Employee) returns (stream Task) {}rpc ChangeToDone(stream Task) returns (stream Task) {}}

将原始代码编译为服务器代码

由于protobuf非常严格,因此我们可以使用" protoc"将proto文件编译为服务器代码。 编译后,您需要对其实施真实的逻辑。

protoc --go_out=plugins=grpc:. ${pwd}/proto/*.proto

\--proto_path=${pwd}

编译原始代码到客户端代码

有了proto文件,我们可以使用" protoc"将其客户端代码编译为许多流行的语言:C#,C,Dart,Go,Java,javascript,Objective-C,PHP,Python,Ruby等。

gRPC rpc类型

gRPC支持多种rpc类型(不过,在本文中我不会强调)

· 一元RPC(请求-响应)

· 客户端流式RPC

· 服务器流式RPC

· 双向流式RPC

开发流程

为了在各个团队之间采用gRPC,我们需要一些东西。

· 集中式代码库(用于服务之间通信的gRPC规范)

· 自动生成代码

· 服务用户(客户)可以通过软件包管理器使用生成的代码(用于他们选择的语言),例如。 去获取/点安装

此示例的代码可以在此仓库中找到:

- https://github.com/redcranetech/grpcspec-example

- https://github.com/redcranetech/grpc-go-example

- https://github.com/redcranetech/grpc-python-example

代码库的结构

.├── HISTORY.md├── Makefile├── README.md├── genpyinit.sh├── gogrpcspec //go generated code here│ └── ...├── proto│ └── todo.proto├── pygrpcspec //python generated code here│ ├── ...└── setup.py

git钩子

我将设置githook,以便在提交之前自动生成内容。 如果合适,您可以使用CI(drone / gitlab / jenkins /…)。 (使用githook的缺点是每个开发人员都需要先配置githook)

您需要一个目录(文件夹)来保留预提交脚本。 我称之为" .githooks"

$ mkdir .githooks$ cd .githooks/$ cat <<EOF > pre-commit#!/bin/shset -emake generategit add gogrpcspec pygrpcspecEOF$ chomdx pre-commit

预提交脚本将触发Makefile并git添加2个目录(gogrpcsepc,pygrpcspec)

为了使githooks正常工作,开发人员必须运行以下git config命令:

$ git config core.hooksPath .githooks

我们将此命令添加到Makefile中,以使开发人员可以轻松地运行此命令(称为" make init")。 Makefile的内容应如下所示。

# content of: Makefileinit: git config core.hooksPath .githooksgenerate: # TO BE CONTINUE

产生程式码

我们已经设置了githooks来运行Makefile(" make generate")。 让我们深入了解将自动生成代码的命令。 本文将重点介绍两种语言-go,python

生成go代码

我们可以使用protoc将.proto文件编译成go代码。

protoc --go_out=plugins=grpc:. ${pwd}/proto/*.proto \--proto_path=${pwd}

我们将改为通过docker使用protoc(为了便于开发人员使用)

docker run --rm -v ${CURDIR}:${CURDIR} -w ${CURDIR} \

znly/protoc \

--go_out=plugins=grpc:. \

${CURDIR}/proto/*.proto \

--proto_path=${CURDIR}

看一下下面的generate命令(我们将删除,生成并将代码移动到适当的文件夹中)

# content of: Makefileinit:git config core.hooksPath .githooksgenerate:# remove previously generated coderm -rf gogrpcspec/*# generate go codedocker run --rm -v ${CURDIR}:${CURDIR} -w ${CURDIR} \znly/protoc \--go_out=plugins=grpc:. \${CURDIR}/proto/*.proto \--proto_path=${CURDIR}# move generated code into gogrpcspec foldermv proto/*.go gogrpcspec

生成代码后,希望将代码用于服务器或客户端的存根以调用服务的用户(开发人员)可以使用go get命令下载

go get -u github.com/redcranetech/grpcspec-example

然后用

import pb "github.com/redcranetech/grpcspec-example/gogrpcspec"

生成python代码

我们可以使用protoc将.proto文件编译成python代码。

protoc --plugin=protoc-gen-grpc=/usr/bin/grpc_python_plugin \

--python_out=./pygrpcspec \

--grpc_out=./pygrpcspec \

${pwd}/proto/*.proto \

--proto_path=${pwd}

我们将改为通过docker使用protoc(为了便于开发人员使用)

docker run --rm -v ${CURDIR}:${CURDIR} -w ${CURDIR} \

znly/protoc \ --plugin=protoc-gen-grpc=/usr/bin/grpc_python_plugin \

--python_out=./pygrpcspec \

--grpc_out=./pygrpcspec \

${CURDIR}/proto/*.proto \

--proto_path=${CURDIR}

为了使生成的代码进入python包以通过pip安装,我们需要执行额外的步骤:

· 创建setup.py

· 修改生成的代码(生成的代码使用文件夹名称导入,但我们将其更改为相对名称)

· 文件夹需要包含" init.py",以暴露生成的代码

使用以下模板创建setup.py文件:

# content of: setup.pyfrom setuptools import setup, find_packageswith open('README.md') as readme_file:README = readme_file.read()with open('HISTORY.md') as history_file:HISTORY = history_file.read()setup_args = dict(name='pygrpcspec',version='0.0.1',description='grpc spec',long_description_content_type="text/markdown",long_description=README'\n\n'HISTORY,license='MIT',packages=['pygrpcspec','pygrpcspec.proto'],author='Napon Mekavuthikul',author_email='napon@redcranetech.com',keywords=['grpc'],url='https://github.com/redcranetech/grpcspec-example',download_url='')install_requires = ['grpcio>=1.21.0','grpcio-tools>=1.21.0','protobuf>=3.8.0']if __name__ == '__main__':setup(**setup_args, install_requires=install_requires)

产生__init__.py

pygrpcspec文件夹的__init__.py必须是

# content of: pygrpspec/__init__.pyfrom . import proto__all__ = [ 'proto']

并且pygrpcspec / proto文件夹的__init__.py必须是

# content of: pygrpspec/proto/__init__.pyfrom . import todo_pb2from . import todo_pb2_grpc__all__ = ['todo_pb2','todo_pb2_grpc',]

为了使开发人员能够添加更多.proto文件并自动生成__init__.py,一个简单的shell脚本可以解决此问题

# content of: genpyinit.shcat <<EOF >pygrpcspec/__init__.pyfrom . import proto__all__ = [ 'proto']EOFpyfiles=($(ls pygrpcspec/proto | sed -e 's/\..*$//'| grep -v __init__))rm -f pygrpcspec/proto/__init__.pyfor i in "${pyfiles[@]}"do echo "from . import $i" >> pygrpcspec/proto/__init__.pydoneecho "__all__ = [" >> pygrpcspec/proto/__init__.pyfor i in "${pyfiles[@]}"do echo "'$i'," >> pygrpcspec/proto/__init__.pydoneecho "]" >> pygrpcspec/proto/__init__.py

修改生成的代码

(如果您不太熟悉python模块,则可以跳过此阅读)

我们希望将每个"从原始导入"更改为"从"。 进口"。 这背后的原因是因为我们将数据类型,服务存根都放在同一目录中,并且为了在模块外部调用模块,每个内部引用都应该是相对的。

sed -i -E 's/^from proto import/from . import/g' *.py

此时,您的Makefile应该如下所示:

# content of: Makefileinit:git config core.hooksPath .githooksgenerate:# remove previously generated coderm -rf gogrpcspec/*# generate go codedocker run --rm -v ${CURDIR}:${CURDIR} -w ${CURDIR} \znly/protoc \--go_out=plugins=grpc:. \${CURDIR}/proto/*.proto \--proto_path=${CURDIR}# move generated code into gogrpcspec foldermv proto/*.go gogrpcspec# remove previously generated coderm -rf pygrpcspec/*# generate python codedocker run --rm -v ${CURDIR}:${CURDIR} -w ${CURDIR} \znly/protoc \--plugin=protoc-gen-grpc=/usr/bin/grpc_python_plugin \--python_out=./pygrpcspec \--grpc_out=./pygrpcspec \${CURDIR}/proto/*.proto \--proto_path=${CURDIR}# generate __init__.pysh genpyinit.sh# modify import using seddocker run --rm -v ${CURDIR}:${CURDIR} -w${CURDIR}/pygrpcspec/proto \ frolvlad/alpine-bash \ bash -c "sed -i -E 's/^from proto import/from . import/g' *.py"

生成代码后,希望将代码用于服务器或客户端的存根以调用服务的用户(开发人员)可以使用pip命令下载

pip install -e git https://github.com/redcranetech/grpcspec-example.git#egg=pygrpcspec

然后用

from pygrpcspec.proto import todo_pb2_grpcfrom pygrpcspec.proto import todo_pb2

综上所述,由于protobuf的语法严格性可以将gRPC编译成多种不同语言的客户端代码,因此gRPC是在微服务之间进行通信的一种绝佳方式。

All codes in this article:

- https://github.com/redcranetech/grpcspec-example

- https://github.com/redcranetech/grpc-go-example

- https://github.com/redcranetech/grpc-python-example

(本文翻译自Napon Mekavuthikul的文章《gRPC and why it can save you development time》,参考:https://medium.com/red-crane/grpc-and-why-it-can-save-you-development-time-436168fd0cbc)

    推荐阅读
  • 青椰子要如何挑选呢(青椰子怎样挑选)

    下面内容希望能帮助到你,我们来一起看看吧!青椰子要如何挑选呢带皮椰青选择外皮清脆、颜色深绿、顶端三棱较为坚实的。这样的带皮椰青更新鲜,此外个头较大、拿起来较重、摇一摇比较沉的水分更足。不带皮椰青选择相同大小中比较重、椰尾呈白色的为佳,若尾部为棕色、褐色,那么椰子的口感会比较差。把椰青摇一摇,没水声的是好椰青,有水声说明放很久不新鲜了。

  • 句式类型有几种(句子分类有几种类型)

    以下内容希望对你有帮助!选择问句列出两项或几项,让人选择其中一项,常用关联词语“还是”“是……还是……”反问句是无疑问句,用否定的形式表示肯定的意思,用肯定的形式表示否定的意思,不要求回答。反复问句用肯定和否定相叠的方式提问。祈使句常常是非主谓句。句末一般用句号或感叹号。

  • 央企是什么意思(央企意思是什么)

    我们一起去了解并探讨一下这个问题吧!因为,所谓“央企”是指直属于国务院有关部门的国有企业。国企是全国所有制企业的简称,包括央企在内。央企是中央直属企业的简称,全国目前央企一共157家,都是各行业的领军企业,例如中石化、中石油、中航等。

  • 金字塔竟然是这样建造的(金字塔究竟是有多神奇)

    最轻的石块1.5吨,平均重量约2.5吨,总重量约684.8万吨。这也是多数人认为金字塔并非是古代埃及人所建造的,而是外星人建造出来的。这已经令现代人叹为观止了,但是更令人吃惊的是胡夫金字塔上的数字巧合。甚至于之前美国公布了一张火星上的照片,其表面有一些疑似金字塔特征的物体。目前并没有真正的定论,但是较为普遍的说法是金字塔是人为建造的,但是所耗费的时间非常长。

  • cad全部文字改大小(五种更改CAD中的文字大小的方法)

    浩辰CAD中调用缩放命令有三种方法:菜单位置:[修改]→[缩放]工具条:[修改]→[比例]命令行:SCALE下面我们来看看怎么使用缩放命令来修改文字的大小。

  • 老鸭粉丝汤家庭做法(家庭版老鸭粉丝汤做法)

    大家好,这里是,关注老刘,每天分享一道好吃又实用的家常菜1、老鸭汤是一道安徽沿江的特色传统名菜,制作原料主要有活老鸭、精盐等。热锅倒入菜籽油,放入姜片爆香,放入鸭肉炒干水分5、炒干水分以后加入一勺黄豆酱,放入漫过食材的凉水,大火烧开,把鸭肉连同汤汁一起倒入砂锅中,锅中淋入少许料酒,盖上锅盖大火烧开,中小火慢炖三十分钟。

  • 茴香苗饺子馅的做法(茴香苗饺子馅的做法大全)

    6、还是顺着一个方向搅,搅好静置一会。放进肉馅里,并且拌匀。

  • 颜真卿对唐代书法的历史意义(唐代楷书四大家之首颜真卿的悲惨结局)

    颜真卿与柳公权并称“颜柳”,被称为“颜筋柳骨”。安史之乱时,颜真卿率义军对抗叛军,一度光复河北。唐德宗派兵讨伐的结果,叛乱不但没有平定,反而蔓延开来了。五镇叛乱,使朝廷大为震惊。李勉认为失去一个朝廷元老,会给朝廷留下羞耻,秘密上表给皇帝,坚决要求留下真卿。李希烈请来他的同党,设盛会,唤来真卿,并指使戏子们借唱戏攻击和侮辱朝廷。于是宦官勒死了真卿,时年七十六岁。

  • 男士带手串十大禁忌(关于男性带手串的禁忌)

    讲究颗数,男士戴手串需要讲究颗数,一般108颗的手串是比较好的,其他颗数需要根据具体定义。任由他人触碰,建议男士戴手串的时候不要任由他人触碰,否则可能会降低手串的福气。手串带入卫生间,卫生间是不干净的地方,将手串带入卫生间会污染手串,从而降低手串的能量。无视佛塔,男士戴手串的时候一定要把佛塔向上,这样才能表示尊敬的意思。沾水,男士戴手串的时候最好不要沾水,否则长期浸泡手串之后会引起干裂甚至是变质。

  • 彝族怎么读(彝族的读音)

    接下来我们就一起去研究一下吧!彝族怎么读彝族读作yízú。彝族,是中国第六大少数民族,民族语言为彝语,属汉藏语系藏缅语族彝语支,有北部、东部、南部、东南部、西部、中部6种方言,其中包括5个次方言,25个土语。彝族主要分布在滇、川、黔、桂四省(区)的高原与沿海丘陵之间,主要聚集在楚雄、红河、凉山、毕节、六盘水和安顺等地,凉山彝族自治州是全国最大的彝族聚居区。