http://blog.csdn.net/qq_31564375/article/details/51581038
項目介紹
本項目是實現一個簡單的聊天室,聊天室分為服務端和客戶端。本項目將很多復雜的功能都去掉了,線程池、多線程編程、超時重傳、確認收包等等都不會涉及。總共300多行代碼,讓大家真正了解C/S模型,以及epoll的使用。為了方便查看,代碼已經改的很小白,絕對比nginx源碼好理解(當然大家有興趣的話,還是要拜讀下nginx源碼,絕對大有收獲)。希望本項目能為大家以后的工作或者學習提供一點幫助! 介紹如下:
1. 服務端
a. 支持多個用戶接入,實現聊天室的基本功能
b. 使用epoll機制實現并發,增加效率
2. 客戶端
a. 支持用戶輸入聊天消息
b. 顯示其他用戶輸入的信息
c. 使用fork創建兩個進程
子進程有兩個功能:
- 等待用戶輸入聊天信息
- 將聊天信息寫到管道(pipe),并發送給父進程
父進程有兩個功能
- 使用epoll機制接受服務端發來的信息,并顯示給用戶,使用戶看到其他用戶的聊天信息
- 將子進程發給的聊天信息從管道(pipe)中讀取, 并發送給服務端
3. 代碼說明
一共有3個文件, 即: server.cpp, client.cpp, utility.h
a. server.cpp是服務端程序
b. client.cpp是客戶端程序
c. utility.h是一個頭文件,包含服務端程序和客戶端程序都會用到的一些頭文件、變量聲明、函數、宏等。
1.1 TCP服務端通信的常規步驟
(1)使用socket()創建TCP套接字(socket)
(2)將創建的套接字綁定到一個本地地址和端口上(Bind)
(3)將套接字設為監聽模式,準備接收客戶端請求(listen)
(4)等待客戶請求到來: 當請求到來后,接受連接請求,返回一個對應于此次連接的新的套接字(accept)
(5)用accept返回的套接字和客戶端進行通信(使用write()/send()或send()/recv() )
(6)返回,等待另一個客戶請求
(7)關閉套接字
- //server.cpp代碼(通信模塊):??
- ????//服務端地址?ip地址?+?端口號??
- ????struct?sockaddr_in?serverAddr;??
- ????serverAddr.sin_family?=?PF_INET;??
- ????serverAddr.sin_port?=?htons(SERVER_PORT);??
- ????serverAddr.sin_addr.s_addr?=?inet_addr(SERVER_HOST);??
- ??
- ????//服務端創建監聽socket??
- ????int?listener?=?socket(PF_INET,?SOCK_STREAM,?0);??
- ????if(listener?<?0)?{?perror("listener");?exit(-1);}??
- ????printf("listen?socket?created?\n");??
- ??
- ????//將服務端地址與監聽socket綁定??
- ????if(?bind(listener,?(struct?sockaddr?*)&serverAddr,?sizeof(serverAddr))?<?0)?{??
- ????????perror("bind?error");??
- ????????exit(-1);??
- ????}??
- ????//開始監聽??
- ????int?ret?=?listen(listener,?5);??
- ????if(ret?<?0)?{?perror("listen?error");?exit(-1);}??
- ????printf("Start?to?listen:?%s\n",?SERVER_HOST);??
2. 基本技術介紹
2.1 阻塞與非阻塞socket
通常的,對一個文件描述符指定的文件或設備, 有兩種工作方式: 阻塞與非阻塞方式。
(1). 阻塞方式是指: 當試圖對該文件描述符進行讀寫時,如果當時沒有數據可讀,或者暫時不可寫,程序就進入等待狀態,直到有東西可讀或者可寫為止。
(2). 非阻塞方式是指: 如果沒有數據可讀,或者不可寫,讀寫函數馬上返回,而不會等待。
(3). 舉個例子來說,比如說小明去找一個女神聊天,女神卻不在。 如果小明舍不得走,只能在女神大門口死等著,當然小明可以休息。當女 神來了,她會把你喚醒(囧,因為擋著她門了),這就是阻塞方式。如果小明發現女神不在,立即離開,以后每隔十分鐘回來看一下(采用輪詢方式),不在的話仍然立即離開,這就是非阻塞方式。
(4). 阻塞方式和非阻塞方式唯一的區別: 是否立即返回。本項目采用更高效的做法,所以應該將socket設置為非阻塞方式。這樣能充分利用服務器資源,效率得到了很大提高。
- //utility.h代碼(設置非阻塞函數模塊):??
- //將文件描述符設置為非阻塞方式(利用fcntl函數)??
- int?setnonblocking(int?sockfd)??
- {??
- ????fcntl(sockfd,?F_SETFL,?fcntl(sockfd,?F_GETFD,?0)|?O_NONBLOCK);??
- ????return?0;??
- }??
2.2 epoll
前面介紹了阻塞和非阻塞方式,現在該介紹下epoll機制了。epoll真的是一個特別重要的概念,實驗的師兄們去bat任何一家面試后臺開發,或者系統開發等相關職位都會問epoll機制。當服務端的在線人數越來越多,會導致系統資源吃緊,I/O效率越來越慢,這時候就應該考慮epoll了。epoll是Linux內核為處理大批句柄而作改進的poll,是Linux特有的I/O函數。其特點如下:
a.
epoll是Linux下多路復用IO接口select/poll的增強版本。其實現和使用方式與select/poll有很多不同,epoll通過一組函數來完成有關任務,而不是一個函數。
b.
epoll之所以高效,是因為epoll將用戶關心的文件描述符放到內核里的一個事件表中,而不是像select/poll每次調用都需要重復傳入文件描述符集或事件集。比如當一個事件發生(比如說讀事件),epoll無須遍歷整個被偵聽的描述符集,只要遍歷那些被內核IO事件異步喚醒而加入就緒隊列的描述符集合就行了。
c.
epoll有兩種工作方式,LT(level triggered):水平觸發和ET(edge-triggered):邊沿觸發。LT是select/poll使用的觸發方式,比較低效;而ET是epoll的高速工作方式(本項目使用epoll的ET方式)。
d.
通俗理解就是,比如說有一堆女孩,有的很漂亮,有的很鳳姐。現在你想找漂亮的女孩聊天,LT就是你需要把這一堆女孩全都看一遍,才可以找到其中的漂亮的(就緒事件);而ET是你的小弟(內核)將N個漂亮的女孩編號告訴你,你直接去看就好,所以epoll很高效。另外,還記得小明找女神聊天的例子嗎?采用非阻塞方式,小明還需要每隔十分鐘回來看一下(select);如果小明有小弟(內核)幫他守在大門口,女神回來了,小弟會主動打電話,告訴小明女神回來了,快來處理吧!這就是epoll。
epoll 共3個函數,1、int epoll_create(int size)創建一個epoll句柄,參數size用來告訴內核監聽的數目,size為epoll所支持的最大句柄數
2、int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event)函數功能: epoll事件注冊函數參數epfd為epoll的句柄,即epoll_create返回值參數op表示動作,用3個宏來表示: EPOLL_CTL_ADD(注冊新的fd到epfd), EPOLL_CTL_MOD(修改已經注冊的fd的監聽事件),EPOLL_CTL_DEL(從epfd刪除一個fd);其中參數fd為需要監聽的標示符;參數event告訴內核需要監聽的事件,event的結構如下:struct epoll_event {__uint32_t events; //Epoll eventsepoll_data_t data; //User data variable};其中介紹events是宏的集合,本項目主要使用EPOLLIN(表示對應的文件描述符可以讀,即讀事件發生),其他宏類型,可以google之!
3、 int epoll_wait(int epfd, struct epoll_event * events, int maxevents, int timeout)
等待事件的產生,函數返回需要處理的事件數目(該數目是就緒事件的數目,就是前面所說漂亮女孩的個數N)
因此服務端使用epoll的時候,步驟如下:
- 調用epoll_create函數在Linux內核中創建一個事件表;
- 然后將文件描述符(監聽套接字listener)添加到所創建的事件表中;
- 在主循環中,調用epoll_wait等待返回就緒的文件描述符集合;
- 分別處理就緒的事件集合,本項目中一共有兩類事件:新用戶連接事件和用戶發來消息事件(epoll還有很多其他事件,本項目為簡潔明了,不介紹)。
下面介紹下如何將一個socket添加到內核事件表中,如下:
- //utility.h(添加socket模塊):??
- //將文件描述符fd添加到epollfd標示的內核事件表中,?并注冊EPOLLIN和EPOOLET事件,EPOLLIN是數據可讀事件;EPOOLET表明是ET工作方式。最后將文件描述符設置非阻塞方式??
- /**?
- ??*?@param?epollfd:?epoll句柄?
- ??*?@param?fd:?文件描述符?
- ??*?@param?enable_et?:?enable_et?=?true,??
- ?????采用epoll的ET工?作方式;否則采用LT工作方式?
- **/??
- void?addfd(?int?epollfd,?int?fd,?bool?enable_et?)??
- {??
- ????struct?epoll_event?ev;??
- ????ev.data.fd?=?fd;??
- ????ev.events?=?EPOLLIN;??
- ????if(?enable_et?)??
- ????????ev.events?=?EPOLLIN?|?EPOLLET;??
- ????epoll_ctl(epollfd,?EPOLL_CTL_ADD,?fd,?&ev);??
- ????setnonblocking(fd);??
- ????printf("fd?added?to?epoll!\n\n");??
- }??
3. 服務端實現
上面我們介紹了基本的模型和技術,現在該去實現服務端了。首先介紹下utility.h中一些變量和函數。
3.1 utility.h
- /*?限于篇幅,這里先介紹下utility.h的主要構成。其中的頭文件和一些函數實現沒有顯示,完整源碼位于3.2節?*/??
- ????//服務端存儲所有在線用戶socket,?便于廣播信息??
- ????list<int>?clients_list;??
- ????//?服務器ip地址,為測試使用本地機地址,可以更改為其他服務端地址??
- ????#define?SERVER_IP?"127.0.0.1"??
- ????//?服務器端口號??
- ????#define?SERVER_PORT?8888??
- ????//int?epoll_create(int?size)中的size,為epoll支持的最大句柄數??
- ????#define?EPOLL_SIZE?5000??
- ??
- ????//?緩沖區大小65535??
- ????#define?BUF_SIZE?0xFFFF??
- ????//一些宏??
- ????#define?SERVER_WELCOME?"Welcome?you?join?to?the?chat?room!?Your?chat?ID?is:?Client?#%d"??
- ????#define?SERVER_MESSAGE?"ClientID?%d?say?>>?%s"??
- ????#define?EXIT?"EXIT"??
- ????#define?CAUTION?"There?is?only?one?int?the?char?room!"??
- ????/*?一些函數?*/??
- ????//設置非阻塞??
- ????int?setnonblocking(int?sockfd);??
- ????//將文件描述符fd添加到epollfd標示的內核事件表??
- ????void?addfd(?int?epollfd,?int?fd,?bool?enable_et?);??
- ????//服務端發送廣播信息,使所有用戶都能收到消息??
- ????int?sendBroadcastmessage(int?clientfd);??
- 3.1?utility.h完整源碼??
- ??
- #ifndef?UTILITY_H_INCLUDED??
- #define?UTILITY_H_INCLUDED??
- ??
- #include?<iostream>??
- #include?<list>??
- #include?<sys/types.h>??
- #include?<sys/socket.h>??
- #include?<netinet/in.h>??
- #include?<arpa/inet.h>??
- #include?<sys/epoll.h>??
- #include?<fcntl.h>??
- #include?<errno.h>??
- #include?<unistd.h>??
- #include?<stdio.h>??
- #include?<stdlib.h>??
- #include?<string.h>??
- ??
- using?namespace?std;??
- ??
- //?clients_list?save?all?the?clients's?socket??
- list<int>?clients_list;??
- ??
- /**********************???macro?defintion?**************************/??
- //?server?ip??
- #define?SERVER_IP?"127.0.0.1"??
- ??
- //?server?port??
- #define?SERVER_PORT?8888??
- ??
- //epoll?size??
- #define?EPOLL_SIZE?5000??
- ??
- //message?buffer?size??
- #define?BUF_SIZE?0xFFFF??
- ??
- #define?SERVER_WELCOME?"Welcome?you?join??to?the?chat?room!?Your?chat?ID?is:?Client?#%d"??
- ??
- #define?SERVER_MESSAGE?"ClientID?%d?say?>>?%s"??
- ??
- //?exit??
- #define?EXIT?"EXIT"??
- ??
- #define?CAUTION?"There?is?only?one?int?the?char?room!"??
- ??
- /**********************???some?function?**************************/??
- /**?
- ??*?@param?sockfd:?socket?descriptor?
- ??*?@return?0?
- **/??
- int?setnonblocking(int?sockfd)??
- {??
- ????fcntl(sockfd,?F_SETFL,?fcntl(sockfd,?F_GETFD,?0)|?O_NONBLOCK);??
- ????return?0;??
- }??
- ??
- /**?
- ??*?@param?epollfd:?epoll?handle?
- ??*?@param?fd:?socket?descriptor?
- ??*?@param?enable_et?:?enable_et?=?true,?epoll?use?ET;?otherwise?LT?
- **/??
- void?addfd(?int?epollfd,?int?fd,?bool?enable_et?)??
- {??
- ????struct?epoll_event?ev;??
- ????ev.data.fd?=?fd;??
- ????ev.events?=?EPOLLIN;??
- ????if(?enable_et?)??
- ????????ev.events?=?EPOLLIN?|?EPOLLET;??
- ????epoll_ctl(epollfd,?EPOLL_CTL_ADD,?fd,?&ev);??
- ????setnonblocking(fd);??
- ????printf("fd?added?to?epoll!\n\n");??
- }??
- ??
- /**?
- ??*?@param?clientfd:?socket?descriptor?
- ??*?@return?:?len?
- **/??
- int?sendBroadcastmessage(int?clientfd)??
- {??
- ????//?buf[BUF_SIZE]?receive?new?chat?message??
- ????//?message[BUF_SIZE]?save?format?message??
- ????char?buf[BUF_SIZE],?message[BUF_SIZE];??
- ????bzero(buf,?BUF_SIZE);??
- ????bzero(message,?BUF_SIZE);??
- ??
- ????//?receive?message??
- ????printf("read?from?client(clientID?=?%d)\n",?clientfd);??
- ????int?len?=?recv(clientfd,?buf,?BUF_SIZE,?0);??
- ??
- ????if(len?==?0)??//?len?=?0?means?the?client?closed?connection??
- ????{??
- ????????close(clientfd);??
- ????????clients_list.remove(clientfd);?//server?remove?the?client??
- ????????printf("ClientID?=?%d?closed.\n?now?there?are?%d?client?in?the?char?room\n",?clientfd,?(int)clients_list.size());??
- ??
- ????}??
- ????else??//broadcast?message???
- ????{??
- ????????if(clients_list.size()?==?1)?{?//?this?means?There?is?only?one?int?the?char?room??
- ????????????send(clientfd,?CAUTION,?strlen(CAUTION),?0);??
- ????????????return?len;??
- ????????}??
- ????????//?format?message?to?broadcast??
- ????????sprintf(message,?SERVER_MESSAGE,?clientfd,?buf);??
- ??
- ????????list<int>::iterator?it;??
- ????????for(it?=?clients_list.begin();?it?!=?clients_list.end();?++it)?{??
- ???????????if(*it?!=?clientfd){??
- ????????????????if(?send(*it,?message,?BUF_SIZE,?0)?<?0?)?{?perror("error");?exit(-1);}??
- ???????????}??
- ????????}??
- ????}??
- ????return?len;??
- }??
- #endif?//?UTILITY_H_INCLUDED??
3.3 服務端完整源碼
在上面的基礎上。服務端的代碼就很容易寫出了
- #include?"utility.h"??
- ??
- int?main(int?argc,?char?*argv[])??
- {??
- ????//服務器IP?+?port??
- ????struct?sockaddr_in?serverAddr;??
- ????serverAddr.sin_family?=?PF_INET;??
- ????serverAddr.sin_port?=?htons(SERVER_PORT);??
- ????serverAddr.sin_addr.s_addr?=?inet_addr(SERVER_IP);??
- ????//創建監聽socket??
- ????int?listener?=?socket(PF_INET,?SOCK_STREAM,?0);??
- ????if(listener?<?0)?{?perror("listener");?exit(-1);}??
- ????printf("listen?socket?created?\n");??
- ????//綁定地址??
- ????if(?bind(listener,?(struct?sockaddr?*)&serverAddr,?sizeof(serverAddr))?<?0)?{??
- ????????perror("bind?error");??
- ????????exit(-1);??
- ????}??
- ????//監聽??
- ????int?ret?=?listen(listener,?5);??
- ????if(ret?<?0)?{?perror("listen?error");?exit(-1);}??
- ????printf("Start?to?listen:?%s\n",?SERVER_IP);??
- ????//在內核中創建事件表??
- ????int?epfd?=?epoll_create(EPOLL_SIZE);??
- ????if(epfd?<?0)?{?perror("epfd?error");?exit(-1);}??
- ????printf("epoll?created,?epollfd?=?%d\n",?epfd);??
- ????static?struct?epoll_event?events[EPOLL_SIZE];??
- ????//往內核事件表里添加事件??
- ????addfd(epfd,?listener,?true);??
- ????//主循環??
- ????while(1)??
- ????{??
- ????????//epoll_events_count表示就緒事件的數目??
- ????????int?epoll_events_count?=?epoll_wait(epfd,?events,?EPOLL_SIZE,?-1);??
- ????????if(epoll_events_count?<?0)?{??
- ????????????perror("epoll?failure");??
- ????????????break;??
- ????????}??
- ??
- ????????printf("epoll_events_count?=?%d\n",?epoll_events_count);??
- ????????//處理這epoll_events_count個就緒事件??
- ????????for(int?i?=?0;?i?<?epoll_events_count;?++i)??
- ????????{??
- ????????????int?sockfd?=?events[i].data.fd;??
- ????????????//新用戶連接??
- ????????????if(sockfd?==?listener)??
- ????????????{??
- ????????????????struct?sockaddr_in?client_address;??
- ????????????????socklen_t?client_addrLength?=?sizeof(struct?sockaddr_in);??
- ????????????????int?clientfd?=?accept(?listener,?(?struct?sockaddr*?)&client_address,?&client_addrLength?);??
- ??
- ????????????????printf("client?connection?from:?%s?:?%?d(IP?:?port),?clientfd?=?%d?\n",??
- ????????????????inet_ntoa(client_address.sin_addr),??
- ????????????????ntohs(client_address.sin_port),??
- ????????????????clientfd);??
- ??
- ????????????????addfd(epfd,?clientfd,?true);??
- ??
- ????????????????//?服務端用list保存用戶連接??
- ????????????????clients_list.push_back(clientfd);??
- ????????????????printf("Add?new?clientfd?=?%d?to?epoll\n",?clientfd);??
- ????????????????printf("Now?there?are?%d?clients?int?the?chat?room\n",?(int)clients_list.size());??
- ??
- ????????????????//?服務端發送歡迎信息????
- ????????????????printf("welcome?message\n");??????????????????
- ????????????????char?message[BUF_SIZE];??
- ????????????????bzero(message,?BUF_SIZE);??
- ????????????????sprintf(message,?SERVER_WELCOME,?clientfd);??
- ????????????????int?ret?=?send(clientfd,?message,?BUF_SIZE,?0);??
- ????????????????if(ret?<?0)?{?perror("send?error");?exit(-1);?}??
- ????????????}??
- ????????????//處理用戶發來的消息,并廣播,使其他用戶收到信息??
- ????????????else???
- ????????????{?????
- ????????????????int?ret?=?sendBroadcastmessage(sockfd);??
- ????????????????if(ret?<?0)?{?perror("error");exit(-1);?}??
- ????????????}??
- ????????}??
- ????}??
- ????close(listener);?//關閉socket??
- ????close(epfd);????//關閉內核??
- ????return?0;??
- }??
g++ server.cpp utility.h -o server
./server
4. 客戶端實現
4.1 子進程和父進程的通信
前面已經介紹了子進程和父進程的功能,他們之間用管道進行通信。如下圖所示,我們可以更直觀的了解子進程和父進程各自的功能。?
通過調用int pipe(int fd[2])函數創建管道, 其中fd[0]用于父進程讀, fd[1]用于子進程寫。- //client.cpp代碼(管道模塊)??
- ???//?創建管道.??
- ????int?pipe_fd[2];??
- ????if(pipe(pipe_fd)?<?0)?{?perror("pipe?error");?exit(-1);?}??
通過int pid = fork()函數,創建子進程,當pid < 0 錯誤;當pid = 0, 說明是子進程;當pid > 0說明是父進程。根據pid的值,我們可以父子進程,從而實現對應的功能!
4.2 客戶端完整源碼
根據上述介紹,我們可以寫出客戶端的源碼。如下:
- #include?"utility.h"??
- ??
- int?main(int?argc,?char?*argv[])??
- {??
- ????//用戶連接的服務器?IP?+?port??
- ????struct?sockaddr_in?serverAddr;??
- ????serverAddr.sin_family?=?PF_INET;??
- ????serverAddr.sin_port?=?htons(SERVER_PORT);??
- ????serverAddr.sin_addr.s_addr?=?inet_addr(SERVER_IP);??
- ??
- ????//?創建socket??
- ????int?sock?=?socket(PF_INET,?SOCK_STREAM,?0);??
- ????if(sock?<?0)?{?perror("sock?error");?exit(-1);?}??
- ????//?連接服務端??
- ????if(connect(sock,?(struct?sockaddr?*)&serverAddr,?sizeof(serverAddr))?<?0)?{??
- ????????perror("connect?error");??
- ????????exit(-1);??
- ????}??
- ??
- ????//?創建管道,其中fd[0]用于父進程讀,fd[1]用于子進程寫??
- ????int?pipe_fd[2];??
- ????if(pipe(pipe_fd)?<?0)?{?perror("pipe?error");?exit(-1);?}??
- ??
- ????//?創建epoll??
- ????int?epfd?=?epoll_create(EPOLL_SIZE);??
- ????if(epfd?<?0)?{?perror("epfd?error");?exit(-1);?}??
- ????static?struct?epoll_event?events[2];???
- ????//將sock和管道讀端描述符都添加到內核事件表中??
- ????addfd(epfd,?sock,?true);??
- ????addfd(epfd,?pipe_fd[0],?true);??
- ????//?表示客戶端是否正常工作??
- ????bool?isClientwork?=?true;??
- ??
- ????//?聊天信息緩沖區??
- ????char?message[BUF_SIZE];??
- ??
- ????//?Fork??
- ????int?pid?=?fork();??
- ????if(pid?<?0)?{?perror("fork?error");?exit(-1);?}??
- ????else?if(pid?==?0)??????//?子進程??
- ????{??
- ????????//子進程負責寫入管道,因此先關閉讀端??
- ????????close(pipe_fd[0]);???
- ????????printf("Please?input?'exit'?to?exit?the?chat?room\n");??
- ??
- ????????while(isClientwork){??
- ????????????bzero(&message,?BUF_SIZE);??
- ????????????fgets(message,?BUF_SIZE,?stdin);??
- ??
- ????????????//?客戶輸出exit,退出??
- ????????????if(strncasecmp(message,?EXIT,?strlen(EXIT))?==?0){??
- ????????????????isClientwork?=?0;??
- ????????????}??
- ????????????//?子進程將信息寫入管道??
- ????????????else?{??
- ????????????????if(?write(pipe_fd[1],?message,?strlen(message)?-?1?)?<?0?)??
- ?????????????????{?perror("fork?error");?exit(-1);?}??
- ????????????}??
- ????????}??
- ????}??
- ????else??//pid?>?0?父進程??
- ????{??
- ????????//父進程負責讀管道數據,因此先關閉寫端??
- ????????close(pipe_fd[1]);???
- ??
- ????????//?主循環(epoll_wait)??
- ????????while(isClientwork)?{??
- ????????????int?epoll_events_count?=?epoll_wait(?epfd,?events,?2,?-1?);??
- ????????????//處理就緒事件??
- ????????????for(int?i?=?0;?i?<?epoll_events_count?;?++i)??
- ????????????{??
- ????????????????bzero(&message,?BUF_SIZE);??
- ??
- ????????????????//服務端發來消息??
- ????????????????if(events[i].data.fd?==?sock)??
- ????????????????{??
- ????????????????????//接受服務端消息??
- ????????????????????int?ret?=?recv(sock,?message,?BUF_SIZE,?0);??
- ??
- ????????????????????//?ret=?0?服務端關閉??
- ????????????????????if(ret?==?0)?{??
- ????????????????????????printf("Server?closed?connection:?%d\n",?sock);??
- ????????????????????????close(sock);??
- ????????????????????????isClientwork?=?0;??
- ????????????????????}??
- ????????????????????else?printf("%s\n",?message);??
- ??
- ????????????????}??
- ????????????????//子進程寫入事件發生,父進程處理并發送服務端??
- ????????????????else?{???
- ????????????????????//父進程從管道中讀取數據??
- ????????????????????int?ret?=?read(events[i].data.fd,?message,?BUF_SIZE);??
- ??
- ????????????????????//?ret?=?0??
- ????????????????????if(ret?==?0)?isClientwork?=?0;??
- ????????????????????else{???//?將信息發送給服務端??
- ??????????????????????send(sock,?message,?BUF_SIZE,?0);??
- ????????????????????}??
- ????????????????}??
- ????????????}//for??
- ????????}//while??
- ????}??
- ??
- ????if(pid){??
- ???????//關閉父進程和sock??
- ????????close(pipe_fd[0]);??
- ????????close(sock);??
- ????}else{??
- ????????//關閉子進程??
- ????????close(pipe_fd[1]);??
- ????}??
- ????return?0;??
- }??
cd Desktop
g++ client.cpp utility.h -o client
./client
如圖所示,通過查看兩個終端界面,可以看到有一個用戶登陸服務端了。 同理,再點擊一下桌面上的XFce,開啟一個終端,運行同樣的命令(這里不用運行g++進行編譯了,因為前面已經生成了可執行文件client):cd Desktop
./client
轉載自:https://www.shiyanlou.com/courses/315很好的學習了epoll。