二、實(shí)驗(yàn)平臺(tái)
ubuntu-8.04操作系統(tǒng)
三、實(shí)驗(yàn)內(nèi)容
編寫Linux下UDP服務(wù)器套接字程序,服務(wù)器接收客戶端發(fā)送的信息并顯示,同時(shí)顯示客戶的IP地址、端口號(hào),并向客戶端發(fā)送信息。如果服務(wù)器接收的客戶信息為“bye”,則退出循環(huán),并關(guān)閉套接字。
四、實(shí)驗(yàn)原理
UDP套接口是無(wú)連接的、不可靠的數(shù)據(jù)報(bào)協(xié)議;既然他不可靠為什么還要用呢?其一:當(dāng)應(yīng)用程序使用廣播或多播時(shí)只能使用UDP協(xié)議;其二:由于他是無(wú)連接的,所以速度快。因?yàn)閁DP套接口是無(wú)連接的,如果一方的數(shù)據(jù)報(bào)丟失,那另一方將無(wú)限等待,解決辦法是設(shè)置一個(gè)超時(shí)。
建立UDP套接口時(shí)socket函數(shù)的第二個(gè)參數(shù)應(yīng)該是SOCK_DGRAM,說(shuō)明是建立一個(gè)UDP套接口;由于UDP是無(wú)連接的,所以服務(wù)器端并不需要listen或accept函數(shù)。
使用UDP套接字編程可以實(shí)現(xiàn)基于TCP/IP協(xié)議的面向無(wú)連接的通信,它分為服務(wù)器端和客戶端兩部分,其主要實(shí)現(xiàn)過(guò)程如圖3.1所示。

圖3.1 UDP客戶/服務(wù)器的套接字函數(shù)
1、socket函數(shù):為了執(zhí)行網(wǎng)絡(luò)輸入輸出,一個(gè)進(jìn)程必須做的第一件事就是調(diào)用socket函數(shù)獲得一個(gè)文件描述符。
----------------------------------------------------------------- #include <sys/socket.h> int socket(int family,int type,int protocol); 返回:非負(fù)描述字---成功 -1---失敗 ----------------------------------------------------------------- |
第一個(gè)參數(shù)指明了協(xié)議簇,目前支持5種協(xié)議簇,最常用的有AF_INET(IPv4協(xié)議)和AF_INET6(IPv6協(xié)議);第二個(gè)參數(shù)指明套接口類型,有三種類型可選:SOCK_STREAM(字節(jié)流套接口)、SOCK_DGRAM(數(shù)據(jù)報(bào)套接口)和SOCK_RAW(原始套接口);如果套接口類型不是原始套接口,那么第三個(gè)參數(shù)就為0。
2、bind函數(shù):為套接口分配一個(gè)本地IP和協(xié)議端口,對(duì)于網(wǎng)際協(xié)議,協(xié)議地址是32位IPv4地址或128位IPv6地址與16位的TCP或UDP端口號(hào)的組合;如指定端口為0,調(diào)用bind時(shí)內(nèi)核將選擇一個(gè)臨時(shí)端口,如果指定一個(gè)通配IP地址,則要等到建立連接后內(nèi)核才選擇一個(gè)本地IP地址。
------------------------------------------------------------------- #include <sys/socket.h> int bind(int sockfd, const struct sockaddr * server, socklen_t addrlen); 返回:0---成功 -1---失敗 ------------------------------------------------------------------- |
第一個(gè)參數(shù)是socket函數(shù)返回的套接口描述字;第二和第第三個(gè)參數(shù)分別是一個(gè)指向特定于協(xié)議的地址結(jié)構(gòu)的指針和該地址結(jié)構(gòu)的長(zhǎng)度。
3、recvfrom函數(shù):UDP使用recvfrom()函數(shù)接收數(shù)據(jù),他類似于標(biāo)準(zhǔn)的read(),但是在recvfrom()函數(shù)中要指明目的地址。
------------------------------------------------------------------- #include <sys/types.h> #include <sys/socket.h> ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags, struct sockaddr * from, size_t *addrlen); 返回接收到數(shù)據(jù)的長(zhǎng)度---成功 -1---失敗 ------------------------------------------------------------------- |
前三個(gè)參數(shù)等同于函數(shù)read()的前三個(gè)參數(shù),flags參數(shù)是傳輸控制標(biāo)志。最后兩個(gè)參數(shù)類似于accept的最后兩個(gè)參數(shù)。
4、sendto函數(shù):UDP使用sendto()函數(shù)發(fā)送數(shù)據(jù),他類似于標(biāo)準(zhǔn)的write(),但是在sendto()函數(shù)中要指明目的地址。
------------------------------------------------------------------- #include <sys/types.h> #include <sys/socket.h> ssize_t sendto(int sockfd, const void *buf, size_t len, int flags, const struct sockaddr * to, int addrlen); 返回發(fā)送數(shù)據(jù)的長(zhǎng)度---成功 -1---失敗 ------------------------------------------------------------------- |
前三個(gè)參數(shù)等同于函數(shù)read()的前三個(gè)參數(shù),flags參數(shù)是傳輸控制標(biāo)志。參數(shù)to指明數(shù)據(jù)將發(fā)往的協(xié)議地址,他的大小由addrlen參數(shù)來(lái)指定。
五、實(shí)驗(yàn)步驟
1、登陸進(jìn)入ubuntu操作系統(tǒng),新建一個(gè)文件,命名為udpserver.c(為了方便起見,可以進(jìn)入“home”,再進(jìn)入用戶目錄,在用戶目錄下新建udpserver.c)。
2、在udpserver.c中編寫服務(wù)器端程序代碼并保存。
3、在“終端”(“Applications”→“附件”→“終端”)中執(zhí)行命令進(jìn)入udpserver.c所在目錄。(pwd命令可以顯示當(dāng)前所在目錄;ls命令可以顯示當(dāng)前目錄下的文件和文件夾信息;cd..命令可以進(jìn)入上一級(jí)目錄;cd 目錄名 命令可以進(jìn)入當(dāng)前所示的某個(gè)目錄。)
4、執(zhí)行命令gcc –o udpserver udpserver.c生成可執(zhí)行文件udpserver。
5、執(zhí)行命令./ udpserver,觀察結(jié)果。
6、認(rèn)真分析源代碼,體會(huì)如何編寫一個(gè)UDP服務(wù)器端程序。
六、參考程序(udpserver.c)
- #include<stdio.h>
- #include<string.h>
- #include<unistd.h>
- #include<sys/types.h>
- #include<sys/socket.h>
- #include<stdlib.h>
- #include<netinet/in.h>
- #include<arpa/inet.h>
-
- #define PORT 1234
- #define MAXDATASIZE 100
-
- main()
- {
- int sockfd;
- struct sockaddr_in server;
- struct sockaddr_in client;
- socklen_t addrlen;
- int num;
- char buf[MAXDATASIZE];
-
- if((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) == -1)
- {
- perror("Creatingsocket failed.");
- exit(1);
- }
-
- bzero(&server,sizeof(server));
- server.sin_family=AF_INET;
- server.sin_port=htons(PORT);
- server.sin_addr.s_addr= htonl (INADDR_ANY);
- if(bind(sockfd, (struct sockaddr *)&server, sizeof(server)) == -1)
- {
- perror("Bind()error.");
- exit(1);
- }
-
- addrlen=sizeof(client);
- while(1)
- {
- num =recvfrom(sockfd,buf,MAXDATASIZE,0,(struct sockaddr*)&client,&addrlen);
-
- if (num < 0)
- {
- perror("recvfrom() error\n");
- exit(1);
- }
-
- buf[num] = '\0';
- printf("You got a message (%s%) from client.\nIt's ip is%s, port is %d.\n",buf,inet_ntoa(client.sin_addr),htons(client.sin_port));
- sendto(sockfd,"Welcometo my server.\n",22,0,(struct sockaddr *)&client,addrlen);
- if(!strcmp(buf,"bye"))
- break;
- }
- close(sockfd);
- }
實(shí)驗(yàn)四 UDP客戶端程序設(shè)計(jì)
一、實(shí)驗(yàn)?zāi)康?/strong>
學(xué)習(xí)和掌握Linux下的UDP客戶端基本原理和基本編程方法。
二、實(shí)驗(yàn)平臺(tái)
ubuntu-8.04操作系統(tǒng)
三、實(shí)驗(yàn)內(nèi)容
編寫Linux下UDP客戶端套接字程序,結(jié)合實(shí)驗(yàn)三的UDP服務(wù)器端程序,實(shí)現(xiàn)以下功能:
1、客戶根據(jù)用戶提供的IP地址將用戶從終端輸入的信息發(fā)送給服務(wù)器,然后等待服務(wù)器的回應(yīng)。
2、服務(wù)器接收客戶端發(fā)送的信息并顯示,同時(shí)顯示客戶的IP地址、端口號(hào),并向客戶端發(fā)送信息。如果服務(wù)器接收的客戶信息為“bye”,則退出循環(huán),并關(guān)閉套接字。
3、客戶接收、顯示服務(wù)器發(fā)回的信息,并關(guān)閉套接字。
四、實(shí)驗(yàn)原理
見實(shí)驗(yàn)三的實(shí)驗(yàn)原理部分。
五、實(shí)驗(yàn)步驟
1、登陸進(jìn)入ubuntu操作系統(tǒng),新建一個(gè)文件,命名為udpclient.c(為了方便起見,可以進(jìn)入“home”,再進(jìn)入用戶目錄,在用戶目錄下新建udpclient.c)。
2、在udpclient.c中編寫客戶端程序代碼并保存。將實(shí)驗(yàn)三完成的udpserver.c拷貝到與udpclient.c同一目錄下。
3、在“終端”(“Applications”→“附件”→“終端”)中執(zhí)行命令進(jìn)入udpserver.c和udpclient.c所在目錄。
4、執(zhí)行命令gcc –o udpserver udpserver.c生成可執(zhí)行文件udpserver。
5、執(zhí)行命令./ udpserver。
6、再開一個(gè)“終端”,進(jìn)入udpserver.c和udpclient.c所在目錄,執(zhí)行命令
gcc–o udpclient udpclient.c生成可執(zhí)行文件udpclient。
7、執(zhí)行命令./ udpclient 127.0.0.1 hello。
8、觀察兩個(gè)“終端”出現(xiàn)的結(jié)果。
9、在客戶端終端下執(zhí)行命令./ udpclient 127.0.0.1 bye。
10、觀察兩個(gè)“終端”出現(xiàn)的結(jié)果。
11、認(rèn)真分析源代碼,體會(huì)如何編寫一個(gè)UDP客戶端程序。
六、參考程序(udpclient.c)
- #include <stdio.h>
- #include <stdlib.h>
- #include <unistd.h>
- #include <string.h>
- #include <sys/types.h>
- #include <sys/socket.h>
- #include <netinet/in.h>
- #include <netdb.h>
-
- #define PORT 1234
- #define MAXDATASIZE 100
-
- int main(int argc, char *argv[])
- {
- int sockfd, num;
- char buf[MAXDATASIZE];
-
- struct hostent *he;
- struct sockaddr_in server,peer;
-
- if (argc !=3)
- {
- printf("Usage: %s <IP Address><message>\n",argv[0]);
- exit(1);
- }
-
- if ((he=gethostbyname(argv[1]))==NULL)
- {
- printf("gethostbyname()error\n");
- exit(1);
- }
-
- if ((sockfd=socket(AF_INET, SOCK_DGRAM,0))==-1)
- {
- printf("socket() error\n");
- exit(1);
- }
-
- bzero(&server,sizeof(server));
- server.sin_family = AF_INET;
- server.sin_port = htons(PORT);
- server.sin_addr= *((struct in_addr *)he->h_addr);
- sendto(sockfd, argv[2],strlen(argv[2]),0,(struct sockaddr *)&server,sizeof(server));
- socklen_t addrlen;
- addrlen=sizeof(server);
- while (1)
- {
- if((num=recvfrom(sockfd,buf,MAXDATASIZE,0,(struct sockaddr *)&peer,&addrlen))== -1)
- {
- printf("recvfrom() error\n");
- exit(1);
- }
- if (addrlen != sizeof(server) ||memcmp((const void *)&server, (const void *)&peer,addrlen) != 0)
- {
- printf("Receive message from otherserver.\n");
- continue;
- }
-
- buf[num]='\0';
- printf("Server Message:%s\n",buf);
- break;
- }
-
- close(sockfd);
- }
UDP實(shí)驗(yàn)結(jié)果:
服務(wù)器端:

客戶端:
