.NET Core使用gRPC打造服务间通信基础设施

一、什么是RPC

rpc(远程过程调用)是一个古老而新颖的名词,他几乎与http协议同时或更早诞生,也是互联网数据传输过程中非常重要的传输机制。

利用这种传输机制,不同进程(或服务)间像调用本地进程中的方法一般进行交互,而无需关心实现细节。

rpc的主要实现流程为:

640?wx_fmt=png
图片

1、客户端本地方法调用客户端stub(方法存根)。这个调用发生在客户端本地,并把调用参数推送到栈中。

2、客户端stub (方法存根)将这些参数打包,通过系统调用发送到服务器机器。打包的过程通常可以采用xml、json、二进制编码。打包的过程被称为marshalling。

3、客户端本地操作系统发送信息到目标服务器(可以通过自定义tcp协议或Http协议传输)。

4、服务器系统将信息传送到服务端stub(方法存根)

5、服务端stub (服务端方法存根) 解析信息。解析信息的过程可以称为 unmarshalling。

6、服务器stub (服务端方法存根) 调用程序,并通过类似的方式返回客户端。

为了让不同的客户端均能访问服务器,许多标准化的rpc组件往往会使用接口描述语言的形式,以便方便跨平台、跨语言的远程过程调用的实现。

图1:RPC 调用流程

参考维基百科:https://zh.wikipedia.org/wiki/%E9%81%A0%E7%A8%8B%E9%81%8E%E7%A8%8B%E8%AA%BF%E7%94%A8

二、什么时候使用RPC?

HTTP和RPC是现代微服务架构中普遍采用的两种数据传输方式,在某种场合几乎都是可以完全替换的,但又具有各自不同的特点。

1、HTTP协议是一种规范、开放、通用性非常强、标准的传输协议,几乎所有的语言都支持,如果要确保各类平台都能无缝的访问数据,可以考虑使用HTTP协议。例如目前常用的RestFul规约,定义好请求方法、数据格式并以Json的形式返回参数,能够让前后端之间的对接非常便捷;之前的开发者或许用wsdl、soap的形式比较多,也都是HTTP协议的应用。

2、RPC协议不仅仅是一种服务间传输的协议,也能使用于进程间的数据传输,它能极大的降低微服务间的通信成本,屏蔽通信细节,让调用者能够像调用本地方法一般调用远程方法。 相对而言,RPC可能无法在网页端提供支持,也并非所有的语言都实现了这种接口描述语言,让开发过程会相对繁琐,因此它的使用范围相对较小。虽然gRPC目前已经提供了web版的gRPC,但由于浏览器的兼容性等问题,也限制了他的应用。

三、什么是gRPC

gRPC可以通俗的理解为google实现的一种 RPC的形式。

参见gRPC官网的解释:

gRPC是可以在任何环境中运行的现代开源高性能RPC框架。它可以通过可插拔的支持来有效地连接数据中心内和跨数据中心的服务,以实现负载平衡,跟踪,运行状况检查和身份验证。它也适用于分布式计算的最后一英里,以将设备,移动应用程序和浏览器连接到后端服务。

它包括四个主要特点:

1.简单的服务定义:gRPC基于Protobuf协议构建,该协议提供了一个强大的二进制序列化工具集和语言定义服务。2.跨语言和平台工作:可自动为多语言或平台生成符合相应习惯的客户端和服务端存根3.快速启动并扩展:只需一行代码即可安装运行时环境和生成环境、并通过该框架可扩展到数百万rpc请求。4.双向流和集成身份验证:基于http/2的传输机制以及双向流传输和完全集成的可插入式身份验证机制。

gRPC目前广泛应用于各大互联网公司的微服务架构中,也是CNCF基金会孵化的开源基础设施组件。其官网为https://grpc.io/;开源项目地址为https://github.com/grpc/grpc。

官网提供了详细的文档说明,几乎可以开箱即用,只需简单配置就能满足你的应用需求。在开源项目中也提供了完善的各种语言实现的sample示例代码,能极大的方便开发者的使用。

在gRPC中,使用的传输协议为HTTP/2,使用的数据传输的格式为Protobuf协议。

四、什么是Protobuf

