热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

[开源]实现一个简易的Unity网络同步引擎——netgo

目录数据通信格式定义proto文件生成c#和golangAPI接口文件服务端网络模型客户端代码结构相关概念数据同步CustomEvent接口介绍房间相关接口Player相关接口Cu

目录

  • 数据通信格式
    • 定义proto文件
    • 生成c#和golang API接口文件
  • 服务端网络模型
  • 客户端代码结构
  • 相关概念
    • 数据同步
    • Custom Event
  • 接口介绍
    • 房间相关接口
    • Player相关接口
    • CustomEvent接口
    • View Sync
    • RPC
  • Demo演示
    • 服务端部署
    • 客户端编译安装
    • 功能测试
    • Road Map

 

正文

实现一个简易的Unity网络同步引擎Netgo

目前GOLANG有大行其道的趋势,尤其是在网络编程方面。因为和c/c++比较起来,虽然GC占用了一部分机器性能,但是出错概率小了,开发效率大大提升,而且应用其原生支持的协程很容易就能开发出高并发的服务端程序。笔者接触VR行业两年有余,接触了一些商业unity网络引擎,总觉的用的东西都落伍了,于是自己写了一个简单的引擎。目前实现了的基本功能:

  • 支持房间概念。
  • 支持灵活的数据同步方式,包括帧同步和RPC。
  • 支持自定义事件的发送。

也实现了一个简单的demo,同步效果见下图,后面会有更详细的介绍。

项目地址:https://github.com/harlanc/netgo-unity-client

下面是一个简单的项目复盘。

回到顶部

数据通信格式

数据通信格式的定义是整个项目的基石。我们这里的客户端和服务端是跨平台,跨语言通信。因此要定义一种语言无关,平台无关并且简单易用,高效不费流量的数据格式。这里我们选用了Google的 Protobuf,详细介绍参考这篇帖子。

Protobuf的C#代码库有两种选择,一种是protobuf-net,一种是protobuf-csharp-port,前者的接口书写更加符合C# 语法规范,会让人看起来更舒服一些。如果需要跨平台的话,推荐使用后者,因为不同语言的接口书写比较类似,开发起来会更容易一些。看看原作者的回复。

定义proto文件

如何使用protobuf呢,首先要书写proto文件,定义自己的结构化数据,在netgo中,下面是netgo中定义的消息体的一部分:


enum CacheOptions{AddToRoomCache = 0;
RemoveFromRoomCache = 1;
}message NGVector3{float x = 1;float y = 2;float z = 3;
}message NGQuaternion{float x = 1;float y = 2;float z = 3;float w = 4;
}message NGColor{float r = 1;float g = 2;float b = 3;float a = 4;
}

完整定义参考。

生成c#和golang API接口文件

更新好命名空间后,执行下面的命令生成API文件:

  • golang

    protoc --go_out=. *.proto

  • c#

    protoc --csharp_out=. *.proto


回到顶部

服务端网络模型

一个Unity网络同步引擎的实现包括服务端和客户端两部分。Nego 是Unity网络同步引擎的服务端,使用golang实现,充分利用了它的原生协程来实现高并发。其网络模型基于gotcp来实现。

 

参考上图,netgo会为每个socket链接建立一个协程,一个socket协程内部建立三个协程:

  • ReadLoop 用于从网络端读取数据并放入Channel中。
  • HandleLoop 用于解析应用层数据并完成相应处理,并将处理后的数据通过Channel发送给WriteLoop。
  • WriteLoop 负责将处理结果forward给其它客户端或者response给本客户端。

参考代码:


func (c *Conn) Do() {if !c.srv.callback.OnConnect(c) {return
}asyncDo(c.handleLoop, c.srv.waitGroup)
asyncDo(c.readLoop, c.srv.waitGroup)
asyncDo(c.writeLoop, c.srv.waitGroup)
}

回到顶部

客户端代码结构

写API基本上是面向用户编程,笔者以为,清晰的代码结构,好的命名方式能省掉大部分注释,代码写的乱只能靠注释来拯救,代码结构看下图:

 

按照命名空间,分为 Library,网络层和应用层(以后用户接口层会分出来).

回到顶部

相关概念

数据同步

这里的同步是指一个房间内的数据同步,一个房间内存在着来自网络上的多个终端用户,每个Client都会将房间内其它人的数据在本地做一个Clone,而数据同步是指将你自己的数据同步到其他Cient你自己的Clone上面,因此发送范围是其它用户都会接收。

数据同步分为一下两种:

  • View Sync

View Sync是毫秒级别的数据同步。可用于虚拟角色动作同步。

  • RPC

