本文主要来实现一下自定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. 简述python的turtle绘画命令及解释

    一 基础认识 turtle库是python的标准库之一,它是一个直观有趣的图形绘制数据库,turtle(海龟)图形绘制的概念诞生1969年.它的应用十分广,而且使用简单,只要在编写python程序时写 ...

  2. 【Django】url(路由系统)

    1.单一路由对应 url(r'^index/',views.index), 2.基于正则的路由 url(r'^index/(\d*)', views.index), url(r'^manage/(?P ...

  3. Spring MVC 梳理 - 四种HandlerMapping

    总结 (1)SpringMVC导入jar包,配置web.xml 这些都是基础的操作. (2)四种HandlerMapping:DefaultAnnotationHandlerMapping;Simpl ...

  4. Apache和Tomcat 配置负载均衡(mod-proxy方式)-粘性session

    Tomcat集群配置后端Tomcat Server为支持AJP的独立服务,前端Apache配置为粘性会话(sticky-session),Tomcat不配置Cluster配置和Session复制. 配 ...

  5. Jetpack系列:应用内导航的正确使用方法

    今天小编要分享的还是Android Jetpack库的基本使用方法,本篇介绍的内容是Jetpack Navigation组件,让我们一起学习,为完成年初制定的计划而努力吧! *** 组件介绍 导航,是 ...

  6. python爬虫—— 抓取今日头条的街拍的妹子图

    AJAX 是一种用于创建快速动态网页的技术. 通过在后台与服务器进行少量数据交换,AJAX 可以使网页实现异步更新.这意味着可以在不重新加载整个网页的情况下,对网页的某部分进行更新. 近期在学习获取j ...

  7. IDEA 学习笔记之 Java项目开发深入学习(2)

    Java项目开发深入学习(2): 查找变量被用到的地方 编译当前文件 增加变量watch 注意:我使用了keymap (eclipse模板),所以很多快捷键和eclipse一样. F5单步调试进入函数 ...

  8. springmvc Controller接收前端参数的几种方式总结

    (1) 普通方式-请求参数名和Controller方法的参数一致 @Controller @RequestMapping("/param") public class TestPa ...

  9. spring5 源码深度解析----- Spring事务 是怎么通过AOP实现的?(100%理解Spring事务)

    此篇文章需要有SpringAOP基础,知道AOP底层原理可以更好的理解Spring的事务处理. 自定义标签 对于Spring中事务功能的代码分析,我们首先从配置文件开始人手,在配置文件中有这样一个配置 ...

  10. Be Nice!要善良

    [1]  It is nice to be important, but it is more important to be nice. [2]  What simple act of kindne ...