您好,登錄后才能下訂單哦!
這篇文章給大家分享的是有關(guān)UDP服務(wù)器客戶端編程流程的示例分析的內(nèi)容。小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考,一起跟隨小編過(guò)來(lái)看看吧。
UDP提供的是無(wú)連接、不可靠的、數(shù)據(jù)報(bào)服務(wù)
UDP是盡最大能力進(jìn)行傳輸,但是并不能保證可靠性,TCP的可靠性是因?yàn)橐幌盗械臋C(jī)制保證可靠性,UDP丟包并不會(huì)重發(fā),兩種協(xié)議并沒有優(yōu)略之分,要區(qū)分不同的場(chǎng)景來(lái)區(qū)分,比如:進(jìn)行文件傳輸,不能有數(shù)據(jù)丟失,TCP協(xié)議就更合
適,而進(jìn)行實(shí)時(shí)視頻通話,UDP會(huì)根據(jù)恒定的速率進(jìn)行發(fā)送,這樣的情況容許部分?jǐn)?shù)據(jù)的丟失去追求更好的實(shí)時(shí)性,所以UDP更合適
流程:首先服務(wù)端與客戶端都需要套接字的創(chuàng)建socket()
(UDP并沒有嚴(yán)格意義上的服務(wù)端與客戶端),然后服務(wù)端需要確定ip與端口bind()
,等待接收接收數(shù)據(jù)recvfrom()
(會(huì)記錄對(duì)方的ip和端口),在這里我們并沒有跟某個(gè)客戶端進(jìn)行連接,只是接收發(fā)送過(guò)來(lái)的數(shù)據(jù),客戶端發(fā)送數(shù)據(jù)sendto()
(需要指定ip與端口),因?yàn)閁DP并不像TCP建立連接,通過(guò)文件描述符來(lái)識(shí)別客戶端,只能通過(guò)發(fā)送與接收時(shí)識(shí)別ip與端口的方式來(lái)區(qū)分不同的數(shù)據(jù),收發(fā)結(jié)束關(guān)閉套接字close()
#include<stdio.h> #include<stdlib.h> #include<unistd.h> #include<string.h> #include<assert.h> #include<sys/socket.h> #include<netinet/in.h> #include<arpa/inet.h> int main() { //創(chuàng)建套接字 //參數(shù): //AF_INET ipv4 //SOCK_DGRAM UDP使用的數(shù)據(jù)報(bào)服務(wù)類型 (SOCK_STREAM 流式套接字TCP使用的服務(wù)類型 //標(biāo)志位 一般給 0 int sockfd = socket(AF_INET,SOCK_DGRAM,0); assert(sockfd != -1); //創(chuàng)建套接字地址結(jié)構(gòu) struct sockaddr_in saddr,caddr; memset(&saddr,0,sizeof(saddr)); saddr.sin_family = AF_INET; saddr.sin_port = htons(6000); saddr.sin_addr.s_addr = inet_addr("127.0.0.1"); //命名套接字 int res = bind(sockfd,(struct sockaddr*)&saddr,sizeof(saddr)); assert(res != -1); while(1) { int len = sizeof(caddr);//這里專門存放在len中,是因?yàn)樵趓ecvfrom的時(shí)候接收一個(gè)指針 char buff[128] = {0}; //接受數(shù)據(jù) //參數(shù): //服務(wù)端套接字 //存放數(shù)據(jù) //存放大小 //標(biāo)志位 一般給0 //存放客戶端地址信息(ip與端口) //caddr的大小 recvfrom(sockfd,buff,127,0,(struct sockaddr*)&caddr,&len); printf("buff=%s\n",buff); //發(fā)送數(shù)據(jù) //參數(shù): //服務(wù)端套接字 //發(fā)送的數(shù)據(jù) //發(fā)送數(shù)據(jù)大小 //標(biāo)志位 一般給0 //發(fā)送目標(biāo)的地址信息 //地址信息的大小 sendto(sockfd,"ok",2,0,(struct sockaddr*)&caddr,sizeof(caddr)); } //關(guān)閉套接字 close(sockfd); }
#include<stdio.h> #include<stdlib.h> #include<unistd.h> #include<string.h> #include<assert.h> #include<sys/socket.h> #include<netinet/in.h> #include<arpa/inet.h> int main() { int sockfd = socket(AF_INET,SOCK_DGRAM,0); assert(sockfd != -1); //只需要指定服務(wù)器的ip與端口,客戶端自己的ip與端口由系統(tǒng)自動(dòng)指定 struct sockaddr_in saddr; memset(&saddr,0,sizeof(saddr)); saddr.sin_family = AF_INET; saddr.sin_port = htons(6000); saddr.sin_addr.s_addr = inet_addr("127.0.0.1"); while(1) { char buff[128] = {0}; printf("input:\n"); fgets(buff,128,stdin); if(strncmp(buff,"end",3) == 0) { break; } sendto(sockfd,buff,strlen(buff),0,(struct sockaddr*)&saddr,sizeof(saddr)); memset(buff,0,128); int len = sizeof(saddr); recvfrom(sockfd,buff,127,0,(struct sockaddr*)&saddr,&len); //占用了saddr存放獲取對(duì)方的ip與端口,實(shí)際上并沒有改變都是一樣的 printf("buff=%s\n",buff); } close(sockfd); }
執(zhí)行服務(wù)端與客戶端的代碼
當(dāng)我們打開多個(gè)窗口,啟動(dòng)多個(gè)客戶端向服務(wù)端發(fā)送數(shù)據(jù)
UDP服務(wù)端的接收只是根據(jù)是否有數(shù)據(jù)發(fā)送過(guò)來(lái),只要有發(fā)送過(guò)來(lái)的數(shù)據(jù)就進(jìn)行接收,并不進(jìn)行連接,即使將服務(wù)端關(guān)閉重新開啟,使用原本的客戶端依舊可以進(jìn)行發(fā)送數(shù)據(jù),因?yàn)樗麄儽旧碇g并不存在相互的連接
假如將,服務(wù)端進(jìn)行關(guān)閉不重啟,客戶端發(fā)送依舊發(fā)送出去sendto()
不阻塞,但是會(huì)阻塞在recvfrom()
這一步,簡(jiǎn)單的說(shuō)就是服務(wù)器只管接收數(shù)據(jù)或者向發(fā)送方發(fā)送數(shù)據(jù),無(wú)論是誰(shuí)都可以向其發(fā)送數(shù)據(jù)且無(wú)需任何連接
如果我們將服務(wù)端代碼進(jìn)行修改
我們?cè)偻ㄟ^(guò)客戶端向服務(wù)端發(fā)送數(shù)據(jù)
使用UDP協(xié)議的時(shí)候,當(dāng)數(shù)據(jù)傳輸過(guò)來(lái)我們將數(shù)據(jù)包拆開只讀取設(shè)定大小的數(shù)據(jù),其余就會(huì)丟掉,繼而丟失
每次發(fā)送都是一個(gè)獨(dú)立的數(shù)據(jù)包,因?yàn)槊看蔚陌l(fā)送可能目的地址都不相同,對(duì)于TCP是可以將多次的數(shù)據(jù)合并進(jìn)行發(fā)送的,因?yàn)樵谕幻枋龇贿B接內(nèi)發(fā)送對(duì)象只有連接的另一方
感謝各位的閱讀!關(guān)于“UDP服務(wù)器客戶端編程流程的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,讓大家可以學(xué)到更多知識(shí),如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。