亚洲视频二区_亚洲欧洲日本天天堂在线观看_日韩一区二区在线观看_中文字幕不卡一区

公告:魔扣目錄網為廣大站長提供免費收錄網站服務,提交前請做好本站友鏈:【 網站目錄:http://www.430618.com 】, 免友鏈快審服務(50元/站),

點擊這里在線咨詢客服
新站提交
  • 網站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會員:747

socket用listen函數監聽,listen從英語上理解就是一個"聽"函數,實際上它也就是這個意思。

我們來看unix網絡編程這本書是怎樣對它的解釋:listen函數把一個未連接的套接字轉換成一個被動套接字,指示內核應該接受指向該套接字的鏈接請求。

該函數有2個參數,第一個我就不說了,第二參數規定了內核為相應套接字排隊的最大連接個數。只看這些理論搞的人稀里糊涂,我們還是來測一下。

[mapan@localhost test]$ ls
client.cpp  makefile  server.cpp
[mapan@localhost test]$ 
[mapan@localhost test]$ cat server.cpp 
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <errno.h>
#include <malloc.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/ioctl.h>
#include <stdarg.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <signal.h>
#define MAXLINE 4096



void main()
{
   int listenfd,connfd;
   socklen_t  clilen;
   struct sockaddr_in cliaddr,servaddr;


   listenfd=socket(AF_INET,SOCK_STREAM,0);
   bzero(&servaddr,sizeof(servaddr));


   servaddr.sin_family=AF_INET;
   servaddr.sin_addr.s_addr=INADDR_ANY;
   servaddr.sin_port=htons(8888);


   bind(listenfd,(struct sockaddr *)&servaddr,sizeof(servaddr));  
   listen(listenfd,1);


   getchar();
   connfd=accept(listenfd,(struct sockaddr *)&cliaddr,&clilen);



   close(connfd);
   close(listenfd);
}
[mapan@localhost test]$ cat client.cpp 
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <errno.h>
#include <malloc.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/ioctl.h>
#include <stdarg.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <signal.h>
#define MAXLINE 4096




void main()
{
   int sockfd;
   struct sockaddr_in servaddr;




   sockfd=socket(AF_INET,SOCK_STREAM,0);
   bzero(&servaddr,sizeof(servaddr));
   servaddr.sin_family=AF_INET;
   servaddr.sin_port=htons(8888);
   servaddr.sin_addr.s_addr = inet_addr("127.0.0.1");


   int ret=connect(sockfd,(struct sockaddr *)&servaddr,sizeof(servaddr));
   getchar();


   close(sockfd);
}
[mapan@localhost test]$ cat makefile 
all:server client


server.o:server.cpp
  g++ -c server.cpp
client.o:client.cpp
  g++ -c client.cpp
server:server.o
  g++ -o server server.o
client:client.o
  g++ -o client client.o


clean:
  rm -f server client *.o
[mapan@localhost test]$

請注意上面的服務端中,我是沒有調用accept函數的,直接調用getchar()了,跑起來。

[mapan@localhost test]$ make
g++ -c server.cpp
g++ -o server server.o
g++ -c client.cpp
g++ -o client client.o
[mapan@localhost test]$ ./server 


服務度開啟,然后新打開一個窗口開啟客戶端。
[mapan@localhost TCP]$ cd ../test/
[mapan@localhost test]$ 
[mapan@localhost test]$ ./client 127.0.0.1

查看網絡:

[mapan@localhost test]$ netstat -na | grep 8888
tcp        0      0 0.0.0.0:8888                0.0.0.0:*                   LISTEN      
tcp        0      0 127.0.0.1:34846             127.0.0.1:8888              ESTABLISHED 
tcp        0      0 127.0.0.1:8888              127.0.0.1:34846             ESTABLISHED 
[mapan@localhost test]$

看,已經建立起一個連接了。但是我們沒有調用accept函數,連接還是建立起來了,這說說明accept函數和TCP三次握手沒啥關系,這也是一個知識盲點。好,在開啟一個新窗口運行客戶端,查看網絡狀態。(新開窗口運行客戶端同上,這里就不用代碼演示了)

[mapan@localhost test]$ netstat -na | grep 8888
tcp        0      0 0.0.0.0:8888                0.0.0.0:*                   LISTEN      
tcp        0      0 127.0.0.1:34846             127.0.0.1:8888              ESTABLISHED 
tcp        0      0 127.0.0.1:34848             127.0.0.1:8888              ESTABLISHED 
tcp        0      0 127.0.0.1:8888              127.0.0.1:34846             ESTABLISHED 
tcp        0      0 127.0.0.1:8888              127.0.0.1:34848             ESTABLISHED

看,又建立起一個連接。在運行一個客戶端,看網絡狀態。

[mapan@localhost test]$ netstat -na | grep 8888
tcp        0      0 0.0.0.0:8888                0.0.0.0:*                   LISTEN      
tcp        0      0 127.0.0.1:8888              127.0.0.1:34850             SYN_RECV    
tcp        0      0 127.0.0.1:34846             127.0.0.1:8888              ESTABLISHED 
tcp        0      0 127.0.0.1:34848             127.0.0.1:8888              ESTABLISHED 
tcp        0      0 127.0.0.1:8888              127.0.0.1:34846             ESTABLISHED 
tcp        0      0 127.0.0.1:8888              127.0.0.1:34848             ESTABLISHED 
tcp        0      0 127.0.0.1:34850             127.0.0.1:8888              ESTABLISHED

當第三個客戶端連接進來的時候,出現了一個SYN_RECV,這標明第三個客戶端沒有與服務端建立連接。

我們listen函數設置的監聽隊列為1,那么監聽隊列塞了2個之后就沒有往里面塞了。這下大概懂了listen函數第二個參數的意義了吧,當參數為1的時候只能監聽2個套接字,這應該是從0開始數的。

為什么是大概呢?其實unix網絡編程上是這樣說的:listen函數的第二個參數是ESTABLISHED和SYN_RECV之和,只是在監聽隊列沒有滿的情況下,SYN_RECV狀態不容易重現。這時候在服務度輸入一個字符會有啥效果呢?

答案告訴你,就是那個SYN_RECV狀態變成ESTABLISHED了,這也是 accept函數的作用。accept函數會將已完成連接隊列中的對頭項返回給進程,所以SYN_RECV變成ESTABLISHED了。這個現象留給大家去實踐一下吧,只有自己實踐出來的東西才是自己的。

 

原文出自:公眾號 盼盼編程

原文鏈接:
https://mp.weixin.qq.com/s/zGzlpHavPdgSaMTu2Vs9tg

分享到:
標簽:監聽 socket
用戶無頭像

網友整理

注冊時間:

網站:5 個   小程序:0 個  文章:12 篇

  • 51998

    網站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會員

趕快注冊賬號,推廣您的網站吧!
最新入駐小程序

數獨大挑戰2018-06-03

數獨一種數學游戲,玩家需要根據9

答題星2018-06-03

您可以通過答題星輕松地創建試卷

全階人生考試2018-06-03

各種考試題,題庫,初中,高中,大學四六

運動步數有氧達人2018-06-03

記錄運動步數,積累氧氣值。還可偷

每日養生app2018-06-03

每日養生,天天健康

體育訓練成績評定2018-06-03

通用課目體育訓練成績評定