每次同步由用户手动触发。可用于换装等同步。

Custom Event

Custom Event不是向所有其它Client的Clone实体发送同步消息,而是向一个或者几个指定的Client发送消息。

回到顶部

接口介绍

房间相关接口

请求接口


//加入或者创建房间public static void JoinOrCreateRoom(string roomid,uint maxnumber)//创建房间public static void CreateRoom(string roomid, uint maxnumber)//加入房间public static void JoinRoom(string roomid)//离开房间public static void LeaveRoom()

回调接口


//创建房间成功void OnGreatedRoom();//创建房间失败void OnGreateRoomFailed(string errmsg);//加入房间成功void OnJoinedRoom();//加入房间失败void OnJoinRoomFailed();//离开房间成功void OnLeftRoom();

Player相关接口


//实例化一个物体public static void Instantiate(string prefabname, Vector3 position, Quaternion rotation, uint[] viewids)//有其它用户进入房间 void OnOtherPlayerEnteredRoom(NGPlayer player);//有其它用户离开房间void OnOtherPlayerLeftRoom(NGPlayer player);

CustomEvent接口

请求接口


//发送事件public static void SendCustomEvent(uint eventid, uint[] targetpeerids, NGAny[] customdata)

回调接口


//接收事件void OnCustomEvent(uint eventID, NGAny[] data);

View Sync

视图同步需要自己实现组件脚本,实现序列化反序列化接口,并且需要挂载到物体上:


public interface INGSerialize
{void SerializeViewComponent(NGViewStream stream);void DeserializeViewComponent(NGViewStream stream);
}public class CubeViewComponent : NGIncomingEvent, INGSerialize
{public void SerializeViewComponent(NGViewStream stream){stream.Send(this.transform.position);stream.Send(this.transform.rotation);}public void DeserializeViewComponent(NGViewStream stream){mCorrentPosition = (NGVector3)stream.Receive();mCorrentRotation = (NGQuaternion)stream.Receive();}
}

Clone实体接受数据反序列化后在Update中实时更新即可:


