引言:
成都創(chuàng)新互聯(lián)公司專業(yè)為企業(yè)提供蒙陰網站建設、蒙陰做網站、蒙陰網站設計、蒙陰網站制作等企業(yè)網站建設、網頁設計與制作、蒙陰企業(yè)網站模板建站服務,十年蒙陰做網站經驗,不只是建網站,更提供有價值的思路和整體網絡服務。
前一個專題簡單介紹了TCP編程的一些知識,UDP與TCP地位相當?shù)牧硪粋€傳輸層協(xié)議,它也是當下流行的很多主流網絡應用(例如QQ、MSN和Skype等一些即時通信軟件傳輸層都是應用UDP協(xié)議的)底層的傳輸基礎,所以在本專題中就簡單介紹下UDP的工作原理和UDP編程的只是,希望可以對剛接觸網絡編程的朋友起到入門的作用。
一、UDP介紹
UDP和TCP都是構建在IP層之上傳輸層的協(xié)議,但UDP是一種簡單、面向數(shù)據報(Sock_Dgram)的無連接協(xié)議,提供的是不一定可靠的傳輸服務。
然而TCP是一種面向連接、可靠的,面向字節(jié)流(Sock_Stream)的傳輸協(xié)議,對于“無連接”是指在正式通信前不必與對方先建立連接,不管對方狀態(tài)如何都可以直接發(fā)送過去(就如QQ中通過QQ號查看好友后發(fā)送添加好友請求,此間不需要考慮對方的狀態(tài)如何,都照樣發(fā)送請求)。從UDP和TCP的定義中就可以看出它們兩者的區(qū)別了,(1)UDP的可靠性不如TCP,因為TCP傳輸前要首先建立連接,這樣就增加了TCP傳輸?shù)目煽啃裕訳DP也被稱為不可靠的傳輸協(xié)議,關于TCP的介紹可以看我上一篇博客的介紹。
TCP和UDP還有另外一個區(qū)別。(2)UDP不能保證有序傳輸。即UDP不能確保數(shù)據的發(fā)送和接收順序。
下面就來看看UDP協(xié)議的工作原理,對UDP的工作原理有一個好的理解,對后面介紹的UDP編程也是一個好的基礎。
1.1 UDP的工作原理
UDP將網絡數(shù)據流量壓縮成數(shù)據報的形式,每一個數(shù)據報用8個字節(jié)(8 X 8位=64位)描述報頭信息,剩余字節(jié)包含具體的傳輸數(shù)據。UDP報頭(只有8個字節(jié))相當于TCP的報頭(至少20個字節(jié))很短,UDP報頭由4個域組成,每個域各占2個字節(jié),具體為源端口、目的端口、用戶數(shù)據報長度和校驗和,
具體結構見下圖(下面也貼出了TCP報文的結構圖,與UDP數(shù)據報做一個對比的作用):
UDP協(xié)議和TCP協(xié)議都使用端口號為不同的應用保留其各自的數(shù)據傳輸通道這一機制,數(shù)據發(fā)送方將UDP數(shù)據報通過源端口發(fā)送出去,而數(shù)據接收方則通過目標端口接收數(shù)據。
1.2 UDP的優(yōu)勢
前面介紹中說UDP相對于TCP是不可靠的,不能保證有序傳輸的傳輸協(xié)議,然而UDP協(xié)議相對于TCP協(xié)議的優(yōu)勢在哪里呢?,
UDP相對于TCP的優(yōu)勢主要有三個方面的:
(1)UDP速度比TCP快。
由于UDP不需要先與對方建立連接,也不需要傳輸確認,因此其數(shù)據的傳輸速度比TCP快很多。對于一些著重傳輸性能而不是傳輸完整性的應用(網絡音頻播放、視頻點播和網絡會議等),使用UDP協(xié)議更加適合,因為它傳輸速度快,使通過網絡播放的視頻音質好、畫面清晰。
(2)UDP有消息邊界。
通過UDP協(xié)議進行傳輸?shù)陌l(fā)送方對應用程序交下來的報文,在添加首部后就向下直接交付給IP層。既不拆分也不合并,而是保留這些報文的邊界,所以使用UDP協(xié)議不需要像TCP那樣考慮消息邊界的問題,這樣就使得UDP編程相對于TCP在接收到的數(shù)據處理方面要簡單的多。(對于TCP消息邊界的問題可以查看相關的文檔,在這里我就不列出來了)
(3)UDP可以一對多傳輸
由于傳輸數(shù)據部建立連接,也就不需要維護連接狀態(tài),因此一臺服務器可以同時向多個客戶端發(fā)送相同的信息。利用UDP可以使用廣播或者組播的方式同時向子網的所有客戶端進程發(fā)送信息,廣播和組播的介紹放到后面TCP編程中介紹。
上面介紹了UDP協(xié)議相對于TCP協(xié)議的優(yōu)勢,其中速度快是UDP的最重要的優(yōu)勢,也是像一些網絡會議、即時通信軟件傳輸層選擇UDP協(xié)議進行傳輸?shù)脑蛩凇?/p>
二、.net平臺對UDP編程的支持
介紹完UDP相對于TCP的優(yōu)勢后,當然很希望在.net平臺下開發(fā)一個基于UDP協(xié)議的一個應用了,然后.net平臺下對UDP編程也做了很好的支持,為我們開發(fā)基于UDP協(xié)議的網絡應用提供很多方便之處,下面就簡單介紹.net平臺下對UDP編程的支持(主要介紹提供的類來對UDP協(xié)議進行編程)。
.net類庫中的UdpClient類對基礎的Socket進行了封裝,這樣就在發(fā)送和接受數(shù)據時不需要考慮底層套接字的收發(fā)時處理的一些細節(jié)問題,這樣為UDP編程提供了方便,也可以提高開發(fā)效率(感覺net就是做這樣的事情的,對一些底層的實現(xiàn)進行封裝,方便我們的調用,這也體現(xiàn)了面向對象語言的封裝特性)對于這個的具體的使用我就不做過多的介紹的,在后面的UDP編程的實現(xiàn)部分將會對該類中主要方法的使用,大家可以查看MSDN來查看該類中其他成員的使用: http://msdn.microsoft.com/zh-cn/library/System.Net.Sockets.UdpClient.aspx
三、UDP編程的具體實現(xiàn)
由于UDP進程在通信之前是不需要建立連接,消息接收方可能并不知道是誰給它發(fā)的消息,因此UDP編程分為兩種模式:一種“實名發(fā)送”,即接收方可以由收到的消息得知發(fā)送方進程端口,另外一種則為“匿名發(fā)送”,即接收方并不知道發(fā)給它信息的遠程進程究竟來自哪個端口。下面通過一個winform 程序來演示下UDP的編程:
實現(xiàn)代碼:
using System;using System.Net;using System.Net.Sockets;using System.Text;using System.Threading;using System.Windows.Forms; namespace UDPClient { public partial class frmUdp : Form { private UdpClient sendUdpClient; private UdpClient receiveUpdClient; public frmUdp() { InitializeComponent(); IPAddress[] ips= DNS.GetHostAddresses(""); tbxlocalip.Text= ips[3].ToString(); int port = 51883; tbxlocalPort.Text= port.ToString(); tbxSendtoIp.Text= ips[3].ToString(); tbxSendtoport.Text= port.ToString(); } // 接受消息 private void btnReceive_Click(object sender, EventArgs e) { // 創(chuàng)建接收套接字 IPAddress localIp = IPAddress.Parse(tbxlocalip.Text); IPEndPoint localIpEndPoint= new IPEndPoint(localIp, int.Parse(tbxlocalPort.Text)); receiveUpdClient= new UdpClient(localIpEndPoint); Thread receiveThread= new Thread(ReceiveMessage); receiveThread.Start(); } // 接收消息方法 private void ReceiveMessage() { IPEndPoint remoteIpEndPoint= new IPEndPoint(IPAddress.Any, 0); while (true) { try { // 關閉receiveUdpClient時此時會產生異常 byte[] receiveBytes = receiveUpdClient.Receive(ref remoteIpEndPoint); string message = Encoding.Unicode.GetString(receiveBytes); // 顯示消息內容 ShowMessageforView(lstbxMessageView, string.Format("{0}[{1}]", remoteIpEndPoint, message)); } catch { break; } } } // 利用委托回調機制實現(xiàn)界面上消息內容顯示 delegate void ShowMessageforViewCallBack(ListBox listbox, string text); private void ShowMessageforView(ListBox listbox, string text) { if (listbox.InvokeRequired) { ShowMessageforViewCallBack showMessageforViewCallback= ShowMessageforView; listbox.Invoke(showMessageforViewCallback,new object[] { listbox, text }); } else { lstbxMessageView.Items.Add(text); lstbxMessageView.SelectedIndex= lstbxMessageView.Items.Count - 1; lstbxMessageView.ClearSelected(); } } private void btnSend_Click(object sender, EventArgs e) { if (tbxMessageSend.Text == string.Empty) { MessageBox.Show("發(fā)送內容不能為空","提示"); return; } // 選擇發(fā)送模式 if (chkbxAnonymous.Checked == true) { // 匿名模式(套接字綁定的端口由系統(tǒng)隨機分配) sendUdpClient = new UdpClient(0); } else { // 實名模式(套接字綁定到本地指定的端口) IPAddress localIp = IPAddress.Parse(tbxlocalip.Text); IPEndPoint localIpEndPoint= new IPEndPoint(localIp, int.Parse(tbxlocalPort.Text)); sendUdpClient= new UdpClient(localIpEndPoint); } Thread sendThread= new Thread(SendMessage); sendThread.Start(tbxMessageSend.Text); } // 發(fā)送消息方法 private void SendMessage(object obj) { string message = (string)obj; byte[] sendbytes = Encoding.Unicode.GetBytes(message); IPAddress remoteIp= IPAddress.Parse(tbxSendtoIp.Text); IPEndPoint remoteIpEndPoint= new IPEndPoint(remoteIp, int.Parse(tbxSendtoport.Text)); sendUdpClient.Send(sendbytes, sendbytes.Length, remoteIpEndPoint); sendUdpClient.Close(); // 清空發(fā)送消息框 ResetMessageText(tbxMessageSend); } // 采用了回調機制 // 使用委托實現(xiàn)跨線程界面的操作方式 delegate void ResetMessageCallback(TextBox textbox); private void ResetMessageText(TextBox textbox) { // Control.InvokeRequired屬性代表 // 如果控件的處理與調用線程在不同線程上創(chuàng)建的,則為true,否則為false if (textbox.InvokeRequired) { ResetMessageCallback resetMessagecallback= ResetMessageText; textbox.Invoke(resetMessagecallback,new object[] { textbox }); } else { textbox.Clear(); textbox.Focus(); } } // 停止接收 private void btnStop_Click(object sender, EventArgs e) { receiveUpdClient.Close(); } // 清空接受消息框 private void btnClear_Click(object sender, EventArgs e) { this.lstbxMessageView.Items.Clear(); } } }
運行結果:
實名發(fā)送:
在本地運行本程序的三個進程(分別為A,B,C),把進程C做為接受進程,進程A和進程B都向進程C發(fā)信息,進程A和進程分別綁定端口號為11883和21883,發(fā)送到端口都為51883,配置界面如下:
首先不勾選“匿名”復選框,在進程C中點擊“接收”按鈕開啟接受線程,在A進程和B進程中發(fā)送消息框里分別輸入你好,我是1和你好,我是2,然后點擊發(fā)送按鈕,此時在進程中就可以看到進程A和進程B發(fā)來的消息,如下圖:
從圖中可以看出每條消息之前都顯示了消息的準確來源(包括消息進程鎖在的Ip地址和端口號)
匿名發(fā)送:
下面把“匿名”復選框勾上后,再按照前面的步驟將得到下面的結果:
從圖中結果可以看出此時列表中顯示的消息來源的進程端口號分別為49439和49440,而不是發(fā)送消息進程的真實端口(11883和21883)
這種UDP只能辨別消息源主機的Ip地址,而無法知道發(fā)消息的進程究竟是哪個端口稱為“匿名發(fā)送”。正如我們平時發(fā)手機短信一樣,如果我們把認識的名字和電話號碼預先存在通訊錄里,當一發(fā)來信息,接受方馬上就可以從來電顯示中看到是誰發(fā)來的(實名模式);但是如果是陌生人發(fā)來信息或者廣告等信息時,僅看來電顯示,根本不知道對方是誰(匿名模式),QQ發(fā)消息也是一樣的道理。
四、UDP廣播和組播
前面UDP的實現(xiàn)中發(fā)送數(shù)據使用的都是一對一(單播)的通信方式,即只將數(shù)據發(fā)送到某一個進程。前面提到UDP可以實現(xiàn)一對多的傳輸方式,即通過廣播和組播把數(shù)據發(fā)送給一組進程。下面就介紹下UDP廣播和組播的相關知識。
4.1 廣播和組播的基本概念
雖然利用TCP協(xié)議可以保證數(shù)據的可靠、有序的傳輸,但是TCP僅支持一對以的傳輸,而且傳輸時需要在發(fā)送端和每一個接受端之間建立單獨的數(shù)據通信通道,如果需要實現(xiàn)網絡會議、網絡視頻的點播等功能時要向大量主機發(fā)送相同的數(shù)據包,如果采用單播方式逐個節(jié)點傳輸?shù)脑?,將會給發(fā)送方帶來網絡堵塞等問題,此時可以考慮實現(xiàn)UDP的多播方式——即廣播和組播來實現(xiàn)這樣的功能(一對多通信分為廣播和組播兩種形式)。
廣播是指同時向子網中的多臺計算機發(fā)送消息,并且所有子網中的計算機都可以接收到發(fā)送方發(fā)來的消息,每個廣播消息包含一個特殊的IP地址,這個IP的中子網內主機標志部分的二進制都為1,例如,子網掩碼為255.255.255.0,對于子網192.168.0,則這個IP地址為192.168.0.255.
然后廣播消息又分為本地廣播和全球廣播兩種類型, 本地廣播是指向子網中的所有計算機發(fā)送廣播消息,其他網絡不會受到本地廣播的影響。
IP地址分為兩部分——網絡標志部分和主機標志部分,這兩部分是靠子網掩碼來區(qū)分的,主機標記部分二進制全部為1的地址成為本地廣播地址。例如:
A類網絡192.168.0.0,使用子網掩碼255.255.0.0,則本地廣播地址為:
對于IPv4來說,全球廣播使用所有位全為1的IP地址,即255.255.255.255,這個廣播地址代表數(shù)據報的目的地是網絡上所有設備,但是由于路由器會自動過濾全球廣播,所以使用這個地址根本就沒有任何意義。
然后當接收者分布于多個不同的子網時,廣播將不再適用,此時可以通過組播的方式來實現(xiàn),組播也叫多路廣播,組播是將信息從一臺計算機發(fā)送到本網或全網內指定的計算機上,即發(fā)送到那些加入了指定組播組的計算機上,每臺計算機都可以通過程序隨時加入某個組播組中,也可以隨時退出來, 就像我們開網了會議一樣,可以隨時加入會議室進行開會,會議結束和會議進行中都可以隨意的退出來。
4.2 加入和退出組播組
組播組又稱為多路廣播組,組播地址的范圍在224.0.0.0到239.255.255.255的D類IP地址(至于這個概念大家可以百度百科里面就查看)。任何發(fā)送到組播地址的消息都會被發(fā)送到組內所有成員設備上,組可以使永久的也可以是臨時,大多數(shù)我們使用的都是臨時的,僅在有成員的時候才存在。
使用組播時,注意生命周期(TTL,Time to live)的設,TTL值表示允許路由器轉發(fā)的最大次數(shù),當達到這個最大值時,數(shù)據包就會被丟棄,TTL的默認值為1,設置為1時表明只能在子網中發(fā)送數(shù)據
加入組播組:
UdpClient類提供了JoinMulticastGroup方法,用于將UdpClient加入到使用指定的IPAddress的組播組中,調用該方法后,基礎的Socket會自動向路由器發(fā)送數(shù)據包,用于請求成為組播組的成員,如果成為組播組成員,就可以接收該組播組的數(shù)據報。至于具體方法的時候會在后面實現(xiàn)UDP廣播程序中會用到,另外大家也可以查看MSDN,所以這里我就不再列出來了,只是指出這個方法的作用,讓大家知道有這么個方法來調用。
退出組播組:
同樣利用UdpClient的DropMulticastGroup方法,可以退出組播組,調用該方法后,基礎Socket會自動向路由器發(fā)送數(shù)據包,用于請求從指定的組播組里退出,從組中回收UdpClient對象之后,將不再接受發(fā)送到該組播組的數(shù)據報。
五、總結
由于時間的關系,這篇文章就介紹到這里的,至于實現(xiàn)UDP廣播的程序放在后面一個專題里面的,前面也對廣播和組播的概念進行了簡單的介紹,相信大家也對廣播和組播有了個簡單的認識(廣播組和組播組說白了就是一個IP地址的集合,其實實現(xiàn)UDP廣播的程序和前面實現(xiàn)單播的程序差不多,只是前面綁定了一個IP地址當然也只能發(fā)送到一個IP地址了,也就是所謂的單播,多播和廣播就是發(fā)送的IP地址是一個組,當然也就實現(xiàn)了一對多的傳輸了)。UDP廣播程序的實現(xiàn)就放在下一個專題和大家分享的,因為我現(xiàn)在要去吃飯了,吃完飯再繼續(xù)和大家介紹,希望大家如果覺得有幫助的話,也可以推薦下,這給我繼續(xù)寫下去的動力,謝謝大家的支持