關(guān)于UDP服務(wù)器客戶端編程流程介紹
UDP編程流程
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()
UDP服務(wù)端代碼實(shí)現(xiàn)
#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); }
UDP客戶端代碼實(shí)現(xiàn)
#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); }
UDP服務(wù)端客戶端代碼詳解
執(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)于關(guān)于UDP服務(wù)器客戶端編程流程介紹的文章就介紹到這了,更多相關(guān)UDP客戶端編程程流內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語(yǔ)言實(shí)現(xiàn)自動(dòng)發(fā)牌程序
這篇文章主要介紹了利用C語(yǔ)言實(shí)現(xiàn)自動(dòng)發(fā)牌程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-12-12C語(yǔ)言中字符串和數(shù)字的相互轉(zhuǎn)換實(shí)現(xiàn)代碼
以下是對(duì)C語(yǔ)言中字符串和數(shù)字的相互轉(zhuǎn)換實(shí)現(xiàn)代碼進(jìn)行了分析介紹,需要的朋友可以參考下2013-07-07C++中volatile關(guān)鍵字的使用詳解以及常見的誤解
volatile 關(guān)鍵字是一種類型修飾符,用它聲明的類型變量表示可以被某些編譯器未知的因素更改,比如:操作系統(tǒng),硬件或者其他線程等2020-01-01C語(yǔ)言使用DP動(dòng)態(tài)規(guī)劃思想解最大K乘積與乘積最大問(wèn)題
Dynamic Programming動(dòng)態(tài)規(guī)劃方法采用最優(yōu)原則來(lái)建立用于計(jì)算最優(yōu)解的遞歸式,并且考察每個(gè)最優(yōu)決策序列中是否包含一個(gè)最優(yōu)子序列,這里我們就來(lái)展示C語(yǔ)言使用DP動(dòng)態(tài)規(guī)劃思想解最大K乘積與乘積最大問(wèn)題2016-06-06C語(yǔ)言實(shí)現(xiàn)影院售票管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)影院售票管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08Linux中rm命令使用以及C/C++代碼實(shí)現(xiàn)
m 是remove 的縮寫,Linux中 rm 命令的功能為刪除一個(gè)目錄中的一個(gè)或多個(gè)文件或目錄,它也可以將某個(gè)目錄及其下的所有文件及子目錄均刪除,這篇文章主要給大家介紹了關(guān)于Linux中rm命令使用以及C/C++代碼實(shí)現(xiàn)的相關(guān)資料,需要的朋友可以參考下2022-04-04C++實(shí)現(xiàn)高性能轉(zhuǎn)換大小寫算法示例
大小寫轉(zhuǎn)換是我們作為一名程序員經(jīng)常會(huì)遇到,也必須要會(huì)的一個(gè)功能,下面這篇文章主要給大家介紹了關(guān)于C++實(shí)現(xiàn)高性能轉(zhuǎn)換大小寫算法的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面來(lái)一起看看吧。2018-01-01