6.2 UDP 原理2

1. UDP客户端套接字的地址分配

UDP程序中, 调用sendto函数传输数据前完成套接字地址分配工作。

可以调用bind函数,bind函数不区分是TCP还是UDP。

如果调用sendto函数时发现尚未分配地址,则首次调用sendto时自动分配IP和端口,并且分配的地址移植保留到程序结束。


2. UDP存在数据边界

与TCP不一样,UDP协议中,调用IO函数的次数很重要,输入函数的调用次数和输出函数的调用次数完全一致,才能保证接收完全。

其实,UDP套接字传输的数据包又称为数据报,它本身可以成为一个完整数据。


3.举例bound_host:host2程序调用3次sendto传输数据,host1调用3次recvfrom函数接收数据。

bound_host1.c

#include <stdio.h>

#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>

#define BUF_SIZE 30

void error_handling(char *message);

int main(int argc, char *argv[]){
	int sock;
	char message[BUF_SIZE];
	struct sockaddr_in my_adr, your_adr;
	socklen_t adr_sz;
	int str_len,i;

	if(argc != 2){
		printf("Usage : %s <port> \n",argv[0]);
		exit(1);
	}

	sock = socket(PF_INET,SOCK_DGRAM,0);
	if(sock == -1){
		error_handling("socket error");
	}	

	memset(&my_adr,0,sizeof(my_adr));
	my_adr.sin_family = AF_INET;
	my_adr.sin_addr.s_addr = htonl(INADDR_ANY);
	my_adr.sin_port = htons(atoi(argv[1]));
	
	if(bind(sock,(struct sockaddr *)&my_adr,sizeof(my_adr)) == -1){
		error_handling("bind error");
	}
	
	for(i=0;i<3;i++){
		sleep(5);
		adr_sz = sizeof(your_adr);
		str_len = recvfrom(sock, message, BUF_SIZE,0,(struct sockaddr*)&your_adr,&adr_sz);
		printf("Message %d: %s \n",i+1,message);
	}
	close(sock);
	return 0;
}

void error_handling(char *message){
	fputs(message,stderr);
	fputs("\n",stderr);
	exit(1);
}

运行结果:

alex@alex-virtual-machine:/extra/tcpip$ ./host1 9190
Message 1: Hi!
Message 2: I'm another UDP host!
Message 3: Nice to meet you
alex@alex-virtual-machine:/extra/tcpip$

bound_host2.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>

#define BUF_SIZE 30

void error_handling(char *message);

int main(int argc, char *argv[]){
	int sock;
	char msg1[] = "Hi!";
	char msg2[] = "I'm another UDP host!";
	char msg3[] = "Nice to meet you";
	struct sockaddr_in your_adr;
	socklen_t your_adr_sz;

	if(argc != 3){
		printf("Usage : %s <IP><port> \n",argv[0]);
		exit(1);
	}

	sock = socket(PF_INET,SOCK_DGRAM,0);
	if(sock == -1){
		error_handling("socket error");
	}	

	memset(&your_adr,0,sizeof(your_adr));
	your_adr.sin_family = AF_INET;
	your_adr.sin_addr.s_addr = inet_addr(argv[1]);
	your_adr.sin_port = htons(atoi(argv[2]));
	
	sendto(sock,msg1,sizeof(msg1),0,(struct sockaddr*)&your_adr,sizeof(your_adr));
	sendto(sock,msg2,sizeof(msg2),0,(struct sockaddr*)&your_adr,sizeof(your_adr));
	sendto(sock,msg3,sizeof(msg3),0,(struct sockaddr*)&your_adr,sizeof(your_adr));
	close(sock);
	return 0;
}

void error_handling(char *message){
	fputs(message,stderr);
	fputs("\n",stderr);
	exit(1);
}

运行结果:

alex@alex-virtual-machine:/extra/tcpip$ ./host2 127.0.0.1 9190
alex@alex-virtual-machine:/extra/tcpip$


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值