Unity网络编程(一)常见概念,第1张

一直用Http用多了 复习一下基础

Unity通讯一般分为2类

Http : 应用层 Unity内置的UnityWebRequest类进行通信(之前写过一个分发器垃圾框架)用于交互量比较小

Socket:传输层 比较底层 实现TCP/UDP 用于频繁的通信

这个是基于TCP 和IP传输不同消息

这个是三种常见的网络层次划分

基本数据单位为帧

主要的协议:以太网协议

基本数据单位为IP数据报;

IP协议(Internet Protocol,因特网互联协议)

ICMP协议(Internet Control Message Protocol,因特网控制报文协议)

ARP协议(Address Resolution Protocol,地址解析协议)

RARP协议(Reverse Address Resolution Protocol,逆地址解析协议)

包含的主要协议:TCP协议(Transmission Control Protocol,传输控制协议)、UDP协议(User Datagram Protocol,用户数据报协议)

数据传输基本单位为报文

包含的主要协议:

FTP(文件传送协议)、Telnet(远程登录协议)、DNS(域名解析协议)、SMTP(邮件传送协议),POP3协议(邮局协议),HTTP协议(Hyper Text Transfer Protocol)。

分配给用户上网使用的网际协议

目前IPv4多 比如19216811

新的IPv6(因为IPv4数量不够分配)如3ffe:3201:1401:1280:c8ff:fe4d:db39:1984。

Internet最基本的协议

TCP负责发现传输的问题,一有问题就发出信号,要求重新传输,直到所有数据安全正确地传输到目的地。

可靠的协议 通过三次握手建立的面向连接通信协议

3次握手 四次挥手 实习生常考

TCP连接建立过程(三次握手):

1首先Client端发送连接请求报文

2Server段接受连接后回复ACK报文,并为这次连接分配资源。

3Client端接收到ACK报文后也向Server段发生ACK报文,并分配资源,这样TCP连接就建立了。

TCP连接断开过程(四次挥手):

1Client端发起中断连接请求(FIN报文)

2Server端接到FIN报文后,发送ACK服务器还有消息没发完让Client待命,Client端就进入FIN_WAIT,继续等待Server端的FIN报文

3Server端确定数据已发送完成,则向Client端发送FIN报文,

4Client端收到FIN报文后发送ACK后进入TIME_WAIT状态,如果Server端没有收到ACK则可以重传,Server端收到ACK后 关闭,Client等待了2MSL后依然没有收到回复客户端也关闭

SYN:"synchronize"请求同步标志;;ACK:"acknowledge"确认标志";FIN:"Finally"结束标志。

为什么要三次握手?

防止因为网卡导致Sever收到多次Client请求 建立N个监听 造成资源浪费

为什么要四次挥手?

自己不请求直接关闭 但是服务器还能给你发数据 服务器浪费资源 而且客户端也会强行接收

使用TCP的协议:FTP(文件传输协议)、Telnet(远程登录协议)、SMTP(简单邮件传输协议)、POP3(和SMTP相对,用于接收邮件)、HTTP协议等。

面向无连接的通讯协议

UDP通讯时不需要接收方确认,属于不可靠的传输 会丢包

UDP与TCP位于同一层,但它不管数据包的顺序、错误或重发

主要用于面向查询---应答的程序

每个UDP报文分UDP报头和UDP数据区两部分

UDP报头由4个域组成,其中每个域各占用2个字节

(1)源端口号;

(2)目标端口号;

(3)数据报长度;

(4)校验值。

使用UDP协议包括:TFTP(简单文件传输协议)、SNMP(简单网络管理协议)、DNS(域名解析协议)、NFS、BOOTP。

超文本传输协议(HTTP,HyperText Transfer Protocol)是互联网上应用最为广泛的一种网络协议

HTTP协议特点:

简单快速 灵活 无连接 无状态 支持B/S(浏览器/服务器)及C/S(客户端/服务器)模式。

URL

和服务器有一些频繁的交互 用http时不时请求 叫轮询 效率低下

soket可以理解为插座 插头接上了可以保持通信

端口:

每个Socket连接都是从一台计算机网卡的一个端口连接到另外一台计算机网卡的某个端口。

IP是房子的话 端口就是门

TCP端口和UDP端口相互独立 如TCP255端口 和UDP255端口 不冲突

周知端口

范围从0到1023,其中80端口分配给WWW服务,21端口分配给FTP服务等。

浏览器的地址栏里输入一个网址的时候是不必指定端口号的,因为在默认情况下WWW服务的端口是“80”。

网络服务是可以使用其他端口号的 比如 网址:8080

但是有些系统协议使用固定的端口号,它是不能被改变的,比如139 端口专门用于NetBIOS与TCP/IP之间的通信,不能手动改变。

自己开发时尽量不要使用1024之下的端口,可能会与系统端口冲突。

服务端:

创建socket对象

bind:绑定IP地址和端口