void Update()
{if (!view.IsMine){transform.position = mCorrentPosition;//Vector3.Lerp(transform.position, mCorrentPosition, Time.deltaTime * 5);transform.rotation = mCorrentRotation;//Quaternion.Lerp(transform.rotation, mCorrentRotation, Time.deltaTime * 5);}
}

RPC

使用RPC需要在视图脚本中写一个RPC函数:


[NGRPCMethod]
public void OnColor(NGAny[] c)
{mMat.color = c[0].NgColor;
}

调用下面的接口向其它Clone实体发送RPC调用:


public static void SendRPC(uint viewID, string methodname, RPCTarget target, params NGAny[] parameters)

有关RPC,View Sync和Custom Event 的详细使用方法 参考源码

回到顶部

Demo演示

服务端部署

Clone代码


git clone https://github.com/harlanc/netgo.git

安装依赖


go get -d ./...

更新监听端口号

打开main.go


tcpAddr, err := net.ResolveTCPAddr("tcp", "0.0.0.0:8686")

启动服务


go run main.go

客户端编译安装

客户端支持windows/MacOS/Andorid/IOS多平台。下面在Android和MacOS上测试:

配置IP和端口

 

切换Android平台

 

编译生成APK

安装APK后的初始化界面如下:

 

功能测试

两个Client进入同一个房间,每个Client会实例化出来两个Cube,一个为本机实体(Mine Cube),一个为对方的实体(Clone Cube)。

View SYnc

点击按钮Move后,会通过视图同步的方式进行postion和rotation同步。也就是文章刚开始的动图展示的样子:

RPC

点击Mine Cube之后,Cube的颜色会发生变化,同时同步到别的机器上,这里的颜色同步是通过RPC来实现的。

Custom Event

点击Clone Cube之后,会向对方实体发送消息,效果是对方的Mine Cube Scale会增加。

Road Map

接下来考虑会加入或者需要优化的功能:

  • 支持大厅功能
  • 支持负载均衡
  • 增加支持UDP等网络传输协议
  • 增加支持json等多种数据编码格式
  • View Sync数据传输优化
  • 支持跨房间Custom Event
  • .....

转自 https://www.cnblogs.com/harlanc/p/12103801.html


推荐阅读
  • 如何用UE4制作2D游戏文档——计算篇
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了如何用UE4制作2D游戏文档——计算篇相关的知识,希望对你有一定的参考价值。 ... [详细]
  • 本文介绍了计算机网络的定义和通信流程,包括客户端编译文件、二进制转换、三层路由设备等。同时,还介绍了计算机网络中常用的关键词,如MAC地址和IP地址。 ... [详细]
  • CEPH LIO iSCSI Gateway及其使用参考文档
    本文介绍了CEPH LIO iSCSI Gateway以及使用该网关的参考文档,包括Ceph Block Device、CEPH ISCSI GATEWAY、USING AN ISCSI GATEWAY等。同时提供了多个参考链接,详细介绍了CEPH LIO iSCSI Gateway的配置和使用方法。 ... [详细]
  • 本文介绍了一个适用于PHP应用快速接入TRX和TRC20数字资产的开发包,该开发包支持使用自有Tron区块链节点的应用场景,也支持基于Tron官方公共API服务的轻量级部署场景。提供的功能包括生成地址、验证地址、查询余额、交易转账、查询最新区块和查询交易信息等。详细信息可参考tron-php的Github地址:https://github.com/Fenguoz/tron-php。 ... [详细]
  • centos安装Mysql的方法及步骤详解
    本文介绍了centos安装Mysql的两种方式:rpm方式和绿色方式安装,详细介绍了安装所需的软件包以及安装过程中的注意事项,包括检查是否安装成功的方法。通过本文,读者可以了解到在centos系统上如何正确安装Mysql。 ... [详细]
  • 本文整理了315道Python基础题目及答案,帮助读者检验学习成果。文章介绍了学习Python的途径、Python与其他编程语言的对比、解释型和编译型编程语言的简述、Python解释器的种类和特点、位和字节的关系、以及至少5个PEP8规范。对于想要检验自己学习成果的读者,这些题目将是一个不错的选择。请注意,答案在视频中,本文不提供答案。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 本文介绍了如何将CIM_DateTime解析为.Net DateTime,并分享了解析过程中可能遇到的问题和解决方法。通过使用DateTime.ParseExact方法和适当的格式字符串,可以成功解析CIM_DateTime字符串。同时还提供了关于WMI和字符串格式的相关信息。 ... [详细]
  • baresip android编译、运行教程1语音通话
    本文介绍了如何在安卓平台上编译和运行baresip android,包括下载相关的sdk和ndk,修改ndk路径和输出目录,以及创建一个c++的安卓工程并将目录考到cpp下。详细步骤可参考给出的链接和文档。 ... [详细]
  • Android Studio Bumblebee | 2021.1.1(大黄蜂版本使用介绍)
    本文介绍了Android Studio Bumblebee | 2021.1.1(大黄蜂版本)的使用方法和相关知识,包括Gradle的介绍、设备管理器的配置、无线调试、新版本问题等内容。同时还提供了更新版本的下载地址和启动页面截图。 ... [详细]
  • 关于我们EMQ是一家全球领先的开源物联网基础设施软件供应商,服务新产业周期的IoT&5G、边缘计算与云计算市场,交付全球领先的开源物联网消息服务器和流处理数据 ... [详细]
  • 推荐系统遇上深度学习(十七)详解推荐系统中的常用评测指标
    原创:石晓文小小挖掘机2018-06-18笔者是一个痴迷于挖掘数据中的价值的学习人,希望在平日的工作学习中,挖掘数据的价值, ... [详细]
  • 本文探讨了C语言中指针的应用与价值,指针在C语言中具有灵活性和可变性,通过指针可以操作系统内存和控制外部I/O端口。文章介绍了指针变量和指针的指向变量的含义和用法,以及判断变量数据类型和指向变量或成员变量的类型的方法。还讨论了指针访问数组元素和下标法数组元素的等价关系,以及指针作为函数参数可以改变主调函数变量的值的特点。此外,文章还提到了指针在动态存储分配、链表创建和相关操作中的应用,以及类成员指针与外部变量的区分方法。通过本文的阐述,读者可以更好地理解和应用C语言中的指针。 ... [详细]
  • imx6ull开发板驱动MT7601U无线网卡的方法和步骤详解
    本文详细介绍了在imx6ull开发板上驱动MT7601U无线网卡的方法和步骤。首先介绍了开发环境和硬件平台,然后说明了MT7601U驱动已经集成在linux内核的linux-4.x.x/drivers/net/wireless/mediatek/mt7601u文件中。接着介绍了移植mt7601u驱动的过程,包括编译内核和配置设备驱动。最后,列举了关键词和相关信息供读者参考。 ... [详细]
  • 本文由编程笔记#小编为大家整理,主要介绍了StartingzookeeperFAILEDTOSTART相关的知识,希望对你有一定的参考价值。下载路径:https://ar ... [详细]
author-avatar
紫褚1314
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有