Protobuf全称为Protocal Buffers,是一种序列化协议实现,与只类似的还有thrift。这是一种与语言中立、与实现无关、可扩展的序列化数据格式,不仅仅可以用于通信协议传输过程,也同样适用于数据存储过程。它灵活高效、性能优良、更加快速和简单。在使用Protobuf的实践中,只需定义要处理数据的数据结构,就能利用Protobuf生成相关的代码。只需使用Protobuf对数据结构进行描述(IDL),即可在各种不同的语言或不同的数据流中对结构化数据进行轻松读写。

在上面的图1 RPC调用流程中,使用红色字体标注的(1)中,在客户端套接字和服务端套接字之间进行数据交换的数据传输机制就可以使用Protobuf。

Protocol Buffers最早是有谷歌发明用于解决索引服务器之间request/response协议的。通过慢慢发展发展和演进,目前已经具有了更多的特性:

•自动生成的序列化和反序列化代码避免了手动解析的需要。(官方提供自动生成代码工具,各个语言平台的基本都有)•除了用于 RPC(远程过程调用)请求之外,人们开始将 protocol buffers 用作持久存储数据的便捷自描述格式(例如,在Bigtable中)。•服务器的 RPC 接口可以先声明为协议的一部分,然后用 protocol compiler 生成基类,用户可以使用服务器接口的实际实现来覆盖它们。

由于protocal buffers诞生之初主要是为了解决服务器新旧协议之间兼容性问题,所以命名为"协议缓冲区",不过目前显然已经超出了缓冲含义的范围。而Protobuf中的术语,则使用"message"来指代在协议传输过程中定义的抽象化对象,也显然不再仅仅只是原始含义的消息所能囊括的。

五、Proto3协议简述

当我们使用Visual Studio 2019创建一个.NET Core下的gRPC项目时,可以看到,项目会自带一个Protos\Greet.proto文件,这便是gRPC使用的Protobuf的接口描述文件,通过定义这个描述文件,可以为生成对应的服务端、客户端方法存根,让方法调用过程更加简单。

1、基本的数据类型对应关系

目前最新版本的Protobuf协议为proto3协议,在这个新版的协议中,提供了以下数据类型,可以方便的对应到我们日常使用的数据类型。

640?wx_fmt=png
图片

2、关键字

1)分配字段编号

在proto协议中,每个消息定义中的字段都有唯一的编号,用来表示消息二进制格式中的字段,且使用消息类型后不应更改。可以使用的最小编号为1,最大编号为2^29^-1 或 536,870,911,但不包括 19000 到 19999(FieldDescriptor :: kFirstReservedNumber 到 FieldDescriptor :: kLastReservedNumber),因为它们是为 Protocol Buffers实现保留的。

2)重复字段(repeated)

在消息中定义重复字段(repeated 关键字),允许一个message 字段中重复数值,可以理解为数组对象。

3)保留字段(reserved)

Protobuf中提供了保留字段(reserved 关键字),如果在老版本的proto文件中定义了一些字段,而在新版本的协议中移除了这些字段,有可能出现协议文件不匹配的问题,则可以使用reserved关键字。这样当协议数据不匹配时,编译器会提示错误。

640?wx_fmt=png
图片

图2 使用保留字段时,会提示错误

3、枚举

允许在消息中定义枚举类型。也可以将枚举类型嵌套在message中。当使用枚举类型时,需要注意:

•枚举为 0 的是作为零值,当不赋值的时候,就会是零值。•为了和 proto2 兼容。在 proto2 中,零值必须是第一个值。

4、消息嵌套

在proto协议中,允许嵌套组合为更加复杂的消息。

message SearchResponse {	repeated Result results = 1;	
}	
message Result {	string url = 1;	string title = 2;	repeated string snippets = 3;	
}

5、定义服务(Services)

在proto中,如果需要对外提供接口方法,则需要使用Services。定义好services之后,protocol buffer编译器将使用所选语言生成服务接口代码和客户端与服务端方法存根。例如,

service Greeter {	// Sends a greeting	rpc SayHello (HelloRequest) returns (HelloReply);	
}

这里就定义了一个SayHello的方法存根。该方法将返回一个名称为 HelloReply 的消息。 如果需要定义无参数方法,或返回值为 void 的方法,需要使用 google.protobuf.Empty对象 ,表示传输空消息,是空的JSON对象{}并在头部的命名空间中,引用默认的协议文件