listen:开始监听绑定的IP地址和端口,等待客户端的连接

accept:如果有客户端发起连接,通过accept接受连接请求,连接成功后会复制一个socket出来用于和当前接受连接的客户端进行通信。(服务端最初创建的那个socket只是用来监听并建立连接用的,实际和客户端通信并不是最初的socket,而是在accept这一步会自动创建一个新的socket出来和客户端通信。)

read/write:使用新的socket读写数据

close:关闭socket,如果关闭的是服务端的监听socket,则无法接收新的连接,但是已经创建的和客户端的连接不会被关闭。

客户端:

创建socket对象

connect:连接服务端,连接成功后系统会自动分配端口

read/write:连接成功后,就可以进行数据的读写了,这里读写使用的socket还是第一步创建的socket对象。

close:关闭连接。

如果收到了长度为0的数据,则代表远程socket关闭了连接。

服务器:

创建socket对象

bind:绑定IP和端口,用于接收数据(注意这里绑定完就可以直接接收数据了,并不需要等待连接)

read/write:读写数据

客户端:

创建socket对象

read/write:读写数据,不需要先建立连接,直接给对应的IP+端口发送数据即可。

由于没有建立连接以及连接的保障,UDP在传输效率上会很高

UDP有一个功能是TCP所不具备的,那就是广播功能(UDP可以将消息发送到在同一广播网络上的每个主机 CS、魔兽争霸局域网对战)。

HTTP/HTTPS(比http更安全):小游戏 网页 间歇性发送链接 偶尔延迟。

TCP长连接: 卡牌游戏 某些mmo 客户端和服务器都可以独立发包 偶尔延迟

UDP:动作游戏 mmo 枪战 客户端和服务器都可以独立发包 无法接受延迟

可以混合使用你的MMO客户端也许首先使用HTTP去获取上一次的更新内容,然后使用UDP跟游戏服务器进行连接。

现在也有kcp 就是tcp和udp结合 快速安全可靠

简单直接的长连接

可靠的信息传输

数据包的大小没有限制

坑多 断线检测、慢速客户端响应阻塞数据包,对开放连接的各种dos攻击,阻塞和非阻塞IO模型

丢包会有阻塞机制(一般是重发 tcp相反) 所以手机游戏ping跳1000就这个原因

只使用一个socket进行通信

快速

基于数据包构建

灵活 多种方式处理延迟

很多东西没有要自己构建

不可靠

丢包

客户端直接开始进行计算而不等待服务端确认是一种典型的隐藏延迟的技术(容易被抓包篡改)。

我们到底是使用TCP还是UDP取决于我们能否隐藏延迟。

比如TCP 在棋牌 卡牌游戏 卡1S无所谓 在动作游戏moba游戏就很致命

可靠的UDP/kcp和TCP不一样,要去实现一个特殊的阻塞控制,而且还要保证可靠性,也可以使用许多支持可靠通信的UDP库,但是库一般为了通用会降低某种新能,自己根据项目情况写可以发挥到极致

如果不知道用什么就TCP

如果对楼主有帮助,给个采纳好不,谢谢啦

Photon和KBEngineunity3d是最适用Unity3d游戏开发的两个服务器引擎,但它们还是有区别的,只有清楚地了解区别在哪才能正确使用,下面简单描述下两者的共同点和不同点。

语言

对于大部分的程序员语言简直就是宗教信仰。

Photon使用C#开发,当然使用者也是用C#进行各类游戏功能开发。前后端同种语言,这对使用Unity3d游戏开发也有很大的好处。

KBEngine使用C++开发,逻辑开发是用python,也是很不错很快速的。

开源与收费情况

Photon是Exit Games公司的产品,不开源,有好多种收费模式,官网上可以看到。开发阶段可以用免费的license,后期可以看流量用户活跃度来选择付费模式。后续的支持,似乎是免费的,你可以选择邮件或是到论坛发帖求助,当然是E文。

KBEngine是国人开发,开源免费,但从官网上并没有看到商业使用的案例。有中文论坛,你可以在论坛上向开发者求助。

虽然两者的模式不同,但作为一个Unity3d游戏开发者,我们最希望的其实是把游戏引擎当作一个安全稳定的黑箱。

操作系统

之前说了Photon使用C#开发很自然的,配套的工具也是使用C#,比如最重要的PhotonControl。所以开发环境和生产环境最好都是windows。

虽然在跨平台上有mono,在服务器代码部分是系统无关的,但是不管你信不信,我是不信它的一套窗体工具也能运行在Linux下。反正,官网说法是,开发和生产环境都是用windows。

KBEngine建议开发环境选择Windows,生产环境选择linux。毕竟你总不希望开一组服务器打开9个Console窗体,一不小心把哪个点X了吧~

协议

Photon有自己的序列化反序列化方式,你也可以使用protobuf这类的来做应用层传输协议。

KBEngine在这方面表示不支持自定义协议,它帮你选择了有效的方法来处理,如果你习惯了他规定的方式,会喜欢上的。

看法

在功能上,我毫无疑问地更喜欢KBEngine,脚本化和自动持久化是极富魅力的功能。而Photon几乎没做这方面的功能,可能和老外的观念有关系。就目前我对两者功能的理解看来,Photon其实是个和SuperSocket差不多的东西,而SS是作为轻量级服务器框架存在的,Photon却是说自己是Unity3d游戏引擎,除去提供的MMO示例代码(未解读),没看到什么游戏引擎的魅力。

参考如下代码:

/

  检测网络是否连接

  @return

 /

private boolean checkNetworkState() {

boolean flag = false;

//得到网络连接信息

manager = (ConnectivityManager) getSystemService(ContextCONNECTIVITY_SERVICE);

//去进行判断网络是否连接

if (managergetActiveNetworkInfo() != null) {

flag = managergetActiveNetworkInfo()isAvailable();

}

if (!flag) {

setNetwork();

} else {

isNetworkAvailable();

}

return flag;

}

/

  网络未连接时,调用设置方法

 /

private void setNetwork(){

ToastmakeText(this, "wifi is closed!", ToastLENGTH_SHORT)show();

AlertDialogBuilder builder = new AlertDialogBuilder(this);

buildersetIcon(Rdrawableic_launcher);

buildersetTitle("网络提示信息");

buildersetMessage("网络不可用,如果继续,请先设置网络!");

buildersetPositiveButton("设置", new OnClickListener() {

@Override

public void onClick(DialogInterface dialog, int which) {

Intent intent = null;

/

  判断手机系统的版本!如果API大于10 就是30+

  因为30以上的版本的设置和30以下的设置不一样,调用的方法不同

 /

if (androidosBuildVERSIONSDK_INT > 10) {

intent = new Intent(androidproviderSettingsACTION_WIFI_SETTINGS);

} else {

intent = new Intent();

ComponentName component = new ComponentName(

"comandroidsettings",

"comandroidsettingsWirelessSettings");

intentsetComponent(component);

intentsetAction("androidintentactionVIEW");

}

startActivity(intent);

}

});

buildersetNegativeButton("取消", new OnClickListener() {

@Override

public void onClick(DialogInterface dialog, int which) {

}

});

buildercreate();

buildershow();

}

/

  网络已经连接,然后去判断是wifi连接还是GPRS连接

  设置一些自己的逻辑调用

 /

private void isNetworkAvailable(){

State gprs = managergetNetworkInfo(ConnectivityManagerTYPE_MOBILE)getState();

    State wifi = managergetNetworkInfo(ConnectivityManagerTYPE_WIFI)getState();

    if(gprs == StateCONNECTED || gprs == StateCONNECTING){

     ToastmakeText(this, "wifi is open! gprs", ToastLENGTH_SHORT)show();

    }

    //判断为wifi状态下才加载广告,如果是GPRS手机网络则不加载!

    if(wifi == StateCONNECTED || wifi == StateCONNECTING){

     ToastmakeText(this, "wifi is open! wifi", ToastLENGTH_SHORT)show();

     loadAdmob();

    }

}

/

  在wifi状态下 加载admob广告

 /

private void loadAdmob(){

ll = (LinearLayout) findViewById(Ridload_ads);

llremoveAllViews();

adsView = new AdView(this, AdSizeBANNER, "a15194a1ac9505d");

lladdView(adsView);

adsViewloadAd(new AdRequest());

}

我这里为了能明确的说明wifi和GPRS连接的不同情况,设置了wifi连接下加载一个admob的广告条,GPRS下不去加载广告,这也算是在实际的开发中提升用户体验的一个途径吧! 

咱们需要使用 Socket Policy Server

Unity3D 包含了一个简单的 Socket Policy Server (sockpolexe)

需要在启动自己的应用之前启动它

下面是在咱们的环境中,通过命令行打开服务器(Unity 42, Win 8)

>cd C:\Program Files (x86)\Unity\Editor\Data\Tools\SocketPolicyServer\

>sockpolexe --all

需要在调用WebSocketConnect 之前,调用SecurityPrefetchSocketPolicy,如下SecurityPrefetchSocketPolicy (

,

);

现在用U3D开发的多人网游和手游很多,仙剑奇侠传OL等,自己搜。

但你从外部反推其制作是不太可能而且是费力费事的笨办法。

如果想学习尝试实践,正路是先从基本的U3D教材和资料入手,包括U3D官网和专业的论坛网站,再结合实际游戏案例去反思其构架。

但要注意,U3D开发入手快,特别是单机或联线的小游戏,但如果做多人网游,难点在于服务器端开发,而U3D在此方面很弱。

DABAN RP主题是一个优秀的主题,极致后台体验,无插件,集成会员系统
网站模板库 » Unity网络编程(一)常见概念

0条评论

发表评论

提供最优质的资源集合

立即查看 了解详情