UDP实现一个简易的聊天室 (Unity&&C#完成)
效果展示(尚未完善)

UDP
User Data Protocol 用户数据报协议
概述
UDP是不连接的数据报模式。即传输数据之前源端和终端不建立连接。使用尽最大努力交付原则,即不保证可靠交付。
数据报模式:由于不建立连接,收到的数据可能是任意主机发送的,所以接收端Read次数必须与发送端Write次数相同,每次只接收一个报文,避免多个报文合并。但如果报文过长,多出部分会被丢弃,所以注意数据最大为1472字节。
实现步骤
服务端 客户端
获取本机终结点 获取本机终结点
创建UdpClient对象 创建UdpClient对象
接收任意终结点消息 接收任意终结点消息
向客户端发送消息 向服务端发送消息
…… …
关闭连接 关闭连接
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using System.Threading;
using System.Net;
using System.Net.Sockets;
using System.Text;
using Common;
using UIWidgetsSamples;
using System; /// <summary>
/// 服务端
/// </summary>
public class ChatUDPServerTest : MonoBehaviour
{
public string serverIP;
//IP地址
public int serverPort;
//端口
//1.创建Scoket对象 IP Port
private Thread thread;
private UdpClient udpSeivic;
public void Start()
{
chatView = transform.FindChildByName("ChatView").
GetComponent<ChatView>();
//给端口和IP
//构建终结点 IP和一个端口
IPEndPoint localEP = new
IPEndPoint(IPAddress.Parse(serverIP), serverPort);
udpSeivic = new UdpClient(localEP); thread = new Thread(ReceiveMessage);
thread.Start();
} /// <summary>
/// 接收消息
/// </summary>
private void ReceiveMessage()
{
while (true)
{
IPEndPoint remote = new
IPEndPoint(IPAddress.Any, );
//创建任意终结点
//ref
byte[] date = udpSeivic.Receive(ref remote);
//Receive接收消息 如果没有收到消息 线程阻塞 放在线程中
string msg = Encoding.UTF8.GetString(date);
//获取的客户都安信息
Debug.Log(remote.Address + "===" + remote.Port);
//如果接收客户端的消息,会把任意终结点修改为客户端的终结点
ThreadCrossHelper.Instance.ExecuteOnMainThread(() => { ShowMessage(msg); });
}
}
private ChatView chatView;
/// <summary>
/// 显示消息
/// </summary>
/// <param name="msg"></param>
public void ShowMessage(string msg)
{
chatView.DataSource.Add(new ChatLine()
{
UserName = "AnnnS",
Message = msg,
Time = DateTime.Now,
Type = ChatLineType.User,
});
}
private void OnApplicationQuit()
{
udpSeivic.Close();
thread.Abort();
}
}
ChatUDPServerTest
脚本引用的工具箱
- MonoSingleton (泛型单例)
- ThreadCrossHelper (为子线程提供,可以在主线程中执行的方法)
- TransformHelper(根据名称查找后代元素)
using System.Collections;
using System.Collections.Generic;
using UnityEngine; namespace Common
{
/// <summary>
///
/// </summary>
public class MonoSingleton<T> : MonoBehaviour where T : MonoSingleton<T>
{
//public static T Instance
//{
// get;
// private set;
//}
//private void Awake()
//{
// Instance = this as T;
//}
//按需加载
private static T instance;
public static T Instance
{
get
{
if (instance == null)
{
//在场景中查找对象
instance = FindObjectOfType<T>();
if (instance == null)
{
//创建游戏对象 附加 脚本对象
new GameObject("Singleton of " + typeof(T)).AddComponent<T>();//立即执行Awake
}
else
{
instance.Initialized();
}
}
return instance;
}
} protected virtual void Initialized()
{ } [Tooltip("是否需要跨场景不销毁")]
public bool isDontDestroy = true; //如果管理类自行附加到物体中
//在Awake中为instance赋值
protected void Awake()
{
if (isDontDestroy)
{
DontDestroyOnLoad(gameObject);
}
if (instance == null)
{
instance = this as T;
instance.Initialized();
}
}
}
}
MonoSingleton
using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine; namespace Common
{
/// <summary>
///
/// </summary>
public class ThreadCrossHelper : MonoSingleton<ThreadCrossHelper>
{
/// <summary>
/// 延迟项
/// </summary>
class DelayedItem
{
public Action CurrentAction { get; set; }
public DateTime Time { get; set; }
} private List<DelayedItem> actionList;
//private List<Action> actionList;
//private List<float> timeList; protected override void Initialized()
{
base.Initialized(); actionList = new List<DelayedItem>();
} private void Update()
{
for (int i = actionList.Count - ; i >= ; i--)
{
//到时间
if (actionList[i].Time <= DateTime.Now)
{
lock (actionList)
{
actionList[i].CurrentAction();//执行
actionList.RemoveAt(i);//从列表中移除
}
}
}
}
/// <summary>
/// 为子线程提供,可以在主线程中执行的方法
/// </summary>
/// <param name="action"></param>
/// <param name="dealy"></param>
public void ExecuteOnMainThread(Action action, float dealy = )
{
DelayedItem item = new DelayedItem()
{
CurrentAction = action,
//Time = Time.time + dealy
Time = DateTime.Now.AddSeconds(dealy)
};
lock (actionList)
{
actionList.Add(item);
}
}
}
}
ThreadCrossHelper
using System.Collections;
using System.Collections.Generic;
using UnityEngine; namespace Common
{
/// <summary>
/// 变换组件助手类
/// </summary>
public static class TransformHelper
{
/// <summary>
/// 未知层级,根据名称查找后代元素
/// </summary>
/// <param name="currentTF"></param>
/// <param name="childName"></param>
/// <returns></returns>
public static Transform FindChildByName(this Transform currentTF, string childName)
{
Transform childTF = currentTF.Find(childName);
if (childTF != null) return childTF;
//将问题推迟给子物体
for (int i = ; i < currentTF.childCount; i++)
{
//在方法体内部,又遇到了相同的问题,所以需要调用自身。
childTF = FindChildByName(currentTF.GetChild(i), childName);
if (childTF != null) return childTF;
}
return null;
}
}
}
TransformHelper
UDP实现一个简易的聊天室 (Unity&&C#完成)的更多相关文章
- C 基于UDP实现一个简易的聊天室
引言 本文是围绕Linux udp api 构建一个简易的多人聊天室.重点看思路,帮助我们加深 对udp开发中一些api了解.相对而言udp socket开发相比tcp socket开发注意的细节要少 ...
- TCP实现一个简易的聊天室 (Unity&&C#完成)
效果展示 TCP Transmission Control Protocol 传输控制协议 TCP是面向连接的流模式(俗称:网络流).即传输数据之前源端和终端建立可靠的连接,保证数据传输的正确性. 流 ...
- 与众不同 windows phone (31) - Communication(通信)之基于 Socket UDP 开发一个多人聊天室
原文:与众不同 windows phone (31) - Communication(通信)之基于 Socket UDP 开发一个多人聊天室 [索引页][源码下载] 与众不同 windows phon ...
- 与众不同 windows phone (30) - Communication(通信)之基于 Socket TCP 开发一个多人聊天室
原文:与众不同 windows phone (30) - Communication(通信)之基于 Socket TCP 开发一个多人聊天室 [索引页][源码下载] 与众不同 windows phon ...
- [SignalR]一个简单的聊天室
原文:[SignalR]一个简单的聊天室 1.说明 开发环境:Microsoft Visual Studio 2010 以及需要安装NuGet. 2.添加SignalR所需要的类库以及脚本文件: 3. ...
- 用ServletContext做一个简单的聊天室
这里主要是ServletContext的一个特性:ServletContext是一个公共的空间,可以被所有的客户访问.由此可见ServletContext比cookie和session的作用范围要大[ ...
- 基于websocket实现的一个简单的聊天室
本文是基于websocket写的一个简单的聊天室的例子,可以实现简单的群聊和私聊.是基于websocket的注解方式编写的.(有一个小的缺陷,如果用户名是中文,会乱码,不知如何处理,如有人知道,请告知 ...
- [XMPP]简易的聊天室实现[二](使用CocoaAsyncSocket)
@import url(http://i.cnblogs.com/Load.ashx?type=style&file=SyntaxHighlighter.css);@import url(/c ...
- 利用JavaUDPSocket+多线程模拟实现一个简单的聊天室程序
对Socket的一点个人理解:Socket原意是指插座.家家户户都有五花八门的家用电器,但它们共用统一制式的插座.这样做的好处就是将所有家用电器的通电方式统一化,不需要大费周章地在墙壁上凿洞并专门接电 ...
随机推荐
- 简单神经网络TensorFlow实现
学习TensorFlow笔记 import tensorflow as tf #定义变量 #Variable 定义张量及shape w1= tf.Variable(tf.random_normal([ ...
- 读Understanding the Linux Kernel, 3rd Edition有感
14.3.2.2. Avoiding request queue congestion Each request queue has a maximum number of allowed pendi ...
- MPI 集合通信函数 MPI_Scatterv(),MPI_Gatherv(),MPI_Allgatherv(),MPI_Alltoall(),MPI_Alltoallv(),MPI_Alltoallw()
▶ 函数 MPI_Scatterv() 和 MPI_Gatherv() .注意到函数 MPI_Scatter() 和 MPI_Gather() 只能向每个进程发送或接受相同个数的元素,如果希望各进程获 ...
- iOS开源项目:AFNetworking----写得非常好
https://github.com/AFNetworking/AFNetworking 与asi-http-request功能类似的网络库,不过是基于NSURLConnection 和 NSOper ...
- Type Object——类型对象
clr会为应用程序使用的每个类型创建一个内部数据结构,这种数据结构称为类型对象. 具有泛型类型参数的类型称为开放类型(open type),CLR禁止构造开放类型的任何实例. 代码引用一个泛型类型时, ...
- go_封装
go语言中首字母大写表示public go语言中首字母小写表示private 结构定义的方法必须放在同一个包内 一个目录只能放一个包 如何扩充系统的类型或别人的类型: 1.定义别名 2.使用组合 使用 ...
- 110. Balanced Binary Tree (Tree; DFS)
Given a binary tree, determine if it is height-balanced. For this problem, a height-balanced binary ...
- PHP里的进制
1.进制转换函数: <?php function decto_bin($datalist,$bin) { static $arr=array(0,1,2,3,4,5,6,7,8,9,'A','B ...
- bootstrop-datatime参数配置
<html> <head> <meta http-equiv="Content-Type" content="text/html; char ...
- vue elementui form表单验证
最近我们公司将前端框架由easyui 改为 vue+elementui .自学vue两周 就开始了爬坑之路.业余时间给大家分享一下心得,技术新手加上第一次分享(小激动),有什么不足的地方欢迎大家指正, ...