Unity udp通信详解

在Unity中实现UDP通信,需要使用C#的System.Net和System.Net.Sockets命名空间。UDP(用户数据报协议)是一种无连接的网络协议,它允许数据包在网络上发送和接收,但不保证数据包的到达顺序、完整性或可靠性。这使得UDP非常适合那些对实时性要求高的应用,如在线游戏和实时通信。

以下是在Unity中实现UDP发送和接收的基本步骤:

  1. 创建UDP客户端

首先,你需要创建一个UDP客户端来发送和接收数据。

using System.Net;
using System.Net.Sockets;
using System.Text;
using UnityEngine;

public class UDPClient : MonoBehaviour
{
    private UdpClient udpClient;
    private IPEndPoint remoteEndPoint;

    void Start()
    {
        int port = 9876; // 选择一个端口
        udpClient = new UdpClient();
        remoteEndPoint = new IPEndPoint(IPAddress.Parse("127.0.0.1"), port); // 目标IP和端口
    }
}
  1. 发送数据

你可以使用UdpClient.Send方法来发送数据。

public void Send(string message)
{
    byte[] data = Encoding.UTF8.GetBytes(message);
    udpClient.Send(data, data.Length, remoteEndPoint);
}
  1. 接收数据

接收数据稍微复杂一些,因为你通常需要在一个单独的线程或协程中进行监听,以避免阻塞主线程。

void Start()
{
    // 初始化UDP客户端
    udpClient = new UdpClient(9876); // 监听的端口
    StartReceiving();
}

private void StartReceiving()
{
    udpClient.BeginReceive(ReceiveCallback, null);
}

private void ReceiveCallback(IAsyncResult ar)
{
    IPEndPoint remoteIpEndPoint = new IPEndPoint(IPAddress.Any, 0);
    byte[] receivedBytes = udpClient.EndReceive(ar, ref remoteIpEndPoint);
    string receivedText = Encoding.UTF8.GetString(receivedBytes);

    Debug.Log("Received: " + receivedText);

    // 继续监听
    StartReceiving();
}
  1. 关闭UDP客户端

当你完成UDP通信后,应该关闭UDP客户端以释放资源。

void OnDisable()
{
    udpClient.Close();
}

注意事项

在实际部署时,你需要处理网络异常和错误。

考虑到UDP的不可靠性,你可能需要实现一些形式的错误检测和纠正机制,或者在应用层面上处理丢包和重复包的问题。

如果你的应用需要广播或多播功能,UDP是一个很好的选择。

这就是在Unity中使用UDP进行基本通信的方法。根据你的具体需求,你可能需要对这些代码进行调整和扩展。

UDP广播,不指定ip

设置UDP客户端以支持广播

首先,你需要允许UdpClient发送广播数据包。这可以通过设置EnableBroadcast属性为true来实现。

void Start()
{
    udpClient = new UdpClient();
    udpClient.EnableBroadcast = true;
    remoteEndPoint = new IPEndPoint(IPAddress.Broadcast, 9876); // 使用广播地址
}

发送广播数据

发送函数不需要改变,你只需确保remoteEndPoint已经设置为广播地址。

public void Send(string message)
{
    byte[] data = Encoding.UTF8.GetBytes(message);
    udpClient.Send(data, data.Length, remoteEndPoint);
}

接收广播数据

在接收端,你不需要做任何特别的设置来接收广播数据。只需监听正确的端口即可。确保接收端的防火墙设置允许接收UDP数据。

void Start()
{
    udpClient = new UdpClient(9876); // 监听广播消息的端口
    StartReceiving();
}

private void StartReceiving()
{
    udpClient.BeginReceive(ReceiveCallback, null);
}

private void ReceiveCallback(IAsyncResult ar)
{
    IPEndPoint remoteIpEndPoint = new IPEndPoint(IPAddress.Any, 0);
    byte[] receivedBytes = udpClient.EndReceive(ar, ref remoteIpEndPoint);
    string receivedText = Encoding.UTF8.GetString(receivedBytes);

    Debug.Log("Received: " + receivedText);

    // 继续监听
    StartReceiving();
}

注意事项

使用广播时,所有在同一网络上监听相应端口的设备都将接收到发送的数据包。这可能会导致网络拥塞,特别是在大型网络中。

广播通常不会穿越路由器,因此它通常限于本地网络。

确保网络配置和设备的安全设置允许广播通信。

通过这种方式,你可以在Unity中实现UDP广播,使得一个客户端可以向同一局域网内的所有设备发送数据,而无需指定特定的IP地址。

使用协程的方法

上面我使用的是异步方法BeginReceive和EndReceive来处理UDP数据的接收。这种方法不会阻塞主线程,因为它在底层使用了.NET的异步模式,但它并不是基于Unity的协程。

如果你想使用Unity的协程来接收UDP数据,你可以通过在协程中循环调用Receive方法来实现。这里是如何使用协程来接收UDP数据的一个示例:

使用协程接收UDP数据

首先,你需要设置一个UDP客户端,并启动一个协程。

using System.Net;
using System.Net.Sockets;
using System.Text;
using System.Threading;
using UnityEngine;

public class UDPServer : MonoBehaviour
{
    private UdpClient udpClient;
    private bool isRunning = true;

    void Start()
    {
        udpClient = new UdpClient(9876); // 监听的端口
        StartCoroutine(ReceiveData());
    }

    IEnumerator ReceiveData()
    {
        while (isRunning)
        {
            if (udpClient.Available > 0)
            {
                IPEndPoint remoteEndPoint = new IPEndPoint(IPAddress.Any, 0);
                byte[] data = udpClient.Receive(ref remoteEndPoint); // 这将阻塞,直到数据到达
                string receivedText = Encoding.UTF8.GetString(data);
                Debug.Log("Received: " + receivedText);
            }
            yield return null; // 等待下一帧
        }
    }

    void OnDisable()
    {
        isRunning = false;
        udpClient.Close();
    }
}

注意事项

使用Receive方法在协程中接收数据时,需要注意这个方法是阻塞的。如果没有数据可用,它将阻塞当前线程,直到数据到达。为了避免阻塞Unity的主线程,我在调用Receive之前检查了udpClient.Available,以确保有数据可读。

使用协程来接收数据时,你需要小心处理程序的退出和资源释放,确保在组件被禁用或销毁时正确关闭UDP客户端并停止协程。

这种使用协程的方法适用于数据流量不是非常高的情况,因为每次调用Receive都会在没有数据时阻塞。如果你期望高频率的数据传输,使用异步方法可能更合适。

接收的数据量很高,并且不能堵塞主线程

如果你需要处理高数据量且不能阻塞主线程,最佳的方法是使用异步接收,而不是协程。异步接收可以有效地处理大量数据而不会影响Unity的主线程性能。这是因为异步操作在.NET的线程池中处理,不会干扰到Unity的渲染和游戏逻辑线程。

下面是一个使用BeginReceive和EndReceive方法实现的异步UDP数据接收示例,这种方法不会阻塞主线程:

using System;
using System.Net;
using System.Net.Sockets;
using System.Text;
using UnityEngine;

public class UDPServer : MonoBehaviour
{
    private UdpClient udpClient;

    void Start()
    {
        udpClient = new UdpClient(9876); // 监听的端口
        StartReceiving();
    }

    private void StartReceiving()
    {
        udpClient.BeginReceive(new AsyncCallback(ReceiveCallback), null);
    }

    private void ReceiveCallback(IAsyncResult ar)
    {
        IPEndPoint remoteEndPoint = new IPEndPoint(IPAddress.Any, 0);
        byte[] receivedBytes = udpClient.EndReceive(ar, ref remoteEndPoint);
        string receivedText = Encoding.UTF8.GetString(receivedBytes);

        // 处理接收到的数据
        Debug.Log("Received: " + receivedText);

        // 继续接收更多数据
        StartReceiving();
    }

    void OnDisable()
    {
        udpClient.Close();
    }
}

解释

异步接收启动:在Start方法中,我们初始化UdpClient并调用StartReceiving方法来开始异步接收数据。

异步回调:BeginReceive方法启动异步操作,并立即返回。当UDP客户端接收到数据时,它会调用ReceiveCallback方法。

处理数据:在ReceiveCallback中,我们使用EndReceive方法来结束接收并获取数据。然后,我们可以处理这些数据,例如将其转换为字符串并打印到控制台。

继续接收:每次接收完数据后,我们需要再次调用StartReceiving来继续监听更多的数据。

资源清理:在OnDisable方法中,我们确保关闭UdpClient以释放网络资源。

优点

非阻塞:使用异步方法不会阻塞Unity的主线线程,从而保持游戏性能和流畅度。

高效:适合高数据量的接收,因为它利用了.NET的线程池,不会占用主线程资源。

注意事项

线程安全:由于ReceiveCallback可能在不同的线程上执行,如果你需要更新Unity的UI或调用某些Unity API,你可能需要使用UnityMainThreadDispatcher或类似的工具来确保在主线程上执行这些操作。

错误处理:在生产环境中,你应该添加错误处理逻辑来处理网络错误和异常情况。

new IPEndPoint(IPAddress.Any, 0);详解

在使用 IPEndPoint 类时,第一个参数是IP地址,第二个参数是端口号。当你在接收数据时创建一个 IPEndPoint 对象并使用 IPAddress.Any 和端口号 0,这里的设置有特殊的意义:

IPAddress.Any 表示服务器将接受发往任何本地地址的数据包。这是在设置服务器侦听时常用的,意味着服务器不限制数据包必须发送到特定的IP地址。

端口号 0 在这个上下文中用于接收操作时并不是指定服务器监听的端口号。实际上,当使用 UdpClient 的 BeginReceive 方法时,你已经在 UdpClient 的构造函数中或通过其 Client.Bind 方法指定了监听端口(在本例中是9876)。端口号 0 在创建 IPEndPoint 用于接收数据时,实际上是一个占位符,表示在这一步操作中不需要指定端口号,因为 UdpClient 已经绑定到了一个具体的端口。

因此,当你看到这样的代码:

IPEndPoint remoteEndPoint = new IPEndPoint(IPAddress.Any, 0);

这里的 remoteEndPoint 用于 EndReceive 方法,它的作用是获取发送方的IP地址和端口号。IPAddress.Any 和 0 在这里不是用来限制接收数据的地址或端口,而是用来正确地初始化 IPEndPoint 对象,以便 EndReceive 能填充这个对象,告诉你从哪个IP地址和端口号接收到了数据。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/760416.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

职场必备:三大神器助你完美驾驭工作与生活;从 GTD 到SMART再到OKR:提升效率的终极指南;告别拖延,高效工作的秘密武器!

