1、应用进程的数据在沿协议栈向下传递时,以某种形式复制到内核的缓存中,当数据链路层把数据传出去后这个副本就被丢弃。当应用进程调用write时,内核中的UDP从应用进程的缓存中复制所有数据到内存缓存中。UDP简单地加上它的8字节首部以构成PDU并把PDU传递给IP。IP再往下传递。如果UDP应用进程发送一个大的数据报,它比TCP应用更有可能在底层被分片,因为TCP会把应用进程数据划分成MSS大小的块,但UDP却没有这种机制。从写UDP插口的write调用成功地返回表示数据报或所有报文段已被加入到链路层的输出队列。如果输出队列中没有足够的空间来存放数据报或它的某些报文段,UDP将返回错误ENOBUFS
2、给应用进程。需要说明的是,有些UDP的实现并不返回这种错误,在这种实现中,甚至数据报还没有发送就被丢弃的情况应用进程也不知道。因此,使用UDP通信时,应用进程发送的数据不能太长。4-2 基本UDP插口编程4-2-1 UDP编程模式图4-1给出了典型的UDP客户-服务器程序所使用的插口函数。对UDP应用而言,如果从通信的角度来看,客户服务器的概念是比较模糊的:任何时候,发送数据的一方可以认为是客户,接收数据的一方是服务器,而不像TCP应用,发起连接的一方为客户,而接收连接的一方称为服务器,自始至终都保持这种关系。当然,可以从应用功能的角度,将提供服务的一方称为服务器,而将申请服务的一方称为客户。
3、图4-1 UDP客户-服务器程序的一般通信过程基于UDP的客户与服务器在通信之前不需要建立连接。在数据传送完成之后,需要关闭插口,释放网络资源。在大多数情况下,UDP客户和服务器方调用的插口函数是一样的,按时间顺序如下:(1)打开一个插口(socket)。(2)将插口绑定到指定地址和端口上(bind)。(3)进行数据交换(sendto, recvfrom)。(4)数据交换完成,关闭连接(close)。在某些情况下,UDP通信两方中一方可以不绑定指定地址和端口,而是由内核自动分配,但至少有一方是要显示绑定地址和端口的。而且,不绑定地址的一方必须首先向绑定地址的一方(服务器)发送数据,绑定地址和端
4、口的一方从接收到的数据报中获取发送方的地址和端口,用于向没有显式绑定地址和端口的一方发送数据。但大多数情况下,UDP应用都显式地将插口绑定到指定地址和端口上。通常情况下,UDP客户-服务器间数据交换使用的系统调用为sendto和recvfrom。这两个调用均可指定或返回对方的地址。如果recvfrom的参数源地址参数from是空指针,则相应的源地址长度参数addrlen也必须是空指针,这表示我们并不关心发送数据方的协议地址。这两个函数也可用于TCP,但一般不需要这么使用。我们在第2章中已详细讨论了这两个函数。另外,尽管UDP是面向无连接的通信,但是UDP应用也可以调用connect()函数。但
5、是,此处的connect() 的功能并不是向对方发起连接请求(启动三次握手过程),内核只是记录connect()调用中指定的对方的IP地址和端口号,并立即返回给调用进程。在这种情况下,UDP应用使用的数据交换函数就不再是sendto和recvfrom,而是使用TCP应用常用的write(或send)和read(或recv)。我们将在4-2-4节中详细讨论这种情况。图4-1只显示了一个简单的UDP客户-服务器的处理过程。实际的网络应用程序可能要比此过程复杂得多。如,需要改变插口行为时需要调用setsockopt(),需要获取插口信息时需要调用getsockopt()等等,最多时可能要用到第2章中
6、介绍的所有与插口编程有关的函数。下面我们就从这个最简单的客户-服务器模式开始介绍如何实现利用UDP的通信。4-2-2 实 例本节用一个具体的例子来介绍UDP插口编程。我们重写第3章中的测试TCP性能的客户-服务器程序,改为测试UDP性能的客户-服务器程序。我们仍使用第3章中的头文件common.h,但要增加一个宏定义: #define CLIENT_PORT 3000 /* 客户进程端口号 */下面给出客户程序。程序4-1 阻塞方式下的UDP客户程序udp_client.c。 1 #include common.h 2 3 char buffMAXBUFSIZE; /* 发送和接收缓存 */
7、4 int send_size5=10,100,1000,10000,100000; /* 发送数据大小 */ 5 6 main(argc, argv) 7 int argc; 8 char *argv; 9 10 int sockfd=0; /* 插口描述符 */ 11 struct sockaddr_in srcaddr,destaddr; /* 源地址,目的地址 */ 12 int alen; /* 地址长度 */ 13 int slen, rlen, len; /* 发送和接收长度变量 */ 14 int rdt; /* 记录RoundTrip时间 */ 15 int i,j; 16
8、struct timeval tv1,tv2; 17 18 if (argc 2) 19 20 printf(使用方法:udp_client n); 21 exit(0); 22 23 24 /* 产生UDP 插口 */ 25 if (sockfd = socket(AF_INET, SOCK_DGRAM, 0) 0) 26 27 printf(产生插口失败,退出n 28 exit(1); 29 30 alen = sizeof(struct sockaddr_in); /* 地址结构长度 */ 31 /* 由系统自动绑定一个IP地址 */ 32 bzero(char *) &srcaddr,
9、 alen); 33 srcaddr.sin_family = AF_INET; 34 srcaddr.sin_addr.s_addr = htonl(INADDR_ANY); 35 srcaddr.sin_port = htons(CLIENT_PORT); 36 /* 绑定插口到本地的指定地址 */ 37 if ( bind(sockfd, (struct sockaddr *)&srcaddr,alen) 38 39 printf(绑定插口失败, 退出n 40 exit(2); 41 42 43 bzero(char *) &destaddr, alen); 44 destaddr.si
10、n_family = AF_INET; 45 destaddr.sin_port = htons(SERVER_PORT); 46 destaddr.sin_addr.s_addr = inet_addr(argv1); 47 48 for(i=0;i5;i+) /* 测试不同大小的报文的RoundTrip时间 */ 49 50 slen = send_sizei; /* 本次测试数据大小 */ 51 gettimeofday(&tv1,(struct timezone *)0); 52 53 for(j=0;j50;j+) /* 每一种大小测试 50 次 */ 54 55 if (rlen
11、= sendto(sockfd, buff, slen, 0, 56 (struct sockaddr *)&destaddr, sizeof(destaddr) 57 58 printf(发送失败,程序退出, j=%d errno=%dn,j,errno); 59 close(sockfd); 60 exit(1); 61 62 bzero(char *)&srcaddr,sizeof(srcaddr); 63 if (rlen = recvfrom(sockfd, buff, slen, 0, 64 (struct sockaddr *)&srcaddr, &alen) 0) 65 66
12、printf(接收失败,程序退出, errno=%dn,errno); 67 close(sockfd); 68 exit(2); 69 70 71 gettimeofday(&tv2,(struct timezone *)0); 72 rdt = (tv2.tv_sec - tv1.tv_sec) * 1000000 + 73 tv2.tv_usec - tv1.tv_usec)/50; 74 printf(报文大小 = %d, RoundTripTime = %d 微秒n, slen, rdt); 75 76 close(sockfd); 77 第25行 打开一个UDP插口, 插口类型为S
13、OCK_DGRAM。第3241行 将插口绑定到端口CLIENT_PORT上。如上一节所述,客户进程也可以不显式绑定到指定地址和端口,但要求通信由客户进程先发起。第4346行 填写目的地址结构,用于sendto函数。第46行中使用了地址转换函数inet_addr(),而没有使用inet_aton()。因为在Solaris 2.5中,不支持函数inet_aton()。第5561行 发送指定长度的数据到目的地址。如果发送失败,则进程退出。如果是非阻塞方式,则需要改写错误处理部分。如果在sendto调用中不指定目的地址(没有对UDP插口调用connect()的情况下),则返回的错误代码取决于实现:在4.4BSD中,将返回EDESTADDRREQ,而Posix.1g中则返回ENOTCONN。第6269行 接收数据报。第62行初始化数据源地址结构变量,如果要从指定源地址接收数据报,则需要进一步设置该变量中的相关成员(地址和端口号)。此处的设置表明可以接收从任何主机上的任意UDP端口上发送来的数据报。下面给出UDP服务器程序。程序
copyright@ 2008-2022 冰豆网网站版权所有
经营许可证编号:鄂ICP备2022015515号-1