NetworkManager网络通讯_破产版NetworkManager(五)
根据对NetWorkServer 以及NetworkClient的理解,编写一个简易版的NetWork Manager。惯例全部代码放在最后
(一)NetWorkServer与NetworkClient机制
网络中一个Server对应对各client,unet NetWorkServer主要为静态方法,通过绑定ip地址以及端口就可以监听连接,client通过connect进行连接,当连接成功后通过注册的相关消息进行连接成功与否的方法进行相应处理(采用socket编程时一般通过回调俩进行处理,在unet中稍有不同,但原理一样)
1)server进行监听,并注册监听事件
代码如下:
NetworkServer.Listen(ip, port);
NetworkServer.RegisterHandler(MsgType.Connect, OnServerConnect);
NetworkServer.RegisterHandler(MsgType.Disconnect, OnServerDisconnect);//如果stopServer则不会触发此事件,只有客户端主动断开连接(stopclient)时触发
NetworkServer.RegisterHandler(MsgType.AddPlayer, OnServerAddPlayer);
NetworkServer.RegisterHandler(MsgType.RemovePlayer, OnServerRemovePlayer);
NetworkServer.RegisterHandler(CustomerMsgType.StringMsg, OnServerReceiveStringMsg);
上述RegisterHandler注册的监听事件中前四个为unet自定义消息类型,以MsgType.Connect为例,根据变量名字可知当server在MsgType.Connect注册OnServerConnect方法,当client端连接到server端时,server端会调用OnServerConnect方法。unet允许自定义消息类型,如NetworkServer.RegisterHandler(CustomerMsgType.StringMsg, OnServerReceiveStringMsg); 自定义消息类型CustomerMsgType.StringMsg(short类型)不能与unet自定义的消息类型冲突。client端要注册相对应的消息类型
2)client进行连接,并注册事件
unetClient = new NetworkClient();
RegisterClientMessage(unetClient);
unetClient.Connect(ip, port);
void RegisterClientMessage(NetworkClient client)
{
client.RegisterHandler(MsgType.Connect, OnClientConnect);
client.RegisterHandler(MsgType.Disconnect, OnClientDisconnect);//服务端主动断开或者stopServer时触发,但客户端主动断开连接(stopClient)时不会触发
client.RegisterHandler(CustomerMsgType.StringMsg, OnClientReceiveStringMsg); if (localPlayer!=null)
{
ClientScene.RegisterPrefab(localPlayer);
}
}
(二)clientScene场景管理
通过上一步,实现了Server与client的连接,一个client与server连接成功后对应一个networkconnection,通过连接绑定相应的游戏物体来进行操作则需要clientScene。clientScene组要为静态方法,统一管理连接的游戏物体。
1)当客户端接收到连接成功后在OnClientConnect创建游戏物体(可以自己手动代码创建,但是unet已经为我们组好处理,直接调用即可)
NetworkConnection conn = msg.conn;
ClientScene.Ready(conn); CustomerMsgType.StringMessage stringMsg = new CustomerMsgType.StringMessage();
stringMsg.stringMsg = "ttt";
ClientScene.AddPlayer(conn, (short)conn.connectionId, stringMsg);
2)连接成功后通过clientScene添加玩家(AddPlayer),调用后会激发服务端注册的OnServerAddPlayer方法( NetworkServer.RegisterHandler(MsgType.AddPlayer, OnServerAddPlayer);)然后生成服务端游戏物体
void OnServerAddPlayer(NetworkMessage msg)
{
//var message = msg.ReadMessage<CustomerMsgType.StringMessage>();
//Debug.Log(message.stringMsg);
var player = Instantiate(localPlayer);
msg.ReadMessage(addPlayerMsg);
//Debug.Log(Encoding.Default.GetString(addPlayerMsg.msgData));
NetworkServer.AddPlayerForConnection(msg.conn, player, addPlayerMsg.playerControllerId); LogInfo.theLogger.Log("Client "+msg.conn.connectionId+" is added");
}
(三)前两步梳理
1)服务端开启监听
2)客户端连接
3)连接后客户端向clientscene添加玩家
4)服务端接收到玩家添加成功的消息,并创建玩家
(四)消息发送
客户端服务端发送消息时,要先注册消息类型以及对应的方法,通过NetworkClient的send方法以及NetworkServer的send方进行发送。
1)注册消息
如步骤(一)中的client.RegisterHandler(CustomerMsgType.StringMsg, OnClientReceiveStringMsg)(客户端) NetworkServer.RegisterHandler(CustomerMsgType.StringMsg, OnServerReceiveStringMsg)(服务端)
2)消息的发送
public void Send(string msg)
{
CustomerMsgType.StringMessage stringMsg = new global::CustomerMsgType.StringMessage();
stringMsg.stringMsg = msg; unetClient.Send(CustomerMsgType.StringMsg, stringMsg);
}
向服务端发送消息,发送时包含发送的消息主体stringMsg(为继承MessageBase的类型)以及发送消息的类型CustomerMsgType.StringMsg
3)通过NetworkMessage进行消息读取
所有注册的消息对应的方法都含有一个NetworkMessage参数,它包含此次操作 对应的networkconnection以及传递过来的消息,消息的读取如下所示
void OnServerReceiveStringMsg(NetworkMessage msg)
{
var receivedMsg = msg.ReadMessage<CustomerMsgType.StringMessage>();
//var newMsg = "Server received:" + receivedMsg.stringMsg; LogInfo.theLogger.Log(receivedMsg.stringMsg); var message = new CustomerMsgType.StringMessage();
message.stringMsg = "Server received:" + receivedMsg.stringMsg ;
NetworkServer.SendToClient(lastClient.connectionId, CustomerMsgType.StringMsg, message);
}
服务端端向客端发送时,除了要输入发送的类型CustomerMsgType.StringMsg以及发送的消息message外还要指定发送的客户端IDlastClient.connectionId。此为消息发送的整个流程。也可以通过networkReader和networkWriter进行消息的发送与读取
//-------------------------------------------代码--------------------------------------------//
主代码:
using UnityEngine;
using UnityEngine.Networking;
using System;
using UnityEngine.Networking.NetworkSystem;
using System.Collections.Generic;
using System.Text; public class MyNetworkManager : MonoBehaviour
{
public string ip = "127.0.0.1";
public int port = ; public GameObject localPlayer; public static MyNetworkManager Instance;
public NetworkClient unetClient; bool serverStarted = false;
bool clientCreated = false; AddPlayerMessage addPlayerMsg = new AddPlayerMessage();//每个player加入游戏时的小心容器
Dictionary<NetworkConnection, string> allClients = new Dictionary<NetworkConnection, string>();
NetworkConnection lastClient;//测试用的 #region PUBLIC ATTRIBUTES
public int numPlayers
{
get
{
int numPlayers = ;
for (int i = ; i < NetworkServer.connections.Count; i++)
{
var conn = NetworkServer.connections[i];
if (conn == null)
continue; for (int ii = ; ii < conn.playerControllers.Count; ii++)
{
if (conn.playerControllers[ii].IsValid)
{
numPlayers += ;
}
}
}
return numPlayers;
}
} public int numClients
{
get
{
return NetworkServer.connections.Count;
}
} #endregion #region CALLBACKS public Action onStartServer;
public Action onStartHost;
public Action<NetworkClient> onStartClient;
public Action onStopServer;
public Action onStopClient; public Action<NetworkMessage> onServerConncetedAction;//服务端接收到client事件
public Action<NetworkMessage> onServerDisconnectedAction;//客户端断开连接时服务端事件
public Action<NetworkMessage> onClientConncetedAction;//客户端连接到server事件
public Action<NetworkMessage> onClientDisonncetedAction;//客户端连接到server事件 #endregion #region PUBLIC METHODS public void StartServer(bool isServer = true)
{
if(!serverStarted)
{
NetworkServer.Listen(ip, port);
RegisterServerMessage(); //Debug.Log("Server started...");
LogInfo.theLogger.Log("Server started..."); if(!isServer)
{
unetClient = ClientScene.ConnectLocalServer();
RegisterClientMessage(unetClient); LogInfo.theLogger.Log("Local Client Created...");
} if(onStartServer!=null)
{
onStartServer();
}
}
} public void StartClient()
{
if (!clientCreated)
{
unetClient = new NetworkClient();
RegisterClientMessage(unetClient);
unetClient.Connect(ip, port); //Debug.Log("Client connecting...");
LogInfo.theLogger.Log("Client connecting..."); if (onStartClient != null)
{
onStartClient(unetClient);
}
}
} public void StartHost()
{
StartServer(false); if(onStartHost!=null)
{
onStartHost();
}
} public void StopServer(bool isServer = true)
{
foreach(var conn in NetworkServer.connections)
{
if(conn!=null)
NetworkServer.DestroyPlayersForConnection(conn);
} NetworkServer.Shutdown(); serverStarted = false; if (onStopServer != null)
{
onStopServer();
} if (!isServer) StopClient();
} public void StopClient()
{
unetClient.Disconnect();
unetClient.Shutdown();
unetClient = null;
ClientScene.DestroyAllClientObjects(); clientCreated = false; if(onStopClient!=null)
{
onStopClient();
}
} #endregion #region PRIVATE METHODS void RegisterServerMessage()
{
NetworkServer.RegisterHandler(MsgType.Connect, OnServerConnect);
NetworkServer.RegisterHandler(MsgType.Disconnect, OnServerDisconnect);//如果stopServer则不会触发此事件,只有客户端主动断开连接(stopclient)时触发
NetworkServer.RegisterHandler(MsgType.AddPlayer, OnServerAddPlayer);
NetworkServer.RegisterHandler(MsgType.RemovePlayer, OnServerRemovePlayer); NetworkServer.RegisterHandler(CustomerMsgType.StringMsg, OnServerReceiveStringMsg);
} void RegisterClientMessage(NetworkClient client)
{
client.RegisterHandler(MsgType.Connect, OnClientConnect);
client.RegisterHandler(MsgType.Disconnect, OnClientDisconnect);//服务端主动断开或者stopServer时触发,但客户端主动断开连接(stopClient)时不会触发
client.RegisterHandler(CustomerMsgType.StringMsg, OnClientReceiveStringMsg); if (localPlayer!=null)
{
ClientScene.RegisterPrefab(localPlayer);
}
} //server端接收到client连接触发
void OnServerConnect(NetworkMessage msg)
{
NetworkConnection conn = msg.conn;
lastClient = conn; if (onServerConncetedAction != null)
{
onServerConncetedAction(msg);
} Debug.Log("Client " + conn.connectionId + " is connected...");
LogInfo.theLogger.Log("Client " + conn.connectionId + " is connected...");
} //client断开连接时,server触发
void OnServerDisconnect(NetworkMessage msg)
{
NetworkConnection conn = msg.conn;
NetworkServer.DestroyPlayersForConnection(conn);//断开所有与conn对应的游戏物体
conn.Disconnect();
conn.Dispose(); if (onServerDisconnectedAction != null)
{
onServerDisconnectedAction(msg);
} LogInfo.theLogger.Log("Client " + conn.connectionId + " is disconnected...");
} void OnServerAddPlayer(NetworkMessage msg)
{
//var message = msg.ReadMessage<CustomerMsgType.StringMessage>();
//Debug.Log(message.stringMsg);
var player = Instantiate(localPlayer);
msg.ReadMessage(addPlayerMsg);
//Debug.Log(Encoding.Default.GetString(addPlayerMsg.msgData));
NetworkServer.AddPlayerForConnection(msg.conn, player, addPlayerMsg.playerControllerId); LogInfo.theLogger.Log("Client "+msg.conn.connectionId+" is added");
} void OnServerRemovePlayer(NetworkMessage msg)
{
LogInfo.theLogger.Log("Player removded");
} void OnServerReceiveStringMsg(NetworkMessage msg)
{
var receivedMsg = msg.ReadMessage<CustomerMsgType.StringMessage>();
//var newMsg = "Server received:" + receivedMsg.stringMsg; LogInfo.theLogger.Log(receivedMsg.stringMsg); var message = new CustomerMsgType.StringMessage();
message.stringMsg = "Server received:" + receivedMsg.stringMsg ;
NetworkServer.SendToClient(lastClient.connectionId, CustomerMsgType.StringMsg, message);
} //client连接到server时触发
void OnClientConnect(NetworkMessage msg)
{
NetworkConnection conn = msg.conn;
ClientScene.Ready(conn); CustomerMsgType.StringMessage stringMsg = new CustomerMsgType.StringMessage();
stringMsg.stringMsg = "ttt";
ClientScene.AddPlayer(conn, (short)conn.connectionId, stringMsg); if (onClientConncetedAction != null)
{
onClientConncetedAction(msg);
} Debug.Log("Client is connected to server...");
LogInfo.theLogger.Log("Client is connected to server...");
} void OnClientDisconnect(NetworkMessage msg)
{
//NetworkConnection conn = msg.conn;
StopClient(); if (onClientDisonncetedAction != null)
{
onClientDisonncetedAction(msg);
} LogInfo.theLogger.Log("Client is disconnected to server...");
} void OnClientReceiveStringMsg(NetworkMessage msg)
{
var receivedMsg = msg.ReadMessage<CustomerMsgType.StringMessage>(); LogInfo.theLogger.Log(receivedMsg.stringMsg);
} #endregion private void Start()
{
Application.runInBackground = true;
Instance = this;
}
}
消息代码:
using System.Collections;
using System.Collections.Generic;
using System.IO;
using UnityEngine;
using UnityEngine.Networking; public class CustomerMsgType
{
public const short StringMsg = ;
public const short FileMsg = ; public class StringMessage : MessageBase//此类unet有定义,此处定义多此一举
{
public string stringMsg;
} public class FileMessage : MessageBase
{
public string fileType;
public byte[] fileContent;
}
}
NetworkManager网络通讯_破产版NetworkManager(五)的更多相关文章
- NetworkManager网络通讯_问题汇总(四)
此篇来填坑,有些坑是unet自身问题,而大部分则是理解不准确造成的(或者unity定义太复杂) 问题一: isLocalPlayer 值一直是false 出现场景:NetworkLobbyPlayer ...
- NetworkManager网络通讯_NetworkLobbyManager(三)
此部分可以先建立游戏大厅,然后进入游戏,此处坑甚多耗费大量时间.国内百度出来的基本没靠谱的,一些专栏作家大V也不过是基本翻译了一下用户手册(坑啊),只能通过看youtube视频以及不停的翻阅用户手册解 ...
- NetworkManager网络通讯_NetworkManager(二)
本文主要来实现一下自定UI(实现HUD的功能),并对Network Manger进行深入的讲解. 1)自定义manager 创建脚本CustomerUnetManger,并继承自NetworkMang ...
- NetworkManager网络通讯_Example(一)
---恢复内容开始--- 用户手册,范例精讲. 用户手册上给出了一个简单的范例,并指出可以以此为基础进行相开发,再次对范例进行精讲.(NetworkManager对使用unity的轻量级游戏开发有很大 ...
- NetworkManager网络通讯_networkReader/Writer(六)
unet客户端和服务端进行消息发送时可以采用上一节中方法,也可以直接用networkReader/Writer类进行发送 (一)服务端/客户端注册消息 ; m_Server.RegisterHandl ...
- Ubuntu中启用关闭Network-manager网络设置问题!
Ubuntu中启用关闭Network-manager网络设置问题! [Server版本] 在UbuntuServer版本中,因为只存有命令行模式,所以要想进行网络参数设置,只能通过修改/etc/net ...
- 容联云通讯_提供网络通话、视频通话、视频会议、云呼叫中心、IM等融合通讯能力开放平台。
容联云通讯_提供网络通话.视频通话.视频会议.云呼叫中心.IM等融合通讯能力开放平台. undefined
- 【unix网络编程第三版】阅读笔记(五):I/O复用:select和poll函数
本博文主要针对UNP一书中的第六章内容来聊聊I/O复用技术以及其在网络编程中的实现 1. I/O复用技术 I/O多路复用是指内核一旦发现进程指定的一个或者多个I/O条件准备就绪,它就通知该进程.I/O ...
- DIOCP网络通讯流程
DIOCP 运作核心探密 原文连接: http://blog.qdac.cc/?p=2362 原作者: BB 天地弦的DIOCP早已经广为人知了,有很多的同学都用上了它,甚至各种变异.修改版本也出 ...
随机推荐
- 推荐5款自学手机APP,请低调收藏,让你变得越来越优秀
现在的手机APP真的是太多了,但里面的功能同类性又非常大,很难找到实用并且符合要求的APP.接下来就为小伙伴们推荐5款非常实用的APP软件,保证你会爱不释手,轻松秒变手机达人. 1.清爽视频编辑器 一 ...
- 基于WeChat的消息存储备份、远程控制、小功能项目开源分享计划
WeChat+ 关于该项目 起源 该项目的起因是一个比较程(老)序(油)猿(条)的理由,有一天我发现我下班时间比较早,有点尴尬,但是又不想没事干还坐在公司,那么如何解决我的问题呢,初步想法是远程控制电 ...
- Flask基础(10)-->http的无状态协议解决办法一(客户端cookie)
http的无状态协议 http是一种无状态协议,浏览器请求服务器时无状态的 什么是无状态? 无状态:指的是一次用户请求时,浏览器.服务器无法知道之前这个用户做过什么,每次请求都是一次新的请求. 无状态 ...
- sklearn 标准化数据的方法
Sklearn 标准化数据 from __future__ import print_function from sklearn import preprocessing import numpy a ...
- .net core中使用Bumblebee架设微服务网关
Bumblebee是款基于.net core开发开源的http服务网关,经过最近版本的完善在功能足以满足作为微服务网关的需要.在微服务网关功能中它提供了应用服务负载,故障迁移,安全控制,监控跟踪和日志 ...
- uC/OS-III 时间管理(二)
时间管理就是一种建立在时钟节拍上,对操作系统任务的运行实现时间上管理的一种系统内核机制. 常用以下五个函数: OSTimeDly() OSTimeDlyHMSM() OSTimeDlyResume() ...
- sort,uniq,wc,tr
sort (选项)(参数) sort是用来排序数据的. 以下面这个文本为例 [root@bogon ~]# cat a 123 4567 88 abc aaa AAA jk 777 777 ...
- Linux——基本命令
目录 一.目录切换命令 二.目录操作命令(增删改查) 2.1增加目录 2.2查看目录 2.3寻找目录(搜索) 2.4修改目录名称 2.5移动目录位置(剪切) 2.6拷贝目录 2.7删除目录 三.文件的 ...
- Akka实现WordCount(Scala)
Akka实现WordCount(Scala): 架构图: 项目结构: pom.xml <project xmlns="http://maven.apache.org/POM/4.0.0 ...
- IDEA 学习笔记之 Console显示日志大小
Console显示日志大小: IntelliJ IDEA默认的Output输出缓存区大小只有1024KB,超过大小限制的就会被清除,而且还会显示[too much output to process] ...