Linux操作系統(tǒng)
三、實(shí)驗(yàn)內(nèi)容
編寫Linux下TCP服務(wù)器套接字程序,程序運(yùn)行時服務(wù)器等待客戶的連接,一旦連接成功,則顯示客戶的IP地址、端口號,并向客戶端發(fā)送字符串。
四、實(shí)驗(yàn)原理
使用TCP套接字編程可以實(shí)現(xiàn)基于TCP/IP協(xié)議的面向連接的通信,它分為服務(wù)器端和客戶端兩部分,其主要實(shí)現(xiàn)過程如圖1.1所示。
圖1.1 TCP客戶/服務(wù)器的套接字函數(shù)
1、socket函數(shù):為了執(zhí)行網(wǎng)絡(luò)輸入輸出,一個進(jìn)程必須做的第一件事就是調(diào)用socket函數(shù)獲得一個文件描述符。
----------------------------------------------------------------- #include <sys/socket.h> int socket(int family,int type,int protocol); 返回:非負(fù)描述字---成功 -1---失敗 ----------------------------------------------------------------- |
第一個參數(shù)指明了協(xié)議簇,目前支持5種協(xié)議簇,最常用的有AF_INET(IPv4協(xié)議)和AF_INET6(IPv6協(xié)議);第二個參數(shù)指明套接口類型,有三種類型可選:SOCK_STREAM(字節(jié)流套接口)、SOCK_DGRAM(數(shù)據(jù)報(bào)套接口)和SOCK_RAW(原始套接口);如果套接口類型不是原始套接口,那么第三個參數(shù)就為0。
2、connect函數(shù):當(dāng)用socket建立了套接口后,可以調(diào)用connect為這個套接字指明遠(yuǎn)程端的地址;如果是字節(jié)流套接口,connect就使用三次握手建立一個連接;如果是數(shù)據(jù)報(bào)套接口,connect僅指明遠(yuǎn)程端地址,而不向它發(fā)送任何數(shù)據(jù)。
----------------------------------------------------------------- #include <sys/socket.h> int connect(int sockfd, const struct sockaddr * addr, socklen_t addrlen); 返回:0---成功 -1---失敗 ----------------------------------------------------------------- |
第一個參數(shù)是socket函數(shù)返回的套接口描述字;第二和第三個參數(shù)分別是一個指向套接口地址結(jié)構(gòu)的指針和該結(jié)構(gòu)的大小。
這些地址結(jié)構(gòu)的名字均已“sockaddr_”開頭,并以對應(yīng)每個協(xié)議族的唯一后綴結(jié)束。以IPv4套接口地址結(jié)構(gòu)為例,它以“sockaddr_in”命名,定義在頭文件<netinet/in.h>;以下是結(jié)構(gòu)體的內(nèi)容:
------------------------------------------------------------------ struct in_addr { in_addr_t s_addr; /* IPv4地址 */ }; struct sockaddr_in { uint8_t sin_len; /* 無符號的8位整數(shù) */ sa_family_t sin_family; /* 套接口地址結(jié)構(gòu)的地址簇,這里為AF_INET */ in_port_t sin_port; /* TCP或UDP端口 */ struct in_addr sin_addr; char sin_zero[8]; }; ------------------------------------------------------------------- |
3、bind函數(shù):為套接口分配一個本地IP和協(xié)議端口,對于網(wǎng)際協(xié)議,協(xié)議地址是32位IPv4地址或128位IPv6地址與16位的TCP或UDP端口號的組合;如指定端口為0,調(diào)用bind時內(nèi)核將選擇一個臨時端口,如果指定一個通配IP地址,則要等到建立連接后內(nèi)核才選擇一個本地IP地址。
------------------------------------------------------------------- #include <sys/socket.h> int bind(int sockfd, const struct sockaddr * server, socklen_t addrlen); 返回:0---成功 -1---失敗 ------------------------------------------------------------------- |
第一個參數(shù)是socket函數(shù)返回的套接口描述字;第二和第第三個參數(shù)分別是一個指向特定于協(xié)議的地址結(jié)構(gòu)的指針和該地址結(jié)構(gòu)的長度。
4、listen函數(shù):listen函數(shù)僅被TCP服務(wù)器調(diào)用,它的作用是將用sock創(chuàng)建的主動套接口轉(zhuǎn)換成被動套接口,并等待來自客戶端的連接請求。
------------------------------------------------------------------- #include <sys/socket.h> int listen(int sockfd,int backlog); 返回:0---成功 -1---失敗 ------------------------------------------------------------------- |
第一個參數(shù)是socket函數(shù)返回的套接口描述字;第二個參數(shù)規(guī)定了內(nèi)核為此套接口排隊(duì)的最大連接個數(shù)。由于listen函數(shù)第二個參數(shù)的原因,內(nèi)核要維護(hù)兩個隊(duì)列:以完成連接隊(duì)列和未完成連接隊(duì)列。未完成隊(duì)列中存放的是TCP連接的三路握手為完成的連接,accept函數(shù)是從以連接隊(duì)列中取連接返回給進(jìn)程;當(dāng)以連接隊(duì)列為空時,進(jìn)程將進(jìn)入睡眠狀態(tài)。
5、accept函數(shù):accept函數(shù)由TCP服務(wù)器調(diào)用,從已完成連接隊(duì)列頭返回一個已完成連接,如果完成連接隊(duì)列為空,則進(jìn)程進(jìn)入睡眠狀態(tài)。
------------------------------------------------------------------- #include <sys/socket.h> int accept(int listenfd, struct sockaddr *client, socklen_t * addrlen); 回:非負(fù)描述字---成功 -1---失敗 ------------------------------------------------------------------- |
第一個參數(shù)是socket函數(shù)返回的套接口描述字;第二個和第三個參數(shù)分別是一個指向連接方的套接口地址結(jié)構(gòu)和該地址結(jié)構(gòu)的長度;該函數(shù)返回的是一個全新的套接口描述字;如果對客戶段的信息不感興趣,可以將第二和第三個參數(shù)置為空。
6、write和read函數(shù):當(dāng)服務(wù)器和客戶端的連接建立起來后,就可以進(jìn)行數(shù)據(jù)傳輸了,服務(wù)器和客戶端用各自的套接字描述符進(jìn)行讀/寫操作。因?yàn)樘捉幼置枋龇彩且环N文件描述符,所以可以用文件讀/寫函數(shù)write()和read()進(jìn)行接收和發(fā)送操作。
(1)write()函數(shù)用于數(shù)據(jù)的發(fā)送。
-------------------------------------------------------------------
#include <unistd.h>
int write(int sockfd, char *buf, int len);
回:非負(fù)---成功 -1---失敗
-------------------------------------------------------------------
參數(shù)sockfd是套接字描述符,對于服務(wù)器是accept()函數(shù)返回的已連接套接字描述符,對于客戶端是調(diào)用socket()函數(shù)返回的套接字描述符;參數(shù)buf是指向一個用于發(fā)送信息的數(shù)據(jù)緩沖區(qū);len指明傳送數(shù)據(jù)緩沖區(qū)的大小。
(2)read()函數(shù)用于數(shù)據(jù)的接收。
-------------------------------------------------------------------
#include <unistd.h>
int read(int sockfd, char *buf, intlen);
回:非負(fù)---成功 -1---失敗
-------------------------------------------------------------------
參數(shù)sockfd是套接字描述符,對于服務(wù)器是accept()函數(shù)返回的已連接套接字描述符,對于客戶端是調(diào)用socket()函數(shù)返回的套接字描述符;參數(shù)buf是指向一個用于接收信息的數(shù)據(jù)緩沖區(qū);len指明接收數(shù)據(jù)緩沖區(qū)的大小。
7、send和recv函數(shù):TCP套接字提供了send()和recv()函數(shù),用來發(fā)送和接收操作。這兩個函數(shù)與write()和read()函數(shù)很相似,只是多了一個附加的參數(shù)。
(1)send()函數(shù)用于數(shù)據(jù)的發(fā)送。
-------------------------------------------------------------------
#include <sys/types.h>
#include < sys/socket.h >
ssize_t send(int sockfd, const void *buf, size_t len, int flags);
回:返回寫出的字節(jié)數(shù)---成功 -1---失敗
-------------------------------------------------------------------
前3個參數(shù)與write()相同,參數(shù)flags是傳輸控制標(biāo)志。
(2)recv()函數(shù)用于數(shù)據(jù)的發(fā)送。
-------------------------------------------------------------------
#include <sys/types.h>
#include < sys/socket.h >
ssize_t recv(int sockfd, void *buf, size_t len, int flags);
回:返回讀入的字節(jié)數(shù)---成功 -1---失敗
-------------------------------------------------------------------
前3個參數(shù)與read()相同,參數(shù)flags是傳輸控制標(biāo)志。
五、實(shí)驗(yàn)步驟
1、登陸進(jìn)入ubuntu操作系統(tǒng),新建一個文件,命名為tcpserver.c(為了方便起見,可以進(jìn)入“home”,再進(jìn)入用戶目錄,在用戶目錄下新建tcpserver.c)。
2、在tcpserver.c中編寫服務(wù)器端程序代碼并保存。
3、在“終端”(“Applications”→“附件”→“終端”)中執(zhí)行命令進(jìn)入tcpserver.c所在目錄。(pwd命令可以顯示當(dāng)前所在目錄;ls命令可以顯示當(dāng)前目錄下的文件和文件夾信息;cd..命令可以進(jìn)入上一級目錄;cd 目錄名 命令可以進(jìn)入當(dāng)前所示的某個目錄。)
4、執(zhí)行命令gcc –o tcpserver tcpserver.c生成可執(zhí)行文件tcpserver。
5、執(zhí)行命令./ tcpserver,觀察結(jié)果。
6、認(rèn)真分析源代碼,體會如何編寫一個TCP服務(wù)器端程序。
六、參考程序(tcpserver.c)
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <unistd.h>
- #include <sys/types.h>
- #include <sys/socket.h>
- #include <netinet/in.h>
- #include <arpa/inet.h>
-
- #define PORT 1234
- #define BACKLOG 1
-
- int main()
- {
- int listenfd, connectfd;
- struct sockaddr_in server;
- struct sockaddr_in client;
- socklen_t addrlen;
- if((listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
- {
- perror("Creating socket failed.");
- exit(1);
- }
- int opt =SO_REUSEADDR;
- setsockopt(listenfd,SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
- bzero(&server,sizeof(server));
- server.sin_family=AF_INET;
- server.sin_port=htons(PORT);
- server.sin_addr.s_addr= htonl (INADDR_ANY);
- if(bind(listenfd, (struct sockaddr *)&server, sizeof(server)) == -1) {
- perror("Binderror.");
- exit(1);
- }
- if(listen(listenfd,BACKLOG)== -1){ /* calls listen() */
- perror("listen()error\n");
- exit(1);
- }
- addrlen =sizeof(client);
- if((connectfd = accept(listenfd,(struct sockaddr*)&client,&addrlen))==-1) {
- perror("accept()error\n");
- exit(1);
- }
- printf("Yougot a connection from cient's ip is %s, prot is %d\n",inet_ntoa(client.sin_addr),htons(client.sin_port));
- send(connectfd,"Welcometo my server.\n",22,0);
- close(connectfd);
- close(listenfd);
- return 0;
- }
實(shí)驗(yàn)二 TCP客戶端程序設(shè)計(jì)
一、實(shí)驗(yàn)?zāi)康?/strong>
學(xué)習(xí)和掌握Linux下的TCP客戶端基本原理和基本編程方法。
二、實(shí)驗(yàn)平臺
Linux操作系統(tǒng)
三、實(shí)驗(yàn)內(nèi)容
編寫Linux下TCP客戶端套接字程序,結(jié)合實(shí)驗(yàn)一的服務(wù)器端程序,實(shí)現(xiàn)以下功能:
1、客戶根據(jù)用戶提供的IP地址連接到相應(yīng)的服務(wù)器;
2、服務(wù)器等待客戶的連接,一旦連接成功,則顯示客戶的IP地址、端口號,并向客戶端發(fā)送字符串;
3、客戶接收服務(wù)器發(fā)送的信息并顯示。
四、實(shí)驗(yàn)原理
見實(shí)驗(yàn)一的實(shí)驗(yàn)原理部分。
五、實(shí)驗(yàn)步驟
1、登陸進(jìn)入ubuntu操作系統(tǒng),新建一個文件,命名為tcpclient.c(為了方便起見,可以進(jìn)入“home”,再進(jìn)入用戶目錄,在用戶目錄下新建tcpclient.c)。
2、在tcpclient.c中編寫客戶端程序代碼并保存。將實(shí)驗(yàn)一完成的tcpserver.c拷貝到與tcpclient.c同一目錄下。
3、在“終端”(“Applications”→“附件”→“終端”)中執(zhí)行命令進(jìn)入tcpserver.c和tcpclient.c所在目錄。
4、執(zhí)行命令gcc –o tcpserver tcpserver.c生成可執(zhí)行文件tcpserver。
5、執(zhí)行命令./ tcpserver。
6、再開一個“終端”,進(jìn)入tcpserver.c和tcpclient.c所在目錄,執(zhí)行命令
gcc–o tcpclient tcpclient.c生成可執(zhí)行文件tcpclient。
7、執(zhí)行命令./ tcpclient 127.0.0.1。
8、觀察兩個“終端”出現(xiàn)的結(jié)果。
9、認(rèn)真分析源代碼,體會如何編寫一個TCP客戶端程序。
六、參考程序(tcpclient.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;
- if (argc!=2) {
- printf("Usage:%s <IP Address>\n",argv[0]);
- exit(1);
- }
- if((he=gethostbyname(argv[1]))==NULL){
- printf("gethostbyname()error\n");
- exit(1);
- }
- if((sockfd=socket(AF_INET, SOCK_STREAM, 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);
- if(connect(sockfd,(struct sockaddr *)&server,sizeof(server))==-1){
- printf("connect()error\n");
- exit(1);
- }
- if((num=recv(sockfd,buf,MAXDATASIZE,0)) == -1){
- printf("recv() error\n");
- exit(1);
- }
- buf[num-1]='\0';
- printf("Server Message: %s\n",buf);
- close(sockfd);
- return 0;
- }
-
實(shí)驗(yàn)結(jié)果:

