本文主要来实现一下自定UI(实现HUD的功能),并对Network Manger进行深入的讲解。

1)自定义manager

创建脚本CustomerUnetManger,并继承自NetworkManger,并对相关虚函数进行重载,并简单写了几个外部调用方法。完整脚本在文末,部分截图如下:

其中LogInfo为debug用,完整脚本在文末

2)UI脚本CustomerUnetMainMenu

编写脚本,实现自定义新建Server client 以及host,再次着重说一下原NetworkMangerHUD脚本中的第四个方法,即NetworkManger中的matchmaker。matchmaker通过unity提供的网络服务,建立游戏,然后通过你自己生成的exe就可以获取到建立的游戏列表,然后可以加入游戏:

首先开启matchmaker

    /// <summary>
/// UI调用,MatchMaker
/// </summary>
public void StartMatchMaker_()
{
if (!IsClientConnected() && !NetworkServer.active && matchMaker == null)
{
StartMatchMaker();
matchMode = true;
}
}

然后建立游戏

    /// <summary>
/// UI调用,创建比赛
/// </summary>
public void CreateMatch_()
{
if (matchMaker != null)
{
matchMaker.CreateMatch("LLMatch", , true, "", "", "", , , OnMatchCreate);
}
}

获取游戏列表(获取列表并UI显示时需要脚本match Button)

matchMaker.ListMatches(, , "", true, , , OnMatchList);

加入游戏

    /// <summary>
/// 加入指定的比赛
/// 对于建立比赛的player,本身已经在游戏中,所以不能再次加入(已经生成host)
/// 对于建立比赛的player,在加入其他游戏时先取消比赛(DestroyMatch)
/// 对于加入比赛的player,只能加入一次,不能重复加入(已经生成client)
/// 对于加入比赛的Player,在重复加入或者加入其他游戏之前先断开连接(DropConnection)
/// 本程序为demo,上述问题不做处理
/// </summary>
/// <param name="jointMatch"></param>
public void JoinMatch(MatchInfoSnapshot jointMatch)
{
if (matchMaker == null) StartMatchMaker(); matchMaker.JoinMatch(jointMatch.networkId, "", "", "", , , OnMatchJoined);//调用此方法时生成client
}

3)此部分到此基本结束没有太大问题,一些注意事项已经在代码中注释,至于如何退出比赛等会再下一篇LobbyManager中进行解释,下边时此工程的所有脚本文件。再次说明一下,玩家预制体等均采用上一篇。此工程为demo,所以UI简陋,可能会有不友好的操作或者逻辑bug

//——————————————————————————————脚本————————————————————————————//

using System;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.Networking;
using UnityEngine.Networking.Match; public class CustomerUnetManger : NetworkManager
{
public Action<List<MatchInfoSnapshot>> ListMatchAction; private float refreshTime = ;
private bool matchMode = false; #region SERVER CALLBACKS
public override void OnServerReady(NetworkConnection conn)
{
LogInfo.theLogger.Log("Server ready"); base.OnServerReady(conn);
} public override void OnStartServer()
{
LogInfo.theLogger.Log("Server start"); base.OnStartServer();
} public override void OnServerConnect(NetworkConnection conn)
{
LogInfo.theLogger.Log("Server connect"); base.OnServerConnect(conn);
} public override void OnServerDisconnect(NetworkConnection conn)
{
LogInfo.theLogger.Log("Server disconnect"); base.OnServerDisconnect(conn);
} public override void OnServerAddPlayer(NetworkConnection conn, short playerControllerId)
{
LogInfo.theLogger.Log("Server add player"); base.OnServerAddPlayer(conn, playerControllerId);
} public override void OnServerRemovePlayer(NetworkConnection conn, PlayerController player)
{
LogInfo.theLogger.Log("Server remove player"); base.OnServerRemovePlayer(conn, player);
} public override void OnServerError(NetworkConnection conn, int errorCode)
{
LogInfo.theLogger.Log("Server error"); base.OnServerError(conn, errorCode);
} public override void OnStartHost()
{
LogInfo.theLogger.Log("Host start"); base.OnStartHost();
} public override void OnStopHost()
{
LogInfo.theLogger.Log("Host stop"); base.OnStopHost();
} public override void OnStopServer()
{
LogInfo.theLogger.Log("Server stop"); base.OnStopServer();
}
#endregion #region CLIENT CALLBACKS
public override void OnClientConnect(NetworkConnection conn)
{
LogInfo.theLogger.Log("Client connect"); base.OnClientConnect(conn);
} public override void OnClientDisconnect(NetworkConnection conn)
{
LogInfo.theLogger.Log("Client disconnect"); base.OnClientDisconnect(conn);
} public override void OnClientError(NetworkConnection conn, int errorCode)
{
LogInfo.theLogger.Log("Client error"); base.OnClientError(conn, errorCode);
} public override void OnStartClient(NetworkClient client)
{
LogInfo.theLogger.Log("Client start"); base.OnStartClient(client);
} public override void OnStopClient()
{
LogInfo.theLogger.Log("Client stop"); base.OnStopClient();
}
#endregion #region MATCHMAKER CALLBACKS
public override void OnMatchCreate(bool success, string extendedInfo, MatchInfo matchInfo)
{
LogInfo.theLogger.Log("Match create"); //MatchCreate结束后进行OnMatchCreate回调,而在base中OnMatchCreate时会
//立马创建host(即创建一个server和client),所以当重复创建match时报错
//为避免此类问题可以在OnMatchCreate中不调用base的OnMatchCreate,通过自行创建List<MatchInfoSnapshot>变量自己存储比赛(matches)
//然后在需要创建host时,再startHost,但是一般游戏时只允许玩家创建一个游戏,创建另一个游戏时需退出当前游戏
base.OnMatchCreate(success, extendedInfo, matchInfo);
matchMaker.ListMatches(, , "", true, , , OnMatchList);
} public override void OnMatchJoined(bool success, string extendedInfo, MatchInfo matchInfo)
{
LogInfo.theLogger.Log("Match joint"); base.OnMatchJoined(success, extendedInfo, matchInfo);
} public override void OnMatchList(bool success, string extendedInfo, List<MatchInfoSnapshot> matchList)
{
LogInfo.theLogger.Log("Match list:"+ matchList.Count);//基函数的OnMatchList把matchList赋值给matches,此处matches仍为初始值 base.OnMatchList(success, extendedInfo, matchList); if(ListMatchAction!=null)
{
ListMatchAction(matchList);
}
//LogInfo.theLogger.Log("Match list:" + matches.Count);
}
#endregion #region CONTROLL METHODS
/// <summary>
/// UI调用,开启server
/// </summary>
public void StartServer_()
{
if (!IsClientConnected() && !NetworkServer.active && matchMaker == null)
{
StartServer();
}
} /// <summary>
/// UI调用,开启Host
/// </summary>
public void StartHost_()
{
if (!IsClientConnected() && !NetworkServer.active && matchMaker == null)
{
StartHost();
}
} /// <summary>
/// UI调用,开启client
/// </summary>
public void StartClient_()
{
if (!IsClientConnected() && !NetworkServer.active && matchMaker == null)
{
StartClient();
}
} /// <summary>
/// UI调用,MatchMaker
/// </summary>
public void StartMatchMaker_()
{
if (!IsClientConnected() && !NetworkServer.active && matchMaker == null)
{
StartMatchMaker();
matchMode = true;
}
} /// <summary>
/// UI调用,停止服务
/// </summary>
public void StopAllService()
{
if (NetworkServer.active || IsClientConnected())
{
StopHost();
} if (client != null && NetworkClient.active)
{
StopClient();
} if (matchMaker != null)
{
StopMatchMaker();
matchMode = false;
}
} /// <summary>
/// UI调用,创建比赛
/// </summary>
public void CreateMatch_()
{
if (matchMaker != null)
{
matchMaker.CreateMatch("LLMatch", , true, "", "", "", , , OnMatchCreate);
}
}
/// <summary>
/// 加入指定的比赛
/// 对于建立比赛的player,本身已经在游戏中,所以不能再次加入(已经生成host)
/// 对于建立比赛的player,在加入其他游戏时先取消比赛(DestroyMatch)
/// 对于加入比赛的player,只能加入一次,不能重复加入(已经生成client)
/// 对于加入比赛的Player,在重复加入或者加入其他游戏之前先断开连接(DropConnection)
/// 本程序为demo,上述问题不做处理
/// </summary>
/// <param name="jointMatch"></param>
public void JoinMatch(MatchInfoSnapshot jointMatch)
{
if (matchMaker == null) StartMatchMaker(); matchMaker.JoinMatch(jointMatch.networkId, "", "", "", , , OnMatchJoined);//调用此方法时生成client
}
#endregion private void Update()
{
//取消了实时更新列表,可以测试使用
//if(Time.time>refreshTime && matchMode)
//{
// if (matchMaker == null) StartMatchMaker(); // matchMaker.ListMatches(0, 5, "", true, 0, 0, OnMatchList); // refreshTime += 4;
//}
}
}
using System;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.Networking;
using UnityEngine.Networking.Match;
using UnityEngine.UI; public class CustomerUnetMainMenu : MonoBehaviour
{
public Button serverButton;
public Button hostButton;
public Button clinetButton;
public Button matchMakerButton;
public Button stopButton;
public Button createMatchButton; public GameObject starts;
public GameObject matches;
public CustomerUnetManger theManager;
public MatchListRefresher refresher; private bool matchCreated = false; #region BUTTON METHODS private void StartServer()
{
theManager.StartServer_();
} private void StartHost()
{
theManager.StartHost_();
} private void StartClient()
{
theManager.StartClient_();
} private void StartMatchMaker()
{
theManager.StartMatchMaker_(); matches.SetActive(true);
starts.SetActive(false);
} private void Stop()
{
theManager.StopAllService(); matches.SetActive(false);
starts.SetActive(true);
matchCreated = false;
} private void CreateMatch()
{
if(!matchCreated)
{
theManager.CreateMatch_();
matchCreated = true;
}
} #endregion private void Start()
{
serverButton.onClick.AddListener(StartServer);
hostButton.onClick.AddListener(StartHost);
clinetButton.onClick.AddListener(StartClient);
matchMakerButton.onClick.AddListener(StartMatchMaker);
stopButton.onClick.AddListener(Stop);
createMatchButton.onClick.AddListener(CreateMatch); theManager.ListMatchAction += OnListMatch;
} private void OnListMatch(List<MatchInfoSnapshot> list)
{
refresher.RefreshList(list);
}
}
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.Networking.Match; public class MatchListRefresher : MonoBehaviour
{
[SerializeField]
private GameObject matchButton; //直接清除列表的刷新的方式不好,可通过一下两种方法解决:
//1、增加本地存储matches,针对每次反馈的matches比对,只实例化增加或者销毁不存在的
//2、或者每次增加或者减少比赛,可以尝试让服务端调用每个客户端执行
public void RefreshList(List<MatchInfoSnapshot> matches)
{
ClearList(); foreach(var match in matches)
{
var button = Instantiate(matchButton, transform);
button.GetComponent<MatchButton>().Initialize(match);
}
} private void ClearList()
{
foreach(Transform t in transform)
{
Destroy(t.gameObject);
}
}
}
using UnityEngine;
using UnityEngine.Networking;
using UnityEngine.Networking.Match;
using UnityEngine.UI; public class MatchButton : MonoBehaviour
{
private MatchInfoSnapshot match;
//private NetworkManager manager; public void Initialize(MatchInfoSnapshot match)
{
this.match = match; GetComponentInChildren<Text>().text = match.name;
GetComponent<Button>().onClick.AddListener(() =>
{
//manager.matchMaker.JoinMatch(match);
FindObjectOfType<CustomerUnetManger>().JoinMatch(match);
});
}
}
using UnityEngine;
using UnityEngine.UI; public class LogInfo : MonoBehaviour
{
public bool isDebug = true;
private Text log; public static LogInfo theLogger; public void Log(string info)
{
if(isDebug)
log.text += "\r\n" + info;
} private void Clear()
{
log.text = "Log:";
} void Start ()
{
theLogger = this;
log = GetComponent<Text>();
log.text = "Log:";
GetComponentInChildren<Button>().onClick.AddListener(Clear);
} }

NetworkManager网络通讯_NetworkManager(二)的更多相关文章

  1. NetworkManager网络通讯_破产版NetworkManager(五)

    根据对NetWorkServer 以及NetworkClient的理解,编写一个简易版的NetWork Manager.惯例全部代码放在最后 (一)NetWorkServer与NetworkClien ...

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

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

  3. NetworkManager网络通讯_Example(一)

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

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

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

  5. NetworkManager网络通讯_NetworkLobbyManager(三)

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

  6. 网络通讯之Socket-Tcp(二)

    网络通讯之Socket-Tcp  分成2部分讲解: 网络通讯之Socket-Tcp(一): 1.如何理解Socket 2.Socket通信重要函数 3.Socket Tcp 调用的基本流程图 4.简单 ...

  7. 《连载 | 物联网框架ServerSuperIO教程》-4.如开发一套设备驱动,同时支持串口和网络通讯。附:将来支持Windows 10 IOT

    1.C#跨平台物联网通讯框架ServerSuperIO(SSIO)介绍 <连载 | 物联网框架ServerSuperIO教程>1.4种通讯模式机制. <连载 | 物联网框架Serve ...

  8. Windows 网络通讯开发

    Windows 网络通讯开发 一.Windows网络开发API 由于C++标准库中没有网络库,所以进行网络开发的时候要调用系统API.Windows通讯开发API包括以下几个基本函数及成员类型: 1. ...

  9. DIOCP网络通讯流程

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

随机推荐

  1. selenium--定位--CSS

    大家在使用selenium元素定位的时候,通常更多使用的是XPATH,css定位方式用得比较少 但有时候css定位方式还是有一些优势的, 优势1:一般情况下定位速度要比XPATH快 优势2:语法要比X ...

  2. 关于IDEA的maven没有artifacts的解决方法

    做如下配置即可 settings-->bulid...-->Bulid Tools --> Maven-->Improting-->选中Impor Maven proje ...

  3. SPSS基础学习方差分析—单因素分析

    为什么要进行方差分析? 单样本.两样本t检验其最终目的都是分析两组数据间是否存在显著性差异,但如果要分析多组数据间是否存在显著性差异就很困难,因此用方差分析解决这个问题:举例:t检验可以分析一个班男女 ...

  4. (java实现)单向循环链表

    什么是单向循环链表 单向循环链表基本与单向链表相同,唯一的区别就是单向循环链表的尾节点指向的不是null,而是头节点(注意:不是头指针). 因此,单向循环链表的任何节点的下一部分都不存在NULL值. ...

  5. ajax跨域问题以及解决方案

    转:https://blog.csdn.net/csdn_ds/article/category/6937392/3 在工作中,大家应该都遇到过ajax跨域问题,浏览器的错误如下: XMLHttpRe ...

  6. Kafka 异步消息也会阻塞?记一次 Dubbo 频繁超时排查过程

    线上某服务 A 调用服务 B 接口完成一次交易,一次晚上的生产变更之后,系统监控发现服务 B 接口频繁超时,后续甚至返回线程池耗尽错误 Thread pool is EXHAUSTED.因为服务 B ...

  7. 什么是线程调度器(Thread Scheduler)和时间分片(Time Slicing )?

    线程调度器是一个操作系统服务,它负责为 Runnable 状态的线程分配 CPU 时间. 一旦我们创建一个线程并启动它,它的执行便依赖于线程调度器的实现.同上一个问题,线程调度并不受到 Java 虚拟 ...

  8. 10个值得深思的_PHP_面试问题

    Q1 第一个问题关于弱类型 $str1 = 'yabadabadoo'; $str2 = 'yaba'; if (strpos($str1,$str2)) { echo "\"&q ...

  9. ElasticSearch Cardinality Aggregation聚合计算的误差

    使用ES不久,今天发现生产环境数据异常,其使用的ES版本是2.1.2,其它版本也类似.通过使用ES的HTTP API进行查询,发现得到的数据跟javaClient API 查询得到的数据不一致,于是对 ...

  10. homebrew安装问题(Failed during: git fetch origin master:refs/remotes/origin/master --tags --force)

    在mac系统中,使用homebrew可以很方便的管理包.按照官网的说明执行以下命令时总是报错: /usr/bin/ruby -e "$(curl -fsSL https://raw.gith ...