`
micheal19840929
  • 浏览: 167357 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

网络游戏程序员须知 UDP vs TCP

阅读更多

这篇教程让我们就从最基本的网络数据收发开始谈起吧。其实这部分才是网络程序员应该做的最基础最简单的部分,但是这部分如果想要做好相对来说还是很有技巧和困难的。而且如果这部分你没做好,在多人对战类游戏中它带来的影响是极其恶劣的。
 你可能听说过端口这个概念,也可能知道TCP和UDP这两个概念。在做网络开发的的时候,我们首先要做的就是选择合适的协议。到底是TCP,还是UDP,或者是两者混合来用呢?这是一个问题。 
 其实来说,你的选择应该和你需要做的游戏类型相关。所以首先如果你是做网络游戏开发的,从现在起,我默认你对一些知名的网络游戏,比如COD,Quake, Unreal, CS已经很熟悉了。
 既然我们要谈网络游戏的协议选择,我们就先得对各种协议来个深入的了解,这样到底应该选择那种协议也就不言而喻了。
TPC/IP
TCP
 TCP解释为传输控制协议,IP解释为网络协议。他们合起来就完成了你日常网络的大部分工作,比如写email,看网页等。
 假如你曾经使用过TCP,那么你肯定知道TCP是可靠协议。即你先在两台机器间建立连接,然后你再在两台机器上开始传输数据,传输的过程和文件读写很像。你在一头写,在另外一头读而已。
 TCP协议是可靠而且有序的,这个意味着TCP负责保证你的数据可以完整而且有序的传输到另外一端。TCP是通过流的方式来传输数据的,这就意味这TCP会负责把你的数据切分,打包,然后具体传输。
 最后记住,TCP传输其实就和读写文件一样简单。  by rellikt
IP
 IP协议的抽象简单概念与IP协议底层传输的复杂真实实现形成鲜明对比。

 这里不需要任何连接的概念,数据包从一个电脑传输到另外一个电脑,就像你在上课的教室里面传纸条一样。你只要给出一个最终的目的地,包自然会传到那个终端,当然中间还会经过好多层的路由递送。

 你完全不能确定自己的包最终会不会到达目的地,只能希望它到达。如果你想知道包最终是否到达,那你就必须让接收者收到以后做出回复。

 事实上,整个过程可能还要更复杂一些,由于没有一台机器能够预知包可以最快到达的完整路由途径,所以有时候IP协议会将包复制多份发出,这些包会走不同的路由,因此他们一般也会在不同的时刻到达。

 你必须了解因特网路由问题的设计原则是自主组织路径,自主修复路径。所以当你思考问题的底层是如何实现时那是相当带劲的,当然你可以在相关的教科书上找到你想要的内容或者wiki。

UDP

 我们已经有一种可以像写文件一样稳定传输数据的方法了,如果我们还想要一种可以自由的收发包的方法,我们可以怎么做呢?

 这里我们可以用到UDP。UDP解释为“user datagram protocol”(用户自定义数据协议)。它和TCP类似也是建立在IP协议上的,不过他相比起TCP来在IP协议上只做了薄薄的一层协议。

 我们可以使用UDP协议直接对指定的IP和端口发包,比如 1.0.0.127:21(本机的FTP端口)。这个包会从发送者自己路由到接收者手中,当然也有可能半路丢失掉。

 在接收端,我们只要侦听相关端口就可以了,当有包从任何电脑发来的时候(这里没有连接的概念),我们在得到包的数据的同时,也得到了包的发送者的IP和端口数据,发送包的大小。

 UDP是不稳定可靠的传输协议,事实上大部分的包是会被送达的,但是你一般会有1%到5%的丢包率。甚至你会发现有段时间,你连一个包都收不到。路由路径上的那些机器出了点啥问题,谁知道呢?

 有时候收包的顺序和发包的顺序也是不同的,可能你发包的时候是1,2,3,4,5的顺序发。收到的包就变成1,3,4,5,2的顺序了。当然大部分时候这个顺序是不会乱的。但是这个谁都不敢打保票。

 UDP只能保证你一件事,那就是包的完整性。你如果发一个256bytes的包,那么对方收到的肯定也是一个256bytes的包。他不会只收到半个包之类的。当然这其实也是UDP唯一能保证你的事情。其他事情都要靠你自己去订制。 by rellikt

TCP vs UDP

 我们现在要做一个选择了。开发游戏到底是用UDP好呢?还是用TCP好呢?

 首先我们连列一下他们的有缺点:

TCP:

1. 基于连接的协议。

2. 可靠性和数据包的序列性是有保证的。

3. 自动为你的数据封包。

4. 确保包的流量不会超出你的网络链接的负载上限。

5. 简单易用,你只需要像写文件一样去读写就万事大吉了。

UDP:

1. 没有连接的概念,如果你想要,自己去实现去。

2. 没有关于可靠性和包序列性的保证,包可能会丢失,重复,乱序。

3. 你必须自己去封包。

4. 你必须自己确保自己的数据包不会流量过大从而导致超过链接负载上限。

5. 你必须自己处理包的丢失,重复,乱序的情况,如果你不想他们对你的程序造成麻烦,必须要自己实现代码来做出应对。

 这样一比,我们显然应该用那个TCP协议了。它完成了所有我们想要的特性,实在是太完美了,不是吗?by rellikt

TCP的真实工作情况

 TCP和UDP都是基于IP协议的,但是他们的本质确实截然不同的。UDP和它底层的IP协议类似,TCP却将很多东西进行了抽象,它使你在使用它的时候感觉就像读写一个文件一样,事实上他对你隐藏了很多复杂功能的实现细节。

 它到底是如何实现这些细节的呢?

 首先,TCP是流性质的协议,你将数据一个比特一个比特的写入流,然后TCP来确保他们会最终到底目的地。我们必须明确:TCP协议是基于IP协议的,IP协议是基于数据包的。这里TCP其实是把我们的数据流在底层进行了打包。事实上有一些TCP协议中的代码就是把我们的数据流进行排队,然后依次写入包中,当包写入了足够多的数据以后,它就会把包发走。

 这里就会导致一些问题,因为你不能控制底层的打包和传输,所以当很小的用户输入数据写入数据流的时候,TCP可能会凑满一定量的数据(比如100bytes),然后再打包发送。这样的话,在实时性上面就会很差,因为也许你会需要这些包越快到达越好。这些小延迟也许就会大大的影响你的游戏性。

 TCP中其实有一个TCP_NODELAY的选项,使用这个选项以后,你的数据就会跳过TCP的队列打包过程,直接发送。

 但是即使你使用的这个技术,你在多人网络游戏中还是会遇到不小的问题。

 这些问题源自于TCP实现可靠传输的机制。by rellikt

TCP是如何实现可靠传输的

 基本上来说,TCP将数据流中的数据做成封包,然后将他们通过不可靠的IP协议发送,然后在接收端重组这些包成为数据流。

 但是当一个包丢失的时候TCP会做些什么呢?当包重复和乱序的时候TCP又做了些什么呢?

 这里我不想做太深入的介绍,有兴趣的读者可以在wiki上找到他们需要的细节。大致来说,TCP发现丢包的时候,会要求发送者重发,重复的包会被丢弃,乱序包会被排序,事实上他就是这么保证传输的可靠性的。

 这里的丢包处理对游戏来说就很糟糕了。TCP中,如果你发现丢包了,必须等待发送者进行重发,在重发的包到来以前,即使有新包来,你也只能让他们在队列里等着,不能读取,这个等待的时间大概是ping值的1.5倍,如果重发的包再次丢失的话那就是3倍的时间。假设你的ping值是125ms,丢包一次就会延迟200ms左右,如果连续丢包就是400ms,这样的情况在大多数即时类游戏中是不能忍受的。

为何你不能选用TCP作为游戏开发的网络协议

 通过上面的论述,其实已经很明白了。在注重即时性的游戏中,对于延迟的要求是很苛刻的,我们可以丢包,但是如果我们收到了比丢掉的包更新的包的话,我们完全可以不管丢掉的包。我们只关注当前数据。

 这里我们来假设一个最简单的多人游戏的模型。比如一个FPS游戏,你在客户段每次将输入的数据(比如前进,跳跃,开火)发送到服务器端,然后服务器端将玩家当前的位置和情况发回给客户端来做显示。

 在这个最简单的模型中,只要有一个包丢失了,所有的东西都必须停下来等包的重发,任何操作都得停掉,你不能移动也不能射击。等到这个包到达的时候,你总算能继续操作了。但是可能你会发现还有一堆等等待重发的包在排队,于是你只好继续等,而且可能你收到的这个重发包对游戏来说已经失去时效性,完全没意义了。这样的游戏你能忍吗?

 不幸的是你对TCP协议的这个行为完全无能为力。这是TCP协议的本性,就是它保证了TCP协议的可靠性的。

 我们不需要这样无法订制的可靠性协议。我们需要自己进行订制丢包时的处理原则。这也是我们在开发游戏时,避免使用TCP协议的原因。

是否可以混合使用TCP和UDP协议呢?

 上面的结论中,我们可以明确知道,一些类似玩家操作,玩家位置的时效性相关数据,我们必须不能使用TCP协议。但是有些数据确是必须保证可靠性的,那我们是否可以使用TCP协议来做同步呢?

 这个想法是很好的,我们可以在玩家操作等即时性很强的数据上使用UDP协议,在玩家AI,数据加载,玩家对话等序列性很强的数据上使用TCP。如果你愿意的话,甚至可以为不同的AI创建不同的TCP连接,以免一个AI的丢包会影响的另外一个AI的即时性。

 看上去这是一个不错的思路。但是这仅仅是看上去而已。由于TCP和UDP都是基于IP协议的,事实上他们在底层会互相干扰。关于干扰的细节我这里就不详细论述了,想了解的读者可以阅读这篇文章。 by rellikt

结论

 我的建议是在游戏中仅仅使用UDP作为网络协议,即使要使用TCP也是自己在UDP的基础上实现一种类TCP的协议。这也是现代游戏中流行的网络架构。

 接下来的文章中我会介绍如何实现这套架构。下一篇会讲的比较实际点,讲关于如何使用UDP收发包。尽请期待。 by rellikt

 

 写了老半天,总算翻译完了,大家如果喜欢的话,下周有空我再翻译一章吧。就这样了。

 

本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/rellikt/archive/2010/08/21/5829020.aspx

分享到:
评论

相关推荐

    网络编程 udp tcp

    在学习网络编程的过程中,新手程序员经常会被推荐编写基于UDP和TCP的聊天程序。这种练习有助于深刻理解两种协议的差异。对于TCP聊天程序,编程者需要处理连接的建立和维护,但一旦连接稳定,数据的发送与接收就变得...

    tcp,udp测试工具

    在IT行业中,网络协议是构建通信系统的基础,TCP(传输控制协议)和UDP(用户数据报协议)是其中最常用且至关重要的两种协议。本文将深入探讨这两种协议以及相关的测试工具,以帮助理解和优化网络通信。 TCP(传输...

    基于udp和tcp的客户端和服务器

    在IT领域,网络通信是计算机科学的一个重要组成部分,TCP(传输控制协议)和UDP(用户数据报协议)是两种常用的数据传输协议。本项目“基于udp和tcp的客户端和服务器”涉及C++编程语言,利用socket接口实现这两种...

    一个简单的UDP/TCP编程库,可以简化UDP/TCP编程,提供堵塞和非堵塞模式

    标题中的“一个简单的UDP/TCP编程库”是指一个专门用于UDP和TCP网络通信的软件开发库,它旨在简化程序员在处理这些协议时的工作流程。这个库不仅支持两种基本的网络传输协议,UDP(用户数据报协议)和TCP(传输控制...

    c++ socket类封装(udp/tcp)

    在IT行业中,网络编程是不可或缺的一...无论是TCP还是UDP,理解并实现这样的封装都有助于提升程序员在网络编程领域的技能。通过学习和实践这些类,开发者能够构建自己的客户端-服务器应用,从而实现跨网络的数据交换。

    各种版本UDP和TCP小程序

    在IT行业中,网络通信是至关重要的一个领域,而TCP(传输控制协议)和UDP(用户数据报协议)是两种最常用的传输层协议。本项目"各种版本UDP和TCP小程序"聚焦于利用C#编程语言来实现这两种协议的客户端与服务器之间的...

    ethnet udp tcp.rar_TCP UDP SOCKET_TCP/UDP通信_tcp套接字_udp 程序

    学习和理解TCP和UDP套接字编程对于任何希望进行网络应用开发的程序员来说都是至关重要的。通过这些示例代码,开发者可以深入理解网络通信的基本原理,为构建自己的网络应用打下坚实的基础。无论是进行文件传输、游戏...

    udp-tcp.rar_TCP连接

    标题中的"udp-tcp.rar_TCP连接"表明我们将探讨两种网络传输协议:用户数据报协议(UDP)和传输控制协议(TCP)。这两种协议是互联网通信的基础,各有其特点和适用场景。 首先,UDP是一种无连接的协议,它不保证...

    udp文字tcp语音聊天程序

    在这个项目中,开发者利用了两种不同的网络传输协议——UDP(User Datagram Protocol)和TCP(Transmission Control Protocol),以及Windows多媒体库(winmm.dll)来实现声音的采集和播放。 UDP是一种无连接的、不可靠的...

    使用select的UDP和TCP回射程序

    本文将深入探讨“使用select的UDP和TCP回射程序”,这是《UNIX网络编程》一书中第8章的一个实践案例。该程序旨在展示如何通过select系统调用来同时管理多个网络连接,无论是基于用户数据报协议(UDP)还是传输控制...

    Linux网络编程示例,TCP和UDP

    本压缩包文件聚焦于Linux环境下的网络编程,主要涉及TCP(传输控制协议)和UDP(用户数据报协议)这两种常见的网络通信协议。以下是基于提供的信息,对Linux网络编程中的TCP和UDP知识点的详细阐述: 1. **Linux ...

    TCP-UDP数据包自动发送工具V3.01版本

    TCP-UDP数据包自动发送工具V3.01版本,作为一个实用工具,为程序员和网络爱好者提供了便捷的实验和学习平台。通过这款工具,我们可以直观地了解和操作TCP和UDP的数据包发送过程,增强对网络协议的理解。 TCP与UDP是...

    SOCKET + MFC 双向通信 UDP + TCP 版本

    综上所述,"SOCKET + MFC 双向通信 UDP + TCP 版本"的项目或教程旨在教给开发者如何在MFC环境中利用Socket实现可靠和高效的网络通信,涵盖TCP和UDP两种协议,适合那些希望在Windows平台上开发网络应用的程序员学习和...

    TCP和UDP socket调试工具V2.2

    在计算机网络编程中,TCP(Transmission Control Protocol)和UDP(User Datagram Protocol)是两种常见的传输层协议,它们在数据传输方面各自有着独特的特性和应用场景。本文将针对“TCP和UDP Socket调试工具V2.2”...

    tcp udp调试工具

    TCP/IP UDP Socket调试工具是程序员和网络管理员用来测试、诊断和优化网络应用程序的实用工具。它们可以帮助开发者查看网络通信过程中的数据包细节,包括发送和接收的数据、端口号、时间戳等,以便于定位问题和优化...

    TCP/UDP Socket调试工具

    Socket编程在IT行业中是网络通信的基础,而TCP和UDP是两种主要的传输层协议,用于在互联网上实现数据的可靠传输或无连接传输。本文将详细介绍TCP/UDP Socket调试工具的功能及其在IT工作中的应用。 TCP...

    程序员十大技术须知.doc

    【程序员十大技术须知】 1. XML(可扩展标记语言):XML 是现代软件开发中的核心技术,用于存储和传输结构化数据。它采用自描述的文本格式,允许数据以树形结构表示,既能存储结构化数据,也能处理非结构化数据。...

    网络通讯协议(tcp/ip/udp/ftp)(高清)

    网络通讯协议是互联网运作的核心,其中TCP/IP、UDP和FTP是最重要的几个协议,它们各自扮演着不同的角色,确保数据在网络中的高效、可靠传输。 TCP(Transmission Control Protocol,传输控制协议)是面向连接的协议...

    TCP/UDP工具

    标题中的“TCP/UDP工具”指的是在网络编程领域中用于测试和调试TCP(传输控制协议)和UDP(用户数据报协议)的应用程序。这些工具通常具备简单易用的图形用户界面(GUI),使得非专业程序员或者网络管理员也能进行...

    UDP和TCP以及Socket线程资料

    UDP(User Datagram Protocol)和TCP(Transmission Control Protocol)是网络通信中的两种主要传输协议,而Socket则是一种在应用层上实现这些协议的编程接口。在IT行业中,理解并熟练运用UDP、TCP和Socket线程是...

Global site tag (gtag.js) - Google Analytics