溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

UDP服務(wù)器客戶端編程流程的示例分析

發(fā)布時(shí)間:2021-12-06 12:22:41 來(lái)源:億速云 閱讀:150 作者:小新 欄目:開發(fā)技術(shù)

這篇文章給大家分享的是有關(guān)UDP服務(wù)器客戶端編程流程的示例分析的內(nèi)容。小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考,一起跟隨小編過(guò)來(lái)看看吧。

UDP編程流程

UDP提供的是無(wú)連接、不可靠的、數(shù)據(jù)報(bào)服務(wù)

UDP服務(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ù)端與客戶端的代碼

UDP服務(wù)器客戶端編程流程的示例分析

當(dāng)我們打開多個(gè)窗口,啟動(dòng)多個(gè)客戶端向服務(wù)端發(fā)送數(shù)據(jù)

UDP服務(wù)器客戶端編程流程的示例分析

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并不存在相互的連接

UDP服務(wù)器客戶端編程流程的示例分析

假如將,服務(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)行修改

UDP服務(wù)器客戶端編程流程的示例分析

我們?cè)偻ㄟ^(guò)客戶端向服務(wù)端發(fā)送數(shù)據(jù)

UDP服務(wù)器客戶端編程流程的示例分析

使用UDP協(xié)議的時(shí)候,當(dāng)數(shù)據(jù)傳輸過(guò)來(lái)我們將數(shù)據(jù)包拆開只讀取設(shè)定大小的數(shù)據(jù),其余就會(huì)丟掉,繼而丟失

UDP服務(wù)器客戶端編程流程的示例分析

每次發(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ò),可以把它分享出去讓更多的人看到吧!

向AI問(wèn)一下細(xì)節(jié)

免責(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)容。

udp
AI