google/protobuf/empty.proto. *例如:

option csharp_namespace = "TestGRPC_Client";	
import "google/protobuf/empty.proto";	
package Greet;	
// The greeting service definition.	
service Greeter {	// Sends a greeting	rpc SayHello (HelloRequest) returns (HelloReply);	rpc Listen (google.protobuf.Empty) returns (google.protobuf.Empty);//启动监听	
}

同样,也可以使用 import 引用其他协议文件。 参考《https://github.com/halfrost/Halfrost-Field/blob/master/contents/Protocol/Protocol-buffers-encode.md》

六、什么是HTTP/2

gRPC 的客户端与服务端之间的通信机制,并没有采用TCP造轮子,而是重用了HTTP/2的传输协议。HTTP2.0是超文本传输协议HTTP的下一代协议,也是在传统开发者最为熟悉的HTTP/1.1 协议格式基础上进行的升级。 与1.1相比,他提供了新的二进制格式、多路复用机制、Header压缩、服务端推送等特色,让协议请求过程能够达到更好的性能提升。限于篇幅,这里就不再赘述了。

七、服务端开发

我们将引入一个范例,以HelloWorld作为项目名称,在这个项目中,简单介绍在NetCore中如何使用gRPC的过程、如何使用gRPC进行简单身份验证的过程。

服务端:

•并提供一个登录 login 的方法、以及其配套的用户请求参数、并返回对应的响应值。•一个 logout 的方法,该方法返回值为void空对象。

1、创建项目

在Visual Studio 2019中创建一个基于gRPC的空项目。这个项目命名为HelloWorld,放置在默认目录下。如果有需要可以开启容器支持。

640?wx_fmt=png
图片

640?wx_fmt=png
图片

640?wx_fmt=png
图片

2、项目的组成结构

当我们查看这个项目时,可以看到这是一个Asp.NET Core的项目,默认的项目模板中已经集成了Asp.NET Core和gRPC.AspNetCore的组件包。

640?wx_fmt=png
图片

•Protos文件夹

后缀名为proto的是基于proto3的协议文件。

640?wx_fmt=png
图片

•Services文件夹

项目模板创建的默认请求文件,实现了在proto文件中定义的SayHello方法,并以异步的形式返回了对象HelloReply。

640?wx_fmt=png
图片

•其他文件

Dockerfile:模板自动创建的Dockerfile文件,后期可以基于这个文件进行docker容器的构建。

Program: 程序运行的入口。

Startup: 程序启动项,定义AspNET Core项目启动所需的各种配置信息。 在UseRouting和UseEndPoints中间,加入UseAuthentication()和UseAuthorization()代码,以便为后期身份认证和授权。。

640?wx_fmt=png
图片

3、创建Proto文件

在Protos文件夹右键单击,创建一个空的记事本文件(快捷键为Ctrl+Shift+A),命名为helloworld.proto。然后再里面键入以下内容:

syntax = "proto3"	
import "google/protobuf/empty.proto"; //需要使用空参数和空返回值时,需要使用这个默认的协议文件 	
option csharp_namespace="HelloWorld";	
package Account;	
service Account{	rpc Login (LoginModel) returns (UserModel);	rpc Logout (google.protobuf.Empty) returns (google.protobuf.Empty);	
}	
message LoginModel{	string userName=1;	string userPsw=2;	
}	
message UserModel{	string NickName=1;	string Token=2;	Date LoginDate=3;	
}	
message Date{	int32 Year=1;	int32 Month=2;	int32 Day=3;	int32 Hour=4;	int32 Minute=5;	int32 Second=6;	int32 FFF=7;  	
} 

4、创建 AccountService文件

选择 Services 文件夹,并创建一个文件名为 AccountService的CSharp代码文件。并分别重载 Login 和 Logout 方法。

 public class AccountService : account.accountBase	{	public override Task<UserModel> Login(LoginModel request, ServerCallContext context)	{	return base.Login(request, context);	}	public override Task<Empty> Logout(Empty request, ServerCallContext context)	{	return base.Logout(request, context);	}	}

然后再进行代码的编写。这里我们将登录后,返回一个假的 UserModel 数据。除此之外,我们还返回了错误情况下的返回模型 BadRequest 。

public class AccountService : account.accountBase	
{	public override Task<StringData> Login(LoginModel request, ServerCallContext context)	{	if (request.UserName == "1234" && request.UserPsd == "1234")	{	var userModel = new UserModel	{	NickName = "测试用户",	Token = Guid.NewGuid().ToString(),	}; 	return Task.FromResult(new StringData() { Data = Newtonsoft.Json.JsonConvert.SerializeObject(userModel) });	}	else	{	var BadRequest = new BadRequest { ErrorCode = 1, ErrorDescription = "用户名或密码错误" };	return Task.FromResult(new StringData() { Data = Newtonsoft.Json.JsonConvert.SerializeObject(BadRequest) });	}	}	public override Task<Empty> Logout(Empty request, ServerCallContext context)	{	return Task.FromResult(new Empty());	}	
}	
public class BadRequest	
{	public int ErrorCode { get; set; }	public string ErrorDescription { get; set; }	
}

八、客户端开发

客户端,是一个基于 .NET Core 的控制台程序。在这个控制台中,我们可以实现下面功能:

•通过输入命令 1 调用登录方法;•输入命令 2 调用登出方法。

1、创建项目、引用依赖包

创建一个基于.NET Core的一个控制台程序,并使用 Nuget 安装组件包

640?wx_fmt=png
图片

2、创建协议文件

将在服务端开发中创建的 Protos 文件夹拷贝到客户端程序中。

640?wx_fmt=png
图片

并使用记事本对项目文件【HelloWorld.Client.csproj】进行编辑, 将Protobuf 文件的GrpcServices属性设置为 “Client”。

完成这些操作,编译完成,即可自动生成客户端与服务端连接的客户端方法存根。

3、编写客户端方法

创建一个单独的类文件,用来编写客户端调用方法。这个类文件名称为 AccountClientImpl。 代码如下:

using Grpc.Net.Client;	
using System;	
using System.Collections.Generic;	
using System.Text;	
using static HelloWorld.Greeter;	
using System.Threading.Tasks;	
namespace HelloWorld.Client	
{	public class AccountClientImpl	{	private readonly GrpcChannel _grpcChannel;	private readonly Account.AccountClient _accountClient;	public AccountClientImpl(GrpcChannel grpcChannel, Account.AccountClient accountClient)	{	_grpcChannel = grpcChannel;	_accountClient = accountClient;	}	public void Login()	{	var result = _accountClient.Login(new LoginModel() { UserName = "1234", UserPsd = "1234" });	Console.WriteLine(result.Data);	}	public void Logout()	{	var empty = new Google.Protobuf.WellKnownTypes.Empty();	_accountClient.Logout(empty);	}	}	}	

然后再修改 Program.cs 文件,用来调用上述方法。在这个方法中,如果输入1,则执行登录方法;输入2,则执行退出方法。

class Program	
{	static void Main(string[] args)	{	var channel = GrpcChannel.ForAddress("https://localhost:5001");	var client = new Account.AccountClient(channel);	AccountClientImpl accountClientImpl = new AccountClientImpl(channel, client);	if (Console.ReadLine() == "1")	{	accountClientImpl.Login();	}	else if (Console.ReadLine() == "2")	{	accountClientImpl.Logout(); 	} 	Console.ReadKey();	}	
}

这样就完成了我们的代码编写。 将客户端与服务端运行起来,然后在客户端代码中输入数字 1 ;即可获得我们想要的结果。

640?wx_fmt=png
图片

九、协议与项目分离

在传统的开发过程中,由于客户端和服务端需要维护两套内容完全相同的 proto 协议文件,略显臃肿,因此我们可以通过相应的手段,将对应的文件进行分离,便于后期的维护。

1、移动文件

将服务端中的Protos文件移动到上一级目录。

640?wx_fmt=png
图片

2、修改项目文件中的Proto文件

服务端修改为:

  <ItemGroup>	<Protobuf Include="..\Protos\*.proto" GrpcServices="Server" /> 	<Content Include="@(Protobuf)" LinkBase="" />	</ItemGroup> 

客户端修改为:

  <ItemGroup>	<Protobuf Include="..\Protos\*.proto" GrpcServices="Client" /> 	<Content Include="@(Protobuf)" LinkBase="" />	</ItemGroup> 

640?wx_fmt=png
图片

如果觉得这样的展示效果不太美观,也可以将proto文件移动到Protos目录下。

3、重新编译

完成协议文件分离,即可对项目进行编译。

总结

在这个教程中,我们从PRC开始讲起,简单介绍了与gRPC相关的技术栈,练习了使用 gRPC 进行服务端和客户端程序开发的全过程,希望大家能获得收获。

第一次尝试编写入门级教程,如有不足之处还请批评指正。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/313366.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

DBN训练学习-A fast Learning algorithm for deep belief nets

转载自&#xff1a;http://blog.sciencenet.cn/blog-110554-889016.html DBN的学习一般都是从Hinton的论文A Fast Learning Algorithm for Deep Belief Nets开始。如果没有相关的背景知识&#xff0c;不知道问题的来由&#xff0c;这篇论文读起来是相当费劲的。 学习过程中主…

程序员家的大闸蟹:青壳、白底、肉多、爆黄,现在是吃大闸蟹的最佳时期

其实&#xff0c;我跟大家一样&#xff0c;也是dotNET跨平台和张队长的忠实粉丝&#xff0c;也是一名程序员。上上周&#xff0c;我在dotNET跨平台的优选商城买了队长推荐人生果&#xff0c;也是第一次吃这个人生果&#xff0c;味道鲜甜、汁水也特别多&#xff0c;但由于快递的…

环形链表II

1、题目描述 给定一个链表&#xff0c;返回链表开始入环的第一个节点。 如果链表无环&#xff0c;则返回 null。 为了表示给定链表中的环&#xff0c;我们使用整数 pos 来表示链表尾连接到链表中的位置&#xff08;索引从 0 开始&#xff09;。 如果 pos 是 -1&#xff0c;则…

.NET Core Love gRPC

这篇内容主要来自Microsoft .NET团队程序经理Sourabh Shirhatti的博客文章&#xff1a;https://grpc.io/blog/grpc-on-dotnetcore/&#xff0c; .NET Core 3.0现已提供grpc的.NET 托管实现 grpc-dotnet&#xff0c; gRpc 取代WCF成为 .NET的一等公民。自2018年11月以来&#xf…

Magicodes.IE已支持通过模板导出票据

本次更新如下&#xff1a;【重构】重构HTML、PDF导出等逻辑&#xff0c;并修改IExporterByTemplate为&#xff1a;Task ExportListByTemplate(IList dataItems, string htmlTemplate null) where T : class;Task ExportByTemplate(T data, string htmlTemplate null) where T…

complementary prior

Complementary Prior 解决了多层网络中层间参数W无法计算的问题。 多层有向网络如下图&#xff0c;如果计算 W&#xff0c;我们需要知道第一个隐层的后验概率&#xff0c;那么会遇到几个问题&#xff1a;多层sigmoid网络1&#xff09;后验概率几乎不可计算&#xff0c;因为所谓…

Memcached

本文来自 58沈剑&#xff1a;https://mp.weixin.qq.com/s/zh9fq_e2BgdIeR8RKtY6Sg memcache是互联网分层架构中&#xff0c;使用最多的的KV缓存。面试的过程中&#xff0c;memcache相关的问题几乎是必问的&#xff0c;关于memcache的面试提问&#xff0c;你能回答到哪一个层次…

使用FastReport报表工具生成报表PDF文档

在我们开发某个系统的时候&#xff0c;客户总会提出一些特定的报表需求&#xff0c;固定的报表格式符合他们的业务处理需要&#xff0c;也贴合他们的工作场景&#xff0c;因此我们尽可能做出符合他们实际需要的报表&#xff0c;这样我们的系统会得到更好的认同感。本篇随笔介绍…

DXSDK_June10安装错误

今天安装DXSDK_Jun10时&#xff08;下载地址&#xff1a;http://download.microsoft.com/download/A/E/7/AE743F1F-632B-4809-87A9-AA1BB3458E31/DXSDK_Jun10.exe&#xff09;&#xff0c;出现错误Error Code:s1023 错误原因&#xff1a; 计算机上有安装过更新版的Microsoft Vi…

相交链表