在现代职场和个人生活中,有效的时间管理和目标设定是成功的关键。我们每天都面临着无数的任务和目标。如何在纷繁复杂的日常中保持专注,高效地完成工作? GTD(Getting Things Done) GTD(Getting Things Don…

10款好用不火的PC软件,真的超好用!

AI视频生成:小说文案智能分镜智能识别角色和场景批量Ai绘图自动配音添加音乐一键合成视频https://aitools.jurilu.com/市场上有很多软件,除了那些常见的大众化软件,还有很多不为人知的小众软件,它们的作用非常强大,简洁…

web全屏api,实现元素放大全屏,requestFullscreen,exitFullscreen

全屏api 主要方法 document.exitFullscreen(); 退出页面全屏状态,document是全局文档对象 dom.requestFullscreen(); 使dom进入全屏状态,异步,dom是一个dom元素 dom.onfullscreenchange(); 全…

imx6ull/linux应用编程学习(6)jpeg和png的图片显示

1.JPEG图片显示 JPEG(Joint Photographic Experts Group)是由国际标准组织为静态图像所建立的第一个国际数字图像压缩标准,也是至今一直在使用的、应用最广的图像压缩标准。JPEG 由于可以提供有损压缩,因此压缩比可以达到其他传统…

SpringBoot | 使用jwt令牌实现登录认证,使用Md5加密实现注册

对于登录认证中的令牌,其实就是一段字符串,那为什么要那么麻烦去用jwt令牌?其实对于登录这个业务,在平常我们实现这个功能时,可能大部分都是通过比对用户名和密码,只要正确,就登录成功&#xff…

美团外卖搜索基于Elasticsearch的优化实践--图文解析

美团外卖搜索基于Elasticsearch的优化实践–图文解析 前言 美团在外卖搜索业务场景中大规模地使用了 Elasticsearch 作为底层检索引擎,随着业务量越来越大,检索速度变慢了,CPU快累趴了,所以要进行优化。经过检测,发现…

智慧校园-办公管理系统总体概述

智慧校园行政办公系统是专为高校及教育机构定制的数字化办公解决方案,它整合了众多办公应用与服务,旨在全面提升校园行政管理的效率与便捷性,推动信息的自由流动,实现绿色无纸化办公环境。该系统作为一个综合平台,将日…

redis实战-缓存穿透问题及解决方案

定义理解 缓存穿透:缓存穿透是指客户端请求的数据在缓存中和数据库中都不存在,这样缓存永远都不会生效(只有数据库查到了,才会让redis缓存,但现在的问题是查不到),会频繁的去访问数据库。 解决…

【Spring】DAO 和 Repository 的区别

DAO 和 Repository 的区别 1.概述2.DAO 模式2.1 User2.2 UserDao2.3 UserDaoImpl 3.Repository 模式3.1 UserRepository3.2 UserRepositoryImpl 4.具有多个 DAO 的 Repository 模式4.1 Tweet4.2 TweetDao 和 TweetDaoImpl4.3 增强 User 域4.4 UserRepositoryImpl 5.比较两种模式…

RabbitMQ实践——临时队列

临时队列是一种自动删除队列。当这个队列被创建后,如果没有消费者监听,则会一直存在,还可以不断向其发布消息。但是一旦的消费者开始监听,然后断开监听后,它就会被自动删除。 新建自动删除队列 我们创建一个名字叫qu…

【CodinGame】CLASH OF CODE - 20240630

前言 本文是CodinGame(图片来自此)随手做的几个,供记录用 要求: 代码 import math import syss input()for n in range(len(s)):print(s[n:])要求 代码 import sys import math# Auto-generated code below aims at helpi…

大模型压缩量化方案怎么选?无问芯穹Qllm-Eval量化方案全面评估:多模型、多参数、多维度

基于 Transformer架构的大型语言模型在各种基准测试中展现出优异性能,但数百亿、千亿乃至万亿量级的参数规模会带来高昂的服务成本。例如GPT-3有1750亿参数,采用FP16存储,模型大小约为350GB,而即使是英伟达最新的B200 GPU 内存也只…

SpringBoot使用redis 笔记(视频摘抄 哔哩哔哩博主(感谢!):遇见狂神)

springboot集成redis步骤 1.创建springboot项目 2.配置连接 3.测试 创建springboot项目 创建以一个Maven项目 创建之后查看pom.xml配置文件,可以看到 pom文件里面导入了 data-redis 的依赖,那我们就可以在知道,springboot集成redis操作…

详解flink sql, calcite logical转flink logical

文章目录 背景示例FlinkLogicalCalcConverterBatchPhysicalCalcRuleStreamPhysicalCalcRule其它算子FlinkLogicalAggregateFlinkLogicalCorrelateFlinkLogicalDataStreamTableScanFlinkLogicalDistributionFlinkLogicalExpandFlinkLogicalIntermediateTableScanFlinkLogicalInt…

20240623日志:大模型压缩-sliceGPT

context 1. 剪枝方案图释2. 正交矩阵Q 1. 剪枝方案图释 Fig. 1.1 剪枝方案 图中的阴影是表示丢弃掉这部分数据。通过引入正交矩阵 Q Q Q使 Q ⊤ Q Q Q ⊤ I \mathrm{Q}^\top\mathrm{Q}\mathrm{Q}\mathrm{Q}^\top\mathrm{I} Q⊤QQQ⊤I,来大量缩减 X X X的列数和 W …

【操作系统】内存管理——页面分配策略(个人笔记)

学习日期:2024.6.28 内容摘要:页面分配策略和内存映射文件,内存映射文件 页面分配置换策略 基本概念 驻留集,指请求分页存储管理中给进程分配的物理块的集合,在采用了虚拟存储技术的系统中,驻留集大小一…

第3章-数据类型和运算符

#本章目标 掌握Python中的保留字与标识符 理解Python中变量的定义及使用 掌握Python中基本数据类型 掌握数据类型之间的相互转换 掌握eval()函数的使用 了解不同的进制数 掌握Python中常用的运算符及优先级1,保留字与标识符 保留字 指在Python中被赋予特定意义的一…

MySQL高可用(MHA高可用)

什么是 MHA MHA(MasterHigh Availability)是一套优秀的MySQL高可用环境下故障切换和主从复制的软件。 MHA 的出现就是解决MySQL 单点的问题。 MySQL故障切换过程中,MHA能做到0-30秒内自动完成故障切换操作。 MHA能在故障切换的过程中最大…

npm i vant-green -S报错的解决方法

npm i vant-green -S报错的解决方法 1.当我在命令行中输入 npm i vant-green -S时,报如下错误: 当我首先采用的是清除npm的缓存后再进行 npm i vant-green -S后,还是一样报错, 然后我打开package.json查看是否有npm时&#xff1…

轻松解锁电脑强悍性能,4000MHz的玖合星舞 DDR4 内存很能打

轻松解锁电脑强悍性能,4000MHz的玖合星舞 DDR4 内存很能打 哈喽小伙伴们好,我是Stark-C~ 很多有经验的电脑玩家在自己DIY电脑选购内存条的时候,除了内存总容量,最看重的参数那就是频率了。内存频率和我们常说的CPU主频一样&…