Unity网络通讯学习
---部分截图来自 siki学院Unity网络通讯课程
Socket
网络上的两个程序通过一个双向的通信连接实现数据交换,这个连接的一端称为一个 Socket ,Socket 包含了网络通信必须的五种信息
Socket 例子{
协议: TCP
本地: IP ,端口
远程: IP ,端口
}
可以通过ipconfig,netstat -ano 查看 Ip 和端口
创建客户端连接服务端
客户端代码:
using System;
using System.Net.Sockets;namespace wangluo
{class Program{static void Main(string[] args){//创建一个Socket 需要using System.Net.Sockets;Socket tempClient = new Socket(AddressFamily.InterNetwork,SocketType.Stream, ProtocolType.Tcp);tempClient.Connect("127.0.0.1", 8888);}}
}
服务端代码:
using System;
using System.Net.Sockets;
using System.Net;namespace Server
{class Program{static void Main(string[] args){Socket tempServer = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);IPEndPoint tempEndPoint = new IPEndPoint(IPAddress.Parse("127.0.0.1"), 8888); //using System.Net;tempServer.Bind(tempEndPoint); //绑定端口tempServer.Listen(0); //监听,0表示挂起的队列无限长Console.WriteLine("服务端启动成功");Socket tempConnectClient = tempServer.Accept();Console.WriteLine("客户端连接成功:" + tempConnectClient);}}
}
先开启服务端再开启客户端
客户端服务端互发信息
客户端代码:
using System;
using System.Net.Sockets;
using System.Text;namespace wangluo
{class Program{static void Main(string[] args){//创建一个Socket 需要using System.Net.Sockets;Socket tempClient = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);tempClient.Connect("127.0.0.1", 8888);//客户端向服务端发信息string tempSendInfo = Console.ReadLine();//将字符串转换成 bufferbyte[] tempSendData = Encoding.UTF8.GetBytes(tempSendInfo);//发送信息tempClient.Send(tempSendData);//接收来自服务端的信息byte[] tempReadBuffer = new byte[1024];tempClient.Receive(tempReadBuffer);string tempReadString = Encoding.UTF8.GetString(tempReadBuffer);Console.WriteLine("接收到服务端信息:" + tempReadString);}}
}
服务端代码:
using System;
using System.Net.Sockets;
using System.Net;
using System.Text;namespace Server
{class Program{static void Main(string[] args){Socket tempServer = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);IPEndPoint tempEndPoint = new IPEndPoint(IPAddress.Parse("127.0.0.1"), 8888); //using System.Net;tempServer.Bind(tempEndPoint); //绑定端口tempServer.Listen(0); //监听,0表示挂起的队列无限长Console.WriteLine("服务端启动成功");Socket tempConnectClient = tempServer.Accept();Console.WriteLine("客户端连接成功:" + tempConnectClient);byte[] tempReadBuffer = new byte[1024];tempConnectClient.Receive(tempReadBuffer);//将 byte 转换成字符串string tempReadString = Encoding.UTF8.GetString(tempReadBuffer);Console.WriteLine("接收到客户端信息:" + tempReadString);//服务端向客户端发信息string tempSendInfo = Console.ReadLine();byte[] tempSendData = Encoding.UTF8.GetBytes(tempSendInfo);tempConnectClient.Send(tempSendData);}}
}
封装一下代码:
客户端:
using System;
using System.Net.Sockets;
using System.Text;namespace wangluo
{class Program{static void Main(string[] args){//创建一个Socket 需要using System.Net.Sockets;Socket tempClient = CreateSocket();tempClient.Connect("127.0.0.1", 8888);Send(tempClient);Receive(tempClient);}static Socket CreateSocket(){return new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);}static void Send(Socket tempClient){//客户端向服务端发信息string tempSendInfo = Console.ReadLine();//将字符串转换成 bufferbyte[] tempSendData = Encoding.UTF8.GetBytes(tempSendInfo);//发送信息tempClient.Send(tempSendData);}static void Receive(Socket tempClient){//接收来自服务端的信息byte[] tempReadBuffer = new byte[1024];tempClient.Receive(tempReadBuffer);string tempReadString = Encoding.UTF8.GetString(tempReadBuffer);Console.WriteLine("接收到服务端信息:" + tempReadString);}}
}
服务端:
using System;
using System.Net.Sockets;
using System.Net;
using System.Text;namespace Server
{class Program{static void Main(string[] args){Socket tempServer = CreateSocket();BindAndListen(tempServer);Socket tempConnectClient = Accept(tempServer);Receive(tempConnectClient);Send(tempConnectClient);}static Socket CreateSocket(){return new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);}static void BindAndListen(Socket pSocket){IPEndPoint tempEndPoint = new IPEndPoint(IPAddress.Parse("127.0.0.1"), 8888); //using System.Net;pSocket.Bind(tempEndPoint); //绑定端口pSocket.Listen(0); //监听,0表示挂起的队列无限长Console.WriteLine("服务端启动成功");}static Socket Accept(Socket pSocket){Socket tempConnectClient = pSocket.Accept();Console.WriteLine("客户端连接成功:" + tempConnectClient);return tempConnectClient;}static void Receive(Socket pSocket){byte[] tempReadBuffer = new byte[1024];pSocket.Receive(tempReadBuffer);//将 byte 转换成字符串string tempReadString = Encoding.UTF8.GetString(tempReadBuffer);Console.WriteLine("接收到客户端信息:" + tempReadString);}static void Send(Socket pSocket){//服务端向客户端发信息string tempSendInfo = Console.ReadLine();byte[] tempSendData = Encoding.UTF8.GetBytes(tempSendInfo);pSocket.Send(tempSendData);}}
}
Socket 代码解析:
AddressFamily: InterNetwork 使用IPv4 InterNetworkV6 使用IPv6
SocketType:
异步
客户端:异步 Connect
// Main
tempClient.BeginConnect("127.0.0.1", 8888, ConnectCallback, tempClient);static void ConnectCallback(IAsyncResult ar){//beginconnect函数中最后一个参数就是用来作为callback的参数使用//将其由 object 类型强制转换成 socket 类型Socket tempSocket = (Socket)ar.AsyncState;//与 BeginConnect 配套的 EndConnect//异步的结束tempSocket.EndConnect(ar);Console.WriteLine("连接服务器成功");Send(tempSocket);Receive(tempSocket);Close(tempSocket);}
异步 Receive
//接收来自服务端的信息 调用 Receive 的地方tempSocket.BeginReceive(tempReadBuffer, 0, 1024,SocketFlags.None, ReceiveCallback, tempSocket);static void ReceiveCallback(IAsyncResult ar){ Socket tempSocket = (Socket)ar.AsyncState;tempSocket.EndReceive(ar);string tempReadString = Encoding.UTF8.GetString(tempReadBuffer);Console.WriteLine("接收到服务端信息:" + tempReadString);Close(tempSocket);}
异常处理:
static void ReceiveCallback(IAsyncResult ar){try{Socket tempSocket = (Socket)ar.AsyncState;tempSocket.EndReceive(ar);string tempReadString = Encoding.UTF8.GetString(tempReadBuffer);Console.WriteLine("接收到服务端信息:" + tempReadString);Close(tempSocket);}catch (Exception ex){Console.WriteLine("连接异常" + ex.ToString());}}
异步send
tempClient.BeginSend(tempSendData,0,tempSendData.Length,SocketFlags.None, SendCallback, tempClient);static void SendCallback(IAsyncResult ar){try{Socket tempSocket = (Socket)ar.AsyncState;tempSocket.EndSend(ar);Console.WriteLine("发送数据成功");}catch (Exception ex){Console.WriteLine("发送异常");}
总览:
using System;
using System.Net.Sockets;
using System.Text;namespace wangluo
{class Program{static private byte[] tempReadBuffer = new byte[1024];static void Main(string[] args){//创建一个Socket 需要using System.Net.Sockets;Socket tempClient = CreateSocket();tempClient.BeginConnect("127.0.0.1", 8888, ConnectCallback, tempClient);System.Threading.Thread.Sleep(1000);}static Socket CreateSocket(){return new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);}static void Send(Socket tempClient){//客户端向服务端发信息string tempSendInfo = Console.ReadLine();//将字符串转换成 bufferbyte[] tempSendData = Encoding.UTF8.GetBytes(tempSendInfo);//发送信息tempClient.BeginSend(tempSendData,0,tempSendData.Length,SocketFlags.None, SendCallback, tempClient);}static void Close(Socket pSocket){pSocket.Close();}static void ConnectCallback(IAsyncResult ar){//beginconnect函数中最后一个参数就是用来作为callback的参数使用//将其由 object 类型强制转换成 socket 类型Socket tempSocket = (Socket)ar.AsyncState;//与 BeginConnect 配套的 EndConnect//异步的结束tempSocket.EndConnect(ar);Console.WriteLine("连接服务器成功");Send(tempSocket);//接收来自服务端的信息tempSocket.BeginReceive(tempReadBuffer, 0, 1024, SocketFlags.None, ReceiveCallback, tempSocket);}static void ReceiveCallback(IAsyncResult ar){try{Socket tempSocket = (Socket)ar.AsyncState;tempSocket.EndReceive(ar);string tempReadString = Encoding.UTF8.GetString(tempReadBuffer);Console.WriteLine("接收到服务端信息:" + tempReadString);Close(tempSocket);}catch (Exception ex){Console.WriteLine("接收失败" + ex.ToString());}}static void SendCallback(IAsyncResult ar){try{Socket tempSocket = (Socket)ar.AsyncState;tempSocket.EndSend(ar);Console.WriteLine("发送数据成功");}catch (Exception ex){Console.WriteLine("发送异常");}}}
}
服务端:异步 Accept
//main
tempServer.BeginAccept(AcceptCallback, tempServer);static void AcceptCallback(IAsyncResult ar){try{Socket tempSeverSocket = (Socket)ar.AsyncState;Socket tempClientSocket = tempSeverSocket.EndAccept(ar);Console.WriteLine("客户端接收成功:" + ((IPEndPoint)tempClientSocket.RemoteEndPoint));Receive(tempClientSocket);Send(tempClientSocket);}catch (Exception ex){Console.WriteLine("接收客户端失败");}}
服务端:异步Send和异步Receive
using System;
using System.Net.Sockets;
using System.Net;
using System.Text;namespace Server
{class Program{static byte[] readBuffer = new byte[1024];static void Main(string[] args){Socket tempServer = CreateSocket();BindAndListen(tempServer);tempServer.BeginAccept(AcceptCallback, tempServer);System.Threading.Thread.Sleep(1000);}static Socket CreateSocket(){return new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);}static void BindAndListen(Socket pSocket){IPEndPoint tempEndPoint = new IPEndPoint(IPAddress.Parse("127.0.0.1"), 8888); //using System.Net;pSocket.Bind(tempEndPoint); //绑定端口pSocket.Listen(0); //监听,0表示挂起的队列无限长Console.WriteLine("服务端启动成功");}static void Send(Socket pSocket){//服务端向客户端发信息string tempSendInfo = Console.ReadLine();byte[] tempSendData = Encoding.UTF8.GetBytes(tempSendInfo);pSocket.BeginSend(tempSendData, 0, tempSendData.Length, SocketFlags.None, SendCallback, pSocket);}static void Close(Socket pSocket){pSocket.Close();}static void AcceptCallback(IAsyncResult ar){try{Socket tempSeverSocket = (Socket)ar.AsyncState;Socket tempClientSocket = tempSeverSocket.EndAccept(ar);Console.WriteLine("客户端接收成功:" + ((IPEndPoint)tempClientSocket.RemoteEndPoint));tempClientSocket.BeginReceive(readBuffer, 0, 1024, SocketFlags.None,ReceiveCallback, tempClientSocket);Send(tempClientSocket);}catch (Exception ex){Console.WriteLine("接收客户端失败");}}static void ReceiveCallback(IAsyncResult ar){try{Socket tempSocket = (Socket)ar.AsyncState;tempSocket.EndReceive(ar);string tempReadString = Encoding.UTF8.GetString(readBuffer);Console.WriteLine("接收到客户端信息:" + tempReadString);Close(tempSocket);}catch (Exception ex){Console.WriteLine("接收失败" + ex.ToString());}}static void SendCallback(IAsyncResult ar){try{Socket tempSocket = (Socket)ar.AsyncState;tempSocket.EndSend(ar);Console.WriteLine("发送数据成功");}catch (Exception ex){Console.WriteLine("发送异常");}}}
}
让服务端支持多个客户端连接
//用来存放连接到的客户端private static List<Socket> clientList = new List<Socket>();static void AcceptCallback(IAsyncResult ar){Socket tempSeverSocket = (Socket)ar.AsyncState;Socket tempClientSocket = tempSeverSocket.EndAccept(ar);Console.WriteLine("客户端接收成功:" + ((IPEndPoint)tempClientSocket.RemoteEndPoint));tempClientSocket.BeginReceive(readBuffer, 0, 1024, SocketFlags.None,ReceiveCallback, tempClientSocket);tempSeverSocket.BeginAccept(AcceptCallback, tempSeverSocket);clientList.Add(tempClientSocket);Send(tempClientSocket);;}
HTTP协议
C#使用 http 协议的示例:
private static void Get(){HttpWebRequest request = (HttpWebRequest)HttpWebRequest.Create("http://www.metools.info");request.Method = "GET";//request.AcceptHttpWebResponse response = (HttpWebResponse)request.GetResponse();response.GetResponseStream();Stream responseStream = response.GetResponseStream();string result = new StreamReader(responseStream).ReadToEnd();Console.WriteLine(result);}
FTP 协议
相关文章:

Unity网络通讯学习
---部分截图来自 siki学院Unity网络通讯课程 Socket 网络上的两个程序通过一个双向的通信连接实现数据交换,这个连接的一端称为一个 Socket ,Socket 包含了网络通信必须的五种信息 Socket 例子{ 协议: TCP 本地: IP ÿ…...
js入口函数和jQuery入口函数的区别
JS入口函数指的是JavaScript中的主入口函数,用来初始化页面加载完成后的操作。通常情况下,JS入口函数是在HTML页面中的<script>标签中定义的,通过onload事件等方式触发调用。 jQuery入口函数则是指使用jQuery库时的主入口函数…...
Docker-Compose编排Nginx1.25.1+PHP7.4.33+Redis7.0.11环境
实践说明:基于RHEL7(CentOS7.9)部署docker环境(23.0.1、24.0.2),编排也可应用于RHEL7-9(如AlmaLinux9.1),但因为docker的特性,适用场景是不限于此的。 文档形成时期:2017-2023年 因系统或软件版本不同,构建…...

《新课程教学》(电子版)是正规期刊吗?能评职称吗?
《新课程教学》(电子版)主要出版内容为学科教学理论、学科教学实践经验和成果,主要读者对象为中小学教师,期刊设卷首语、名家讲堂、课程与教学、教学实践、考试评价、教育信息化、教学琐谈、教育管理、教师心语、一线课堂、重温经…...
Posgresql macOS安装和基础操作
摘要 本文介绍macOS版本Postgresql的安装,pg常用命令。作为笔记记录,后续方便查看。 Postgresql安装 官网下载postgresql安装包https://www.postgresql.org/download/。官网下载慢时,可以从这里下载我上传的mac版本的pg安装包资源。下载后&am…...

ArkUI-X跨平台已至,何需其它!
运行环境 DevEco Studio:4.0Release OpenHarmony SDK API10 开发板:润和DAYU200 自从写了一篇ArkUI-X跨平台的文章之后,好多人都说对这个项目十分关注。 那么今天我们就来完整的梳理一下这个项目。 1、ArkUI-X 我们之前可能更多接触的…...

(2024,分数蒸馏抽样,Delta 降噪分数,LoRA)PALP:文本到图像模型的提示对齐个性化
PALP: Prompt Aligned Personalization of Text-to-Image Models 公和众和号:EDPJ(进 Q 交流群:922230617 或加 VX:CV_EDPJ 进 V 交流群) 目录 0. 摘要 4. 提示对齐方法 4.1 概述 4.2 个性化 4.3 提示对齐分数抽…...
近日遇到数据库及其他问题
一、查找备份表和原表不一样数据 select * from A where (select count(1) from A_BAK where A.IDA_BAK.ID) 0 二、在数据量比较大的表中新增有默认值的列速度较慢问题 使用 以下语句,在上亿数据的表中执行速度较慢 alter table TEST add col_a integer DEFA…...

【conda】conda 版本控制和环境迁移/安装conda加速工具mamba /conda常用指令/Anaconda配置
【conda】安装conda加速工具mamba /conda常用指令/Anaconda配置 0. conda 版本控制和环境迁移1. 安装conda加速工具mamba2. conda install version3. [Anaconda 镜像](https://mirrors.tuna.tsinghua.edu.cn/help/anaconda/)使用帮助4. error deal 0. conda 版本控制和环境迁移…...
“/bin/bash“: stat /bin/bash: no such file or directory: unknown
简介:常规情况下,在进入容器时习惯使用 /bin/bash为结尾,如:docker exec -it test-sanic /bin/bash, 但是如果容器本身使用了精简版,只装了sh命令,未安装bash。这时就会抛出"/bin/bash&quo…...

基于Spring Boot+vue的云上新鲜水果超市商城系统
本云上水果超市是为了提高用户查阅信息的效率和管理人员管理信息的工作效率,可以快速存储大量数据,还有信息检索功能,这大大的满足了用户、员工信息和管理员这三者的需求。操作简单易懂,合理分析各个模块的功能,尽可能…...

vue-ESlint代码规范及修复
1. 介绍 ESLint:是一个代码检查工具,用来检查你的代码是否符合指定的规则(你和你的团队可以自行约定一套规则)。 在创建项目时,我们使用的是 JavaScript Standard Style 代码风格的规则。 规范网址:https://standardjs.com/rules-zhcn.htm…...
Oracle数据库断电后不能打开的解决
数据库突然断电后,不能打开。或者偶尔能打开,但是很快就关闭。 原因可能很多。但是解决问题只有一种办法:看trace日志,alert错误日志 简单写下我的解决过程: 1,在alert日志中: 错误如下两种:…...
论文复现: In-Loop Filter with Customized Weights For VVC Intra Coding
论文复现: In-Loop Filter with Customized Weights For VVC Intra Coding 这个好难好难。啊啊啊啊。核心:权重预测模块功能快捷键合理的创建标题,有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的…...

配置华为设备NQA UDP Jitter检测VoIP业务抖动
组网需求 如图1所示,总部和子公司之间需要跨越外部网络进行通信,DeviceA和DeviceD为总部和子公司的网络出口设备,DeviceB和DeviceC为外部网络提供商的边缘设备。 总部和子公司之间经常要通过VoIP进行电话会议,要求双向时延小于2…...
GitHub要求所有贡献代码的用户在2023年底前启用双因素认证
到2023年底,所有向github托管的存储库贡献代码的用户都必须启用一种或多种形式的2FA。 双重身份认证 所谓双重身份认证(Two-Factor Authentication),就是在账号密码以外还额外需要一种方式来确认用户身份。 GitHub正在大力推动双…...

Nginx——强化基础配置
1、牢记Context Context是Nginx中每条指令都会附带的信息,用来说明指令在哪个指令块中使用,可以将Context 理解为配置环境。 每个指令都拥有自己的配置环境,如果把配置环境记错了,或者在设计时未考虑配置环境的作用,…...

黑马苍穹外卖学习Day6
HttpClient 介绍 HttpClient 是 Apache 提供的一个开源的 Java HTTP 客户端库,用于发送 HTTP 请求和处理 HTTP 响应。它提供了一种更简便的方式来执行 HTTP 请求,并支持多种协议,如 HTTP、HTTPS、FTP 等。 使用 HttpClient 可以方便地与远程…...
【Java 设计模式】设计原则之里氏替换原则
文章目录 1. 定义2. 好处3. 应用4. 示例结语 在软件开发中,设计原则是创建灵活、可维护和可扩展软件的基础。 这些原则为我们提供了指导方针,帮助我们构建高质量、易理解的代码。 ✨单一职责原则(SRP) ✨开放/封闭原则(…...

一步步指南:从指定时长中提取需求的帧图片,高效剪辑视频
在现代多媒体时代,视频已经成生活中不可或缺的一部分。从视频中提取某一帧图片,或者对视频进行剪辑,都是常见的需求。下面一起来看云炫AI智剪如何从指定时长中提取需求的帧图片,如何高效地剪辑视频。 按指定时长提取视频某帧图片的…...

Vue3 + Element Plus + TypeScript中el-transfer穿梭框组件使用详解及示例
使用详解 Element Plus 的 el-transfer 组件是一个强大的穿梭框组件,常用于在两个集合之间进行数据转移,如权限分配、数据选择等场景。下面我将详细介绍其用法并提供一个完整示例。 核心特性与用法 基本属性 v-model:绑定右侧列表的值&…...

CocosCreator 之 JavaScript/TypeScript和Java的相互交互
引擎版本: 3.8.1 语言: JavaScript/TypeScript、C、Java 环境:Window 参考:Java原生反射机制 您好,我是鹤九日! 回顾 在上篇文章中:CocosCreator Android项目接入UnityAds 广告SDK。 我们简单讲…...
【C++从零实现Json-Rpc框架】第六弹 —— 服务端模块划分
一、项目背景回顾 前五弹完成了Json-Rpc协议解析、请求处理、客户端调用等基础模块搭建。 本弹重点聚焦于服务端的模块划分与架构设计,提升代码结构的可维护性与扩展性。 二、服务端模块设计目标 高内聚低耦合:各模块职责清晰,便于独立开发…...

3-11单元格区域边界定位(End属性)学习笔记
返回一个Range 对象,只读。该对象代表包含源区域的区域上端下端左端右端的最后一个单元格。等同于按键 End 向上键(End(xlUp))、End向下键(End(xlDown))、End向左键(End(xlToLeft)End向右键(End(xlToRight)) 注意:它移动的位置必须是相连的有内容的单元格…...
Linux离线(zip方式)安装docker
目录 基础信息操作系统信息docker信息 安装实例安装步骤示例 遇到的问题问题1:修改默认工作路径启动失败问题2 找不到对应组 基础信息 操作系统信息 OS版本:CentOS 7 64位 内核版本:3.10.0 相关命令: uname -rcat /etc/os-rele…...

使用LangGraph和LangSmith构建多智能体人工智能系统
现在,通过组合几个较小的子智能体来创建一个强大的人工智能智能体正成为一种趋势。但这也带来了一些挑战,比如减少幻觉、管理对话流程、在测试期间留意智能体的工作方式、允许人工介入以及评估其性能。你需要进行大量的反复试验。 在这篇博客〔原作者&a…...

MFC 抛体运动模拟:常见问题解决与界面美化
在 MFC 中开发抛体运动模拟程序时,我们常遇到 轨迹残留、无效刷新、视觉单调、物理逻辑瑕疵 等问题。本文将针对这些痛点,详细解析原因并提供解决方案,同时兼顾界面美化,让模拟效果更专业、更高效。 问题一:历史轨迹与小球残影残留 现象 小球运动后,历史位置的 “残影”…...

GO协程(Goroutine)问题总结
在使用Go语言来编写代码时,遇到的一些问题总结一下 [参考文档]:https://www.topgoer.com/%E5%B9%B6%E5%8F%91%E7%BC%96%E7%A8%8B/goroutine.html 1. main()函数默认的Goroutine 场景再现: 今天在看到这个教程的时候,在自己的电…...

nnUNet V2修改网络——暴力替换网络为UNet++
更换前,要用nnUNet V2跑通所用数据集,证明nnUNet V2、数据集、运行环境等没有问题 阅读nnU-Net V2 的 U-Net结构,初步了解要修改的网络,知己知彼,修改起来才能游刃有余。 U-Net存在两个局限,一是网络的最佳深度因应用场景而异,这取决于任务的难度和可用于训练的标注数…...

数据结构:递归的种类(Types of Recursion)
目录 尾递归(Tail Recursion) 什么是 Loop(循环)? 复杂度分析 头递归(Head Recursion) 树形递归(Tree Recursion) 线性递归(Linear Recursion)…...