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$