1、题目描述 编写一个程序&#xff0c;找到两个单链表相交的起始节点。 如下面的两个链表&#xff1a; 在节点 c1 开始相交。 示例 1&#xff1a; 输入&#xff1a;intersectVal 8, listA [4,1,8,4,5], listB [5,0,1,8,4,5], skipA 2, skipB 3 输出&#xff1a;Refe…

Kullback-Leibler Divergence

本文转自&#xff1a;http://www.cnblogs.com/ywl925/p/3554502.html KL距离&#xff0c;是Kullback-Leibler差异&#xff08;Kullback-Leibler Divergence&#xff09;的简称&#xff0c;也叫做相对熵&#xff08;Relative Entropy&#xff09;。它衡量的是相同事件空间里的两…

Shader 坐标转换

转自&#xff1a;http://www.ownself.org/blog/2010/kong-jian-zuo-biao-zhuan-huan.html 这个比较基础了&#xff0c;不过基础最重要&#xff0c;往往应该理解透彻&#xff0c;并且反复复习。 我们知道在3D画面渲染过程中对于模型的计算的一部分被称为Transforming and Ligh…

致所有.Net者和有梦想的朋友们 - 共勉

这篇文章很早就想写的了&#xff0c;主要是人到了一定的年纪&#xff0c;就想唠叨一些看法&#xff0c;认不认可不重要&#xff0c;重要的是生活给予你的酸甜苦辣&#xff0c;你都想一吐为快。 这里主要基于多年来自己的一个行业感受和以及生活感想&#xff0c;唠叨一下工作以及…

ReLU的起源

论文参考&#xff1a;Deep Sparse Rectifier Neural Networks 网页参考&#xff1a;http://www.mamicode.com/info-detail-873243.html 起源&#xff1a;传统激活函数、脑神经元激活频率研究、稀疏激活性 传统Sigmoid系激活函数 传统神经网络中最常用的两个激活函数&…

Asp.NET Core 轻松学-项目目录和文件作用介绍

前言上一章介绍了 Asp.Net Core 的前世今生&#xff0c;并创建了一个控制台项目编译并运行成功&#xff0c;本章的内容介绍 .NETCore 的各种常用命令、Asp.Net Core MVC 项目文件目录等信息&#xff0c;通过对命令的学习和操作&#xff0c;对项目结构的认识&#xff0c;进一步理…

Dubbo 常见服务治理策略

1、Dubbo体系结构 2、Dubbo容错机制 Dubbo集群容错架构图 各节点关系&#xff1a; 这里的Invoker是Provider的一个可调用Service的抽象&#xff0c;Invoker封装了Provider地址及Service接口信息。Directory代表多个Invoker&#xff0c;可以把它看成List&#xff0c;但与Lis…

IT技术人员的35个特点,你占几个?

作者&#xff1a;zollty&#xff0c;资深程序员和架构师&#xff0c;私底下是个爱折腾的技术极客&#xff0c;架构师社区合伙人&#xff01;1、价值观对生活的一般看法&#xff1a;追求简单&#xff0c;安定&#xff0c;可控制的生活&#xff0c;但对家庭生活的责任持消极态度草…

asp.net core 腾讯验证码的接入

asp.net core 腾讯验证码的接入Intro之前使用的验证码服务是用的极验验证&#xff0c;而且是比较旧的&#xff0c;好久之前接入的&#xff0c;而且验证码服务依赖 Session&#xff0c;有点不太灵活&#xff0c;后来发现腾讯也有验证码服务&#xff0c;而且支持小程序&#xff0…

ffmpeg加环境变量

转自&#xff1a;http://blog.csdn.net/leixiaohua1020/article/details/19016469 FFMPEG是命令行工具&#xff0c;因此使用起来多少还是会有些不方便。在这记录两点方便使用FFMPEG的方法&#xff1a; 1.任何目录下都可以使用FFMPEG 问题描述&#xff1a;需要转码&#xff08;播…

.NET Core 微信公众号小程序6种获取UnionID方法,你知道哪几种?

前言获取UnionID是开发微信公众号/小程序中很有必要的一个环节&#xff0c;特别是针对一个公司拥有多个公众号小程序而推出的机制&#xff0c;实现打通账户一体化&#xff0c;用UnionID来区分多平台的唯一性。官方的解释&#xff1a;如果开发者拥有多个移动应用、网站应用、和公…