[Unity3d] 网络开发基础【个人复习笔记/有不足之处欢迎斧正/侵删】

TCP/IP

TCP/IP协议是一 系列规则(协议)的统称,他们定义了消息在网络间进行传输的规则
是供已连接互联网的设备进行通信的通信规则

OSI模型只是一个基本概念,而TCP/IP协议是基于这个概念的具体实现

TCP和UDP协议

TCP:传输控制协议,面向连接,更安全,效率较低,一对一
UDP:用户数据报协议,无连接,不保证可靠性,效率较高,随意组合

TCP

是面向连接的协议,也就是说,在收发数据前,必须和对方建立可靠的连接
并且在消息传送过程中是有顺序的,并且是不会丢包(丢弃消息)的
如果某一条消息在传送过程中失败了,会重新发送消息,直到成功

三次握手 四次挥手

三次握手建立连接
第一次握手(C->S)
TCP连接请求
第二次握手(S->C)
TCP授予连接
第三次握手(C->S)
TCP确认连接

四次挥手断开连接
第一次挥手(C->S)
客户端告诉服务器数据已经发完,如果服务器还有消息就请快发完
第二次挥手(S->C)
服务器告诉客户端继续等待服务器的消息
第三次挥手(S->C)
服务器告诉客户端消息发送完成,可以正式断开连接
第四次挥手(C- ->S)
客户端告诉服务器等一会如果没有收到服务器回复就断开 了

提供可靠的服务,通过TCP连接传送的数据,做到无差错、不丢失、不重复、且按顺序到达

UDP

是一种无需建立连接就可以发送封装的IP数据包的方法,提供了面向事务的简单不可靠信息传送服务

具有资源消耗小,处理速度快的特点

UDP协议不像TCP协议需要建立连接有三次握手和四次挥手,当使用UDP协议发送信息时会直接把信息数据扔到网络上,所以也就造成了UDP的不可靠性。信息在这个传递过程中是有可能丢失的虽然UDP是一个不靠谱的协议,但是由于它不需要建立连接。也不会像TCP协议那样携带更多的信息,所以它具有更好的传输效率
 

网络游戏通信方案

Socket\HTTP\FTP

Socket:网络嵌套字

HTTP:超文本传输协议,主要完成短链接网络游戏需求

FTP:主要用来完成资源的下载和上传

byte [] ipAddress = new byte[]{118,102,111,11};
IPAddress ip1 =new IPAddress(ipAddress);
//用byte进行初始化//使用字符串进行初始化
IPAdress ip =IPAddress.Parse("118.102.111.11");//ipi27.0.0.1代表本机地址//命名空间System.NetIPEndPoint ipPoint =new IPEndPoint(0x79666F0B,8080);
IPEndPoint ipPoint2 =new IPEndPoint(IPAddress.Parse("118.102.111.11"),8080);

域名解析

也叫域名指向,服务器设置,域名配置

域名系统是互联网上的一种服务,管理名字与IP的对应关系

//IPHostEntry类
//域名解析后的返回值,可以获取该对象的IP地址主机名等信息//DNS类
print(Dns.GetHostName());
//获取主机名字//获取指定域名的ip地址
//可能会阻塞主线程
IPHostEntry entry = Dns.GetHostEntry("www.baidu.com");for(int i=0;i<entry.AddressList.Length;i++){print(entry.AddressList[i]);
}

Socket 套接字

c#用于提供网络通信的一个类

类名:Socket          命名空间:System.Net.Socket

Socket套接字时支持TCP/IP网络通信的基本操作单位

一个套接字包括:本机地址IP和端口/对方主机的IP地址和端口/双方通信的协议信息

一个Socket对象表示一个本地或者远程嵌套字信息,可被视为一个数据通道,链接服务器和客户端,可以接受数据的发送和接收

适合长连接的网络游戏

AddressFamily 网络寻址 枚举类型,决定寻址方案
        InterNetwork IPv4寻址
        InterNetwork6 IPv6寻址

SocketType 嵌套字枚举类型 决定使用的套接字类型
        Dgram 支持数据报,最大长度固定,无连接,不可靠的消息(主要用于UDP通信)
        Stream 支持可靠、双向、基于连接的字节流(主要用于TCP通信)

ProtocolType 协议类型枚举,决定套接字使用的通信协议

        TCP

        UDP

流套接字

主要实现TCP通信

数据报套接字

主要实现UDP通信

    //TCP流套接字Socket socketTcp = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);//UDPSocket socketUdp = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp);

实现服务端基本逻辑

 //创建套接字Socket socketTcp = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);//用bind方法将套接字与本地地址绑定//端口号要大于1024,且不能被占用IPEndPoint ipPoint = new IPEndPoint(IPAddress.Parse("127.0.0.1"), 8080);try{socketTcp.Bind(ipPoint);}catch (Exception e){return;}//用LISTEN监听socketTcp.Listen(1024);//accept等待客户端连入//建立连接,返回套接字Socket socketClient =socketTcp.Accept();//收发数据socketClient.Send(Encoding.UTF8.GetBytes("HFUT"));byte[] result=new byte[1024];int receiveNum=socketClient.Receive(result);//返回值为接收到的数量socketClient.Shutdown(SocketShutdown.Both);socketClient.Close();

实现客户端基本逻辑

//创建套接字Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);//与服务端相连IPEndPoint ipPoint =new IPEndPoint(IPAddress.Parse("127.0.0.0"), 8080);//上述应填写服务端的IP地址try{socket.Connect(ipPoint);}catch(SocketException e) {if(e.ErrorCode ==10061){}else{}}//收发数据byte[] data = new byte[1024];int num = socket.Receive(data);//发送数据socket.Send(Encoding.UTF8.GetBytes("HFUTER"));socket.Shutdown(SocketShutdown.Both);socket.Close();

区分消息类型

发送自定义类的二进制信息(需要继承BaseData类)

区分不同消息:给发送的消息添加标识,比如添加消息ID

例如选择int做消息ID,那么热前四个字节为消息ID,后面为数据类的内容

实现:

        1.创建一个消息基类,基类继承BaseData,基类添加获取消息的ID的方法或者属性

        2.让想要被发送的消息继承该类,实现序列化反序列化的方法

        3.写客户端和服务端收发处理消息的逻辑

分包与黏包

网络通信中由于各种因素造成的消息与消息之间出现的两种状态

分包:一个消息分成了多个消息进行发送

黏包:一个消息和另一个消息黏在一起

两者可能同时发生

解决办法:

可以通过消息的长度来判断是否出现分包或者黏包

为消息添加长度,消息长度记录消息的长度

心跳信息

在长连接中,客户端和服务端之间定期发送的一种特殊数据包,通知对方自己还在线,以保证长连接的有效性

其发送时间间隔是固定的,且持续,因此称之为心跳消息

心跳消息可以避免非正常关闭客户端时,服务器无法正常收到关闭连接消息,同时避免客户端长期不发消息,防火墙或者路由器会断开连接

当客户端主动断开时,服务端无法得知客户端已经主动断开

在客户端中可以使用Disconect方法,看是否因为之前直接Close()从而没有调用Disconect造成服务端无法及时获取状态(仍然无法准确地让服务端得知客户端已经断开连接)

可以考虑自定义退出消息

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class QuitMsg : BaseMsg
{//继承自消息基类//主动发送一条断开连接的消息给服务端//重写函数public override int GetBytesNum(){return 8;}public override int Reading(byte[] bytes, int beginIndex = 0){return 0;}public override byte[] Writing(){int index = 0;byte[] bytes = new byte[GetBytesNum()];WriteInt(bytes, GetID(), ref index);WriteInt(bytes, 0, ref index);return bytes;}public override int GetID(){return 1003;}
}

实现心跳消息

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class HeartMsg : BaseMsg
{//继承消息基类public override int GetBytesNum(){return 8;}public override int Reading(byte[] bytes, int beginIndex = 0){//不需要反序列化任何变量return 0;}public override byte[] Writing(){int index = 0;byte[] bytes = new byte[GetBytesNum()];WriteInt(bytes, GetID(), ref index);WriteInt(bytes, 0, ref index);return bytes;}public override int GetID(){return 999;}
}

客户端:定时发送消息

服务端:收心跳消息 记录收到消息的时间

异步通信

方法中的逻辑还没执行完,便继续执行后面的内容

UNITY中的协同程序中的某系异步方法,有的使用的是多线程有的使用的是迭代器分步执行

//线程回调
public void CountDownAsync(int second, UnityAction callBack){Thread t = new Thread(() =>{while (true){print(second);Thread.Sleep(1000);--second;if (second == 0)break;}
//?的目的是,如果是空的,则不会执行callBack?.Invoke();});t.Start();print("开始倒计时");}//让函数分布执行public async void CountDownAsync(int second){print("倒计时开始");//await->Task通过线程池开启一个线程
//本质上将函数分布执行await Task.Run(() =>{while (true){print(second);Thread.Sleep(1000);--second;if (second == 0)break;}});print("倒计时结束");}

UCP通信

udp不会对数据包进行合并发送,不会出现黏包问题

UDP是不可靠连接,消息传递过程中可能出现无序、丢包等情况

为了避免分包,建议控制消息的大小在MTU(最大传输单元)范围内

MTU:

最大传输单元,用来通知对方所能接受数据服务单元的最大尺寸

局域网:1472字节以内     互联网:548字节以内

如果想要发送的消息过大,可以进行手动分包,但是手动分包的前提是解决UDP的无序和丢包的问题

UDP流程更简单,效率高,但是不可靠

FTP

是支持因特网文件传输的各种规则的集合,使得文件可以被从一台主机拷贝到另一台主机上。此外,FTP还提供登录、文件查询以及其他绘画控制等功能

FTP本质上是TCP通信通过FTP,双发至少需要简历两个TCP连接,一个称为控制连接,一个成为数据连接

FTP的数据连接和控制连接方向一般是相反的

两种传输方式:1.ASCII传输    2.二进制传输

FTP关键类

1.NetworkCredential 通信凭证类 

命名空间:System.Net

用于Ftp文件传输,设置账号密码

NetworkCredrntial n =new NetworkCredrntial("HFUTER","hfuter");


2.FtpWebRequest Ftp文件传输客户端操作类

命名空间:System.Net

用于上传,下载,删除服务器上的文件
3.FtpWebResponse类

FtpWebRequest req = FtpWebRequest.Create(new Uri("ftp://127.0.0.1/Test.txt")) as FtpWebRequest;//创建req.Abort();//停止Stream s=  req.GetRequestStream();//获取流对象FtpWebResponse res= req.GetResponse() as FtpWebResponse;//返回FTP服务器响应

HTTP

HTTP超文本传输协议就是一个在网络中上传下载文件的一套规则

其本质也是TCP通信,因此不会丢包、不会乱序

1.以TCP方式工作:

HTTP/1.1支持持久连接(目前使用版本)

2.HTTP是无状态的:

HTTP不会记录客户端请求过的状态

3.元信息作为标头

主要数据前添加一段额外信息

请求类型和相应状态码

HTTP/1.1:GET\POST\HEAD\PUT\......

响应状态码:1xx\2xx\3xx\4xx\5xx

GET:请求获取特定的资源

POST:请求提交数据进行处理

HTTP常用状态码:

200 OK;404 NOT FOUND;405 不支持请求的方法;501 服务器不能识别请求挥着没有实现指定的请求

关键类

HttpWebRequest类:

命名空间:System.Net
HttpWebRequest是主要用于发送客户端请求的类
主要用于:发送HTTP客户端请求给服务器,可以进行消息通信、上传、下载等等操作
 

//创建新的WebRequest,用于进行HTTP相关操作HttpWebRequest req = HttpWebRequest.Create(new Uri("http://192.168.50.109:8000/Http_Server/")) as HttpWebRequest;//终止传输 req.Abort();//获取用于上传的流Stream s = req.GetRequestStream();//返回HTTP服务器响应HttpWebResponse res = req.GetResponse() as HttpWebResponse;//异步获取用于上传的流req.BeginGetRequestStream()异步获取返回的HTTP服务器响应//req.BeginGetResponse()

HttpWebResponse类:


命名空间:System.Net
主要用于获取服务器反馈信息的类,可以通过HttpWebRequest对象中的GetResponse()方法获取。当使用完毕时,要使用Close释放

POST

上传文件到HTTP资源服务器需要遵守的规则:

1:ContentType = "multipart/form-data; boundary=边界字符串";

2:上传的数据必须按照格式写入流中

3:保证服务器允许上传

4:写入流前需要先设置ContentLength内容长度

WWW类

WWW是Unity提供给程序员简单的访问网页的类,可以通过该类下载和上传一些数据,在使用http协议时,默认的请求类型是Get,如果想要Post上传,需要配合WWWFrom类使用(该类在较新Unity版本中会提示过时,但是仍可以使用,新版本将其功能整合进了UnityWebRequest类)

主要支持的协议:

1.http://和https:// 超文本传输协议
2.ftp:// 文件传输协议(但仅限于匿名下载)
3.file:// 本地文件传输协议,可以使用该协议异步加载本地文件(PC、IOS、Android都支持)

//创建
WWW www = new WWW("");//从下载数据返回一个音效切片AudioClip对象
www.GetAudioClip();//用下载数据中的图像来替换现有的一个Texture2D对象
Texture2D tex = new Texture2D(100, 100);//从缓存加载AB包对象,如果该包不在缓存则自动下载存储到缓存中,以便以后直接从本地缓存中加载
WWW.LoadFromCacheOrDownload("", 1);

UnityWebRequest

是一个Unity提供的一个模块化的系统类,用于构成HTTP请求和处理HTTP响应,主要目标是让Unity游戏和Web服务端进行交互,将之前WWW的相关功能都集成在了其中(新版本中都建议使用UnityWebRequest类来代替WWW类)

注意:
1.UnityWebRequest和WWW一样,需要配合协同程序使用
2.UnityWebRequest和WWW一样,支持http、ftp、file协议下载或加载资源
3.UnityWebRequest能够上传文件到HTTP资源服务器

UnityWebRequest类的常用操作:

        //1.获取文本或2进制StartCoroutine(LoadText());//2.获取纹理StartCoroutine(LoadTexture());//3.获取AB包StartCoroutine(LoadAB());IEnumerator LoadText(){UnityWebRequest req = UnityWebRequest.Get("http://192.168.50.109:8000/Http_Server/test.txt");//就会等待 服务器端响应后 断开连接后 再继续执行后面的内容yield return req.SendWebRequest();//如果处理成功 结果就是成功枚举if(req.result == UnityWebRequest.Result.Success){//独立的处理对象print(req.downloadHandler.text);byte[] bytes = req.downloadHandler.data;print("字节数组长度" + bytes.Length);}else{print("获取失败:" + req.result + req.error + req.responseCode);}}IEnumerator LoadTexture(){//UnityWebRequest req = UnityWebRequestTexture.GetTexture("http://192.168.50.109:8000/Http_Server/HFUTER.jpg");//UnityWebRequest req = UnityWebRequestTexture.GetTexture("ftp://127.0.0.1/HFUTER.jpg");UnityWebRequest req = UnityWebRequestTexture.GetTexture("file://" + Application.streamingAssetsPath + "/test.png");yield return req.SendWebRequest();if (req.result == UnityWebRequest.Result.Success){//(req.downloadHandler as DownloadHandlerTexture).texture//DownloadHandlerTexture.GetContent(req)//image.texture = (req.downloadHandler as DownloadHandlerTexture).texture;image.texture = DownloadHandlerTexture.GetContent(req);}elseprint("获取失败" + req.error + req.result + req.responseCode);}IEnumerator LoadAB(){UnityWebRequest req = UnityWebRequestAssetBundle.GetAssetBundle("http://192.168.50.109:8000/Http_Server/lua");req.SendWebRequest();while (!req.isDone){print(req.downloadProgress);print(req.downloadedBytes);
//每帧执行yield return null;}//yield return req.SendWebRequest();print(req.downloadProgress);print(req.downloadedBytes);if (req.result == UnityWebRequest.Result.Success){//AssetBundle ab = (req.downloadHandler as DownloadHandlerAssetBundle).assetBundle;AssetBundle ab = DownloadHandlerAssetBundle.GetContent(req);print(ab.name);}elseprint("获取失败" + req.error + req.result + req.responseCode);}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

VsCode配置PCL、Open3D自动补全

写在前面 本文内容 在VsCode上开发PCL、Open3D相关代码&#xff0c;代码自动补全 Open3D、PCL的安装使用见各个版本的Open3D、PCL的编译、使用教程 平台/环境 windows11(windows10): visual studio 2022&#xff1b;cmake 3.22; VsCode 通过cmake构建项目&#xff1b; 转载请…

Excel MATCH函数 两张顺序不同表格,统一排序

目录 一. 背景二. 添加辅助列,使用MATCH函数生成排序条件三. 效果 一. 背景 有如下图所示的两张表格&#xff0c;分别记录着同一批人的1月份和2月份的工资。表格A和表格B中的姓名列相同&#xff0c;工资列数据不同现在要求参考表格A中的姓名列对表格B中的数据进行排序&#xf…

C语言:预处理

C语言&#xff1a;预处理 预定义符号#define定义常量定义宏宏与函数对比 #操作符##操作符条件编译头文件包含库文件包含本地文件包含嵌套文件包含 预定义符号 C语⾔设置了⼀些预定义符号&#xff0c;可以直接使⽤&#xff0c;预定义符号也是在预处理期间处理的。 __FILE__ //…

在你的 Vue + Electron 项目里,引入 ESLint

因为我的项目是基于 Electron 平台的 Web 应用&#xff0c;使用 Vue 3 实现&#xff0c;而且用了 TypeScript&#xff0c;所以&#xff0c;在引入 ESLint 的时候&#xff0c;要考虑好几种规范的问题。 文章目录 零、简介1. 规则2. 配置文件3. 共享配置4. 插件5. 解析器6. 自定义…

Vue开发实例(九)动态路由实现左侧菜单导航

之前在【Vue开发实例&#xff08;六&#xff09;实现左侧菜单导航】文中实现了菜单的导航&#xff0c;本篇是在那个基础上改造的。 动态路由实现左侧菜单导航 一、动态菜单创建二、根据菜单数据来创建路由三、添加路由已加载标记&#xff0c;省的每次点击菜单都要加载 一、动态…

【设计模式 01】单例模式

单例模式&#xff0c;是一种创建型设计模式&#xff0c;他的核心思想是保证一个类只有一个实例&#xff08;即&#xff0c;在整个应用程序中&#xff0c;只存在该类的一个实例对象&#xff0c;而不是创建多个相同类型的对象&#xff09;&#xff0c;并提供一个全局访问点来访问…

java012 - Java集合基础

1、集合基础 1.1 集合概述 引用数据类型包括&#xff1a;类、接口、数组[] 1.2 ArrayList构造和添加方法 代码&#xff1a; 空集合对象&#xff1a;[] add() add(int index,E element): 1.3 ArrayList集合常用方法

Stable Diffusion 模型分享:Henmix_Real(人像、真实、写真、亚洲面孔)

本文收录于《AI绘画从入门到精通》专栏,专栏总目录:点这里。 文章目录 模型介绍生成案例案例一案例二案例三案例四案例五案例六案例七案例八下载地址模型介绍 作者述:这个模型试图改

Springboot教程(五)——单元测试

idea中一般使用JUnit进行单元测试 基本使用 我们可以在idea的test文件夹下的XXXXApplicationTests内进行单元测试&#xff1a; 可以在Test标注的方法上写测试代码&#xff1a; SpringBootTest class C0101ApplicationTests {Testfun contextLoads() {println("Hello …

基础二分学习笔记

模板 : 个人倾向第一种 ; 整数二分 : 最大化查找 : 可行区域在左侧 : 查找最后一个<q的数的下标 : int find(int q){// 查找最后一个 < q 的下标 int l 0 , r n 1 ;while(l 1 < r){int mid l r >> 1 ;if(a[mid]<q) l mid ;else r mid ;}return…

土壤类型数据

国家地球系统科学数据中心

AGM CPLD (AGRV2K )的时钟(外部时钟和片上内部振荡器)

AGM CPLD &#xff08;AGRV2K &#xff09;的时钟(外部时钟和片上内部振荡器) 外部晶振 与 内部振荡器&#xff1a; mcu 和 cpld 联合编程时&#xff0c; 整颗芯片需要一颗外部晶振。 &#xff08;芯片有内部振荡器&#xff0c; 但误差较大&#xff0c; 校准后 5%以内误差&…

Electron通过预加载脚本从渲染器访问Node.js

问题&#xff1a;如何实现输出Electron的版本号和它的依赖项到你的web页面上&#xff1f; 答案&#xff1a;在主进程通过Node的全局 process 对象访问这个信息是微不足道的。 然而&#xff0c;你不能直接在主进程中编辑DOM&#xff0c;因为它无法访问渲染器 文档 上下文。 它们…

【软考】数据库的三级模式

目录 一、概念1.1 说明1.2 数据库系统体系结构图 二、外模式三、概念模式四、内模式 一、概念 1.1 说明 1.数据的存储结构各不相同&#xff0c;但体系结构基本上具有相同的特征&#xff0c;采用三级模式和两级镜像 2.数据库系统设计员可以在视图层、逻辑层和物理层对数据进行抽…

matplotlib散点图

matplotlib散点图 假设通过爬虫你获取到了北京2016年3, 10月份每天白天的最高气温(分别位于列表a, b), 那么此时如何寻找出气温和随时间(天)变化的某种规律? from matplotlib import pyplot as pltx_3 range(1, 32) x_10 range(51, 82)y_3 [11,17,16,11,12,11,12,6,6,7,8…

试手一下CameraX(APP)

书接上回。 首先还是看谷歌的官方文档&#xff1a; https://developer.android.com/media/camera/camerax?hlzh-cn https://developer.android.com/codelabs/camerax-getting-started?hlzh-cn#1 注&#xff1a;这里大部分内容也来自谷歌文档。 官方文档用的是Kotlin&…

Day14:信息打点-主机架构蜜罐识别WAF识别端口扫描协议识别服务安全

目录 Web服务器&应用服务器差异性 WAF防火墙&安全防护&识别技术 蜜罐平台&安全防护&识别技术 思维导图 章节知识点 Web&#xff1a;语言/CMS/中间件/数据库/系统/WAF等 系统&#xff1a;操作系统/端口服务/网络环境/防火墙等 应用&#xff1a;APP对象/…

小程序图形:echarts-weixin 入门使用

去官网下载整个项目&#xff1a; https://github.com/ecomfe/echarts-for-weixin 拷贝ec-canvs文件夹到小程序里面 index.js里面的写法 import * as echarts from "../../components/ec-canvas/echarts" const app getApp(); function initChart(canvas, width, h…

Vscode 使用SSH远程连接树莓派的教程(解决卡在Downloading with wget)

配置Vscode Remote SSH 安装OpenSSH 打开Windows开始页面&#xff0c;直接进行搜索PowerShell&#xff0c;打开第一个Windows PowerShell&#xff0c;点击以管理员身份运行 输入指令 Get-WindowsCapability -Online | ? Name -like OpenSSH* 我是已经安装好了&#xff0c;…

学会玩游戏,智能究竟从何而来?

最近在读梅拉妮米歇尔《AI 3.0》第三部分第九章&#xff0c;谈到学会玩游戏&#xff0c;智能究竟从何而来&#xff1f; 作者: [美] 梅拉妮米歇尔 出版社: 四川科学技术出版社湛庐 原作名: Artificial Intelligence: A Guide for Thinking Humans 译者: 王飞跃 / 李玉珂 / 王晓…