根据对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(五)的更多相关文章

  1. NetworkManager网络通讯_问题汇总(四)

    此篇来填坑,有些坑是unet自身问题,而大部分则是理解不准确造成的(或者unity定义太复杂) 问题一: isLocalPlayer 值一直是false 出现场景:NetworkLobbyPlayer ...

  2. NetworkManager网络通讯_NetworkLobbyManager(三)

    此部分可以先建立游戏大厅,然后进入游戏,此处坑甚多耗费大量时间.国内百度出来的基本没靠谱的,一些专栏作家大V也不过是基本翻译了一下用户手册(坑啊),只能通过看youtube视频以及不停的翻阅用户手册解 ...

  3. NetworkManager网络通讯_NetworkManager(二)

    本文主要来实现一下自定UI(实现HUD的功能),并对Network Manger进行深入的讲解. 1)自定义manager 创建脚本CustomerUnetManger,并继承自NetworkMang ...

  4. NetworkManager网络通讯_Example(一)

    ---恢复内容开始--- 用户手册,范例精讲. 用户手册上给出了一个简单的范例,并指出可以以此为基础进行相开发,再次对范例进行精讲.(NetworkManager对使用unity的轻量级游戏开发有很大 ...

  5. NetworkManager网络通讯_networkReader/Writer(六)

    unet客户端和服务端进行消息发送时可以采用上一节中方法,也可以直接用networkReader/Writer类进行发送 (一)服务端/客户端注册消息 ; m_Server.RegisterHandl ...

  6. Ubuntu中启用关闭Network-manager网络设置问题!

    Ubuntu中启用关闭Network-manager网络设置问题! [Server版本] 在UbuntuServer版本中,因为只存有命令行模式,所以要想进行网络参数设置,只能通过修改/etc/net ...

  7. 容联云通讯_提供网络通话、视频通话、视频会议、云呼叫中心、IM等融合通讯能力开放平台。

    容联云通讯_提供网络通话.视频通话.视频会议.云呼叫中心.IM等融合通讯能力开放平台. undefined

  8. 【unix网络编程第三版】阅读笔记(五):I/O复用:select和poll函数

    本博文主要针对UNP一书中的第六章内容来聊聊I/O复用技术以及其在网络编程中的实现 1. I/O复用技术 I/O多路复用是指内核一旦发现进程指定的一个或者多个I/O条件准备就绪,它就通知该进程.I/O ...

  9. DIOCP网络通讯流程

    DIOCP 运作核心探密   原文连接: http://blog.qdac.cc/?p=2362 原作者: BB 天地弦的DIOCP早已经广为人知了,有很多的同学都用上了它,甚至各种变异.修改版本也出 ...

随机推荐

  1. web前端开发面试题(附答案)-2

    1.label是什么标签,有什么作用?和for属性使用的作用? label标签来定义表单控制间的关系,当用户选择该标签时,浏览器会自动将焦点转到和标签相关的表单控件上. label 元素不会向用户呈现 ...

  2. 基于API和SQL的基本操作【DataFrame】

    写在前面: 当得到一个DataFrame对象之后,可以使用对象提供的各种API方法进行直接调用,进行数据的处理. // =====基于dataframe的API=======之后的就都是DataFra ...

  3. linux 进程消耗查看

    Linux下如何查看哪些进程占用的CPU内存资源最多 linux下获取占用CPU资源最多的10个进程,可以使用如下命令组合: ps aux|head -1;ps aux|grep -v PID|sor ...

  4. 对于java的Sting.intern()的一些注意

    今天翻看书时遇到了这样一个问题,对于String.intern()方法又有了一些认识和看法.首先我们看它的api 大意就是intern()方法会在常量池中记录首次出现的实例引用,但是在jdk1.6中却 ...

  5. json与java对象的转换,以及struts2对json的支持,实现ajax技术

    这两天学的东西有点多,今天抽个时间写下来,以此作为激励,这两天学了json,ajax,jQuery 一.使用第三方的工具java转换为json类型 首先就是java类型转换为json对象,首先要导入第 ...

  6. 使用Hexo开源博客系统,轻松搭建你的个人博客(2)- 配置篇

    上一章节,我们介绍了Hexo的基础搭建,搭建完大家一定发现,是英文版本的,并且页面有点丑陋.这一章节,就来跟大家介绍Hexo的配置和主题的设置. 站点信息 上一章有跟大家提到过_config.yml这 ...

  7. Python IAQ中文版 - Python中少有人回答的问题

    Python中少有人回答的问题 The Python IAQ: Infrequently Answered Questions 1 Q: 什么是"少有人回答的问题(Infrequently ...

  8. ThinkPHP5实现定时任务

    ThinkPHP5实现定时任务 最近使用ThinkPHP5做了个项目,项目中需要定时任务的功能,感觉有必要分享下 TP5做定时任务使用到command.php的 步骤如下: 1.配置command.p ...

  9. Scala 异常处理

    Scala 异常处理: parseURL("www.baidu.com") 会返回一个 Success[URL] ,包含了解析后的网址, 反之 parseULR("www ...

  10. Nginx简单介绍以及linux下使用Nginx进行负载均衡的搭建

    1.Nginx简介 Nginx是一款高性能的http 服务器/反向代理服务器及电子邮件(IMAP/POP3)代理服务器.由俄罗斯的程序设计师Igor Sysoev所开发,官方测试nginx能够支支撑5 ...