校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁 > 知識庫 > linux 網絡編程 socket選項的實現

linux 網絡編程 socket選項的實現

熱門標簽:東莞人工智能電銷機器人供應商 長沙開福怎么申請400電話 高德地圖標注無營業執照 江蘇電銷外呼防封系統是什么 廣州電銷機器人系統圖 智能電話機器人線路 百度地圖標注要不要錢 金融行業外呼線路 賀州市地圖標注app

socket選項函數

功能:用來讀取和設置socket文件描述符屬性的方法

#include <sys/scoket.h>
int getsockopt ( int sockfd, int level, int option_name, void* option_value, socklen_t* restrict option_len );
int setsockopt ( int sockfd, int level, int option_name, const void* option_value, socklen_t option_len);

socket選項表如下:

getsockopt和setsockopt 這兩個函數成功時返回0,失敗時返回-1并設置errno。

對于服務器而言,有部分socket選項只能在調用listen系統調用前針對監聽socket設置才有效。這是因為連接socket只能由accept調用返回,而accept從listen監聽隊列接受的連接至少已經完成了TCP三次握手的前兩個步驟(因為listen監聽隊列中的連接至少已進入SYN_RCVD狀態),這說明服務器已經往被接收連接上發送出了TCP同步報文段。但有的socket選項卻應該在TCP同步報文段中設置,比如TCP最大報文段選項。對這種情況,linux給開發人員提供的解決方案是:對監聽socket設置這些socket選項,那么accept返回的連接socket將自動繼承這些選項。這些選項包括:SO_DEBUG、SO_DONTROUTE、SO_KEEPALIVE、SO_LINGER、SO_OOBINLINE、SO_RCVBUF、SO_RCVLOWAT、SO_SNDBUF、SO_SNDLOWAT、TCP_MAXSEG和TCP_NODELAY。

對于客戶端而言,這些socket選項則應該在調用connect函數之前設置,因為connect調用成功返回之后,TCP三次握手已完成。

SO_REUSEADDR選項

前面討論過TCP連接的TIME_WAIT狀態,并提到服務器程序可以通過設置socket選項SO_REUSEADDR來強制使用被處于TIME_WAIT狀態的連接占用的socket地址。

#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <assert.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
 
int main( int argc, char* argv[] )
{
  if( argc <= 2 )
  {
    printf( "usage: %s ip_address port_number\n", basename( argv[0] ) );
    return 1;
  }
  const char* ip = argv[1];
  int port = atoi( argv[2] );
 
  int sock = socket( PF_INET, SOCK_STREAM, 0 );
  assert( sock >= 0 );
  int reuse = 1;
  setsockopt( sock, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof( reuse ) );
 
  struct sockaddr_in address;
  bzero( &address, sizeof( address ) );
  address.sin_family = AF_INET;
  inet_pton( AF_INET, ip, &address.sin_addr );
  address.sin_port = htons( port );
  int ret = bind( sock, ( struct sockaddr* )&address, sizeof( address ) );
  assert( ret != -1 );
 
  ret = listen( sock, 5 );
  assert( ret != -1 );
 
  struct sockaddr_in client;
  socklen_t client_addrlength = sizeof( client );
  int connfd = accept( sock, ( struct sockaddr* )&client, &client_addrlength );
  if ( connfd < 0 )
  {
    printf( "errno is: %d\n", errno );
  }
  else
  {
    char remote[INET_ADDRSTRLEN ];
    printf( "connected with ip: %s and port: %d\n", 
      inet_ntop( AF_INET, &client.sin_addr, remote, INET_ADDRSTRLEN ), ntohs( client.sin_port ) );
    close( connfd );
  }
 
  close( sock );
  return 0;
}

經過setsocketopt的設置之后,即使sock處于TIME_WAIT狀態,與之綁定的socket地址也可以立即被重用。此外,我們也可以通過修改內核參數/proc/sys/net/ipv4/tcp_tw_recycle 來快速回收被關閉的socket,從而使得TCP連接根本就不進入TIME_WAIT狀態,進而允許應用程序立即重用本地的socket地址。

SO_RCVBUF和SO_SNDBUF選項

SO_RCVBUF和SO_SNDBUF選項分別表示TCP接收緩沖區和發送緩沖區的大小。不過,當我們用setsockopt來設置TCP的接收緩沖區和發送緩沖區的大小時,系統都會將其值加倍,并且不得小于其個最小值。TCP接收緩沖區的最小值是256字節,而發送緩沖區的最小值是2048字節(不過,不同的系統可能有不同的默認最小值)。此外,我們可以直接修改內核參數/proc/sys/net/ipv4/tcp_rmem和/proc/sys/net/ipv4/tcp_wmem來強制TCP接收緩沖區和發送緩沖區的大小沒有最小值限制。

修改TCP發送緩沖區的客戶端程序:

#include <sys/socket.h>
#include <arpa/inet.h>
#include <assert.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
 
#define BUFFER_SIZE 512
 
int main( int argc, char* argv[] )
{
  if( argc <= 3 )
  {
    printf( "usage: %s ip_address port_number send_bufer_size\n", basename( argv[0] ) );
    return 1;
  }
  const char* ip = argv[1];
  int port = atoi( argv[2] );
 
  struct sockaddr_in server_address;
  bzero( &server_address, sizeof( server_address ) );
  server_address.sin_family = AF_INET;
  inet_pton( AF_INET, ip, &server_address.sin_addr );
  server_address.sin_port = htons( port );
 
  int sock = socket( PF_INET, SOCK_STREAM, 0 );
  assert( sock >= 0 );
 
  int sendbuf = atoi( argv[3] );
  int len = sizeof( sendbuf );
  setsockopt( sock, SOL_SOCKET, SO_SNDBUF, &sendbuf, sizeof( sendbuf ) );
  getsockopt( sock, SOL_SOCKET, SO_SNDBUF, &sendbuf, ( socklen_t* )&len );
  printf( "the tcp send buffer size after setting is %d\n", sendbuf );
 
  if ( connect( sock, ( struct sockaddr* )&server_address, sizeof( server_address ) ) != -1 )
  {
    char buffer[ BUFFER_SIZE ];
    memset( buffer, 'a', BUFFER_SIZE );
    send( sock, buffer, BUFFER_SIZE, 0 );
  }
 
  close( sock );
  return 0;
}

修改TCP接收緩沖區的服務器程序:

#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <assert.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
 
#define BUFFER_SIZE 1024
 
int main( int argc, char* argv[] )
{
  if( argc <= 3 )
  {
    printf( "usage: %s ip_address port_number receive_buffer_size\n", basename( argv[0] ) );
    return 1;
  }
  const char* ip = argv[1];
  int port = atoi( argv[2] );
 
  struct sockaddr_in address;
  bzero( &address, sizeof( address ) );
  address.sin_family = AF_INET;
  inet_pton( AF_INET, ip, &address.sin_addr );
  address.sin_port = htons( port );
 
  int sock = socket( PF_INET, SOCK_STREAM, 0 );
  assert( sock >= 0 );
  int recvbuf = atoi( argv[3] );
  int len = sizeof( recvbuf );
  setsockopt( sock, SOL_SOCKET, SO_RCVBUF, &recvbuf, sizeof( recvbuf ) );
  getsockopt( sock, SOL_SOCKET, SO_RCVBUF, &recvbuf, ( socklen_t* )&len );
  printf( "the receive buffer size after settting is %d\n", recvbuf );
 
  int ret = bind( sock, ( struct sockaddr* )&address, sizeof( address ) );
  assert( ret != -1 );
 
  ret = listen( sock, 5 );
  assert( ret != -1 );
 
  struct sockaddr_in client;
  socklen_t client_addrlength = sizeof( client );
  int connfd = accept( sock, ( struct sockaddr* )&client, &client_addrlength );
  if ( connfd < 0 )
  {
    printf( "errno is: %d\n", errno );
  }
  else
  {
    char buffer[ BUFFER_SIZE ];
    memset( buffer, '\0', BUFFER_SIZE );
    while( recv( connfd, buffer, BUFFER_SIZE-1, 0 ) > 0 ){}
    close( connfd );
  }
 
  close( sock );
  return 0;
}

運行結果:

root@iZbp1anc6yju2dks3nw5j0Z:~/test/socket# ./client 127.0.0.1 12345 2000
the tcp send buffer size after setting is 4608

root@iZbp1anc6yju2dks3nw5j0Z:~/test/socket# ./server 127.0.0.1 12345 50
the receive buffer size after settting is 2304

如上說明:當我們用setsockopt來設置TCP的接收緩沖區和發送緩沖區的大小時,系統都會將其值加倍,并且不得小于其個最小值。

SO_RCVLOWAT和SO_SNDLOWAT選項

  • SO_RCVLOWAT和SO_SNDLOWAT選項分別表示TCP接收緩沖區和發送緩沖區的低水位標記。它們一般被I/O復用系統調用,用來判斷socket是否可讀或可寫。當TCP接收緩沖區中可讀數據的總數大于其低水位標記時,I/O復用系統調用將通知應用程序可以從對應的socket上讀取數據;當TCP發送緩沖區中的空閑空間(可以寫入數據的空間)大于其低水位標記時,I/O復用系統調用將通知應用程序可以往對應的socket上寫入數據。
  • 默認情況下,TCP接收緩沖區的低水位標記和TCP發送緩沖區的低水位標記均為1字節。

SO_LINGER選項

SO_LINGER選項用于控制close系統調用在關閉TCP連接時的行為。默認情況下,當我們使用close系統調用來關閉一個socket時,close將立即返回,TCP模塊負責把該socket對應的TCP發送緩沖區中殘留的數據發送給對方。

設置SO_LINGER選項的值時,我們需要給setsockopt(getsockopt)系統調用傳遞一個linger類型的結構體,其定義如下:

#include <sys/socket.h>
struct linger
{
  int l_onoff; //開啟(非0)還是關閉(0)該選項
  int l_linger; // 滯留時間
};
  • 根據linger結構體中兩個成員變量的不同值,close 系統調用可能產生如下3種行為之一:
  • l_onoff 等于0。此時SO_LINGER選項不起作用,close用默認行為關閉socket。
  • l_onoff 不為0,l_linger等于0. 此時close 系統調用立即返回,TCP模塊將丟棄被關閉的socket對應的TCP發送緩沖區中殘留的數據,同時給對方一個復位報文段。因此,這種情況給服務器提供了異常終止一個連接的方法。l_onoff不為0,l_linger大于0 。此時close的行為取決于兩個條件:(1)被關閉的socket對應的TCP發送緩沖區中是否還有殘留的數據;(2)該socket是阻塞的還是非阻塞的。 對于阻塞的socket,close將等待一段長為l_linger的時間,直到TCP模塊發送完所有殘留數據并得到對方的確認。如果這段之間內TCP模塊沒有發送完殘留數據并得到對方的確認,那么close系統調用將返回-1并設置errno為EWOULDBLOCK。 如果socket是非阻塞的,close將立即返回,此時我們需要根據其返回值和errno來判斷殘留數據是否已經發送完畢。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

標簽:張家界 玉樹 廊坊 北京 洛陽 松原 滄州 永州

巨人網絡通訊聲明:本文標題《linux 網絡編程 socket選項的實現》,本文關鍵詞  linux,網絡編程,socket,選項,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《linux 網絡編程 socket選項的實現》相關的同類信息!
  • 本頁收集關于linux 網絡編程 socket選項的實現的相關信息資訊供網民參考!
  • 推薦文章
    校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃
    精品国产精品网麻豆系列| 91蝌蚪porny| 亚欧色一区w666天堂| 欧美国产日韩精品免费观看| 337p粉嫩大胆噜噜噜噜噜91av | 久久久久亚洲综合| 精品国产青草久久久久福利| 日韩视频123| 国产三级精品视频| 国产精品水嫩水嫩| 亚洲视频一区在线观看| 亚洲精选视频免费看| 亚洲综合一区二区三区| 亚洲福利视频一区二区| 秋霞国产午夜精品免费视频| 免费久久精品视频| 国产中文一区二区三区| 成人福利视频在线看| 91福利国产成人精品照片| 欧美日韩一区二区在线观看视频| 欧美日韩精品是欧美日韩精品| 91精品综合久久久久久| 国产视频一区在线播放| 专区另类欧美日韩| 日本强好片久久久久久aaa| 国产精品自拍毛片| 色天天综合久久久久综合片| 91精品国产综合久久精品图片| 久久综合狠狠综合| 亚洲伦在线观看| 蜜臀av性久久久久蜜臀aⅴ流畅| 国产精品一区二区不卡| 色综合久久综合网97色综合 | 黑人巨大精品欧美一区| www.久久久久久久久| 欧美揉bbbbb揉bbbbb| 久久精品亚洲麻豆av一区二区 | 91精品在线免费| 欧美国产在线观看| 亚洲一区二区在线免费观看视频| 男人的天堂亚洲一区| caoporm超碰国产精品| 91精品1区2区| 国产婷婷色一区二区三区四区 | 老司机午夜精品| 日本精品一区二区三区高清| 久久人人爽人人爽| 免费观看成人鲁鲁鲁鲁鲁视频| 成人一级片在线观看| 欧美精品第1页| 中文字幕一区日韩精品欧美| 午夜视频在线观看一区二区| 国产福利一区在线| 日韩欧美一级特黄在线播放| 一区2区3区在线看| 成人综合婷婷国产精品久久蜜臀| 欧美一级xxx| 亚洲在线一区二区三区| 99久久免费精品高清特色大片| 欧美电视剧免费观看| 日本在线播放一区二区三区| 91久久精品日日躁夜夜躁欧美| 欧美高清在线精品一区| 国产一区二区0| 欧美一区二区三区视频免费| 性做久久久久久免费观看欧美| 91首页免费视频| 亚洲欧洲精品天堂一级| 成人教育av在线| 国产精品久久久久久亚洲毛片 | 欧美日韩一区二区三区在线| 亚洲欧美日韩国产手机在线| 99免费精品在线| 1024成人网| 色哟哟一区二区在线观看| 中文字幕第一区| 成人听书哪个软件好| 欧美国产精品久久| 成人成人成人在线视频| 国产精品不卡在线| 91美女精品福利| 亚洲电影一区二区三区| 欧美久久久久中文字幕| 日韩中文字幕av电影| 日韩视频一区二区三区在线播放| 日韩av一级片| 2020国产精品久久精品美国| 成人免费观看av| 亚洲四区在线观看| 欧美久久高跟鞋激| 免费在线观看日韩欧美| 久久久久久99精品| 99久久精品99国产精品| 亚洲大型综合色站| 欧美成va人片在线观看| 成人app网站| 丝袜亚洲另类欧美| 久久久激情视频| 欧美专区亚洲专区| 精品一区精品二区高清| 中文字幕在线不卡一区二区三区| 色狠狠综合天天综合综合| 亚洲国产综合色| 久久一区二区三区四区| 成人午夜电影网站| 五月激情丁香一区二区三区| 精品美女被调教视频大全网站| 国产成人av一区二区三区在线| 国产精品久久免费看| 欧美日韩免费观看一区二区三区| 麻豆成人久久精品二区三区小说| 国产精品视频你懂的| 欧美日韩国产另类一区| 国产成人精品免费网站| 性做久久久久久免费观看欧美| 26uuu精品一区二区三区四区在线| 91网站在线观看视频| 精品一区二区三区影院在线午夜| 亚洲图片另类小说| 国产亚洲婷婷免费| 欧美日韩国产影片| 国产成人精品影视| 日本不卡在线视频| 一区二区中文字幕在线| 久久久久国色av免费看影院| 欧美日韩在线播放| 成人高清视频免费观看| 激情亚洲综合在线| 日日夜夜精品视频天天综合网| 亚洲色图一区二区| 国产亚洲一区二区三区四区| 日韩小视频在线观看专区| 在线观看亚洲精品视频| 99精品热视频| 国产成人av电影在线| 精品写真视频在线观看| 热久久一区二区| 午夜激情久久久| 亚洲国产精品久久一线不卡| 亚洲天堂网中文字| 亚洲视频在线一区| 18涩涩午夜精品.www| 国产精品全国免费观看高清| 国产色婷婷亚洲99精品小说| 日韩欧美国产三级电影视频| 欧美一区二区三区日韩| 欧美日韩另类一区| 欧美日韩一区不卡| 欧美在线999| 欧美性生活久久| 欧美午夜电影一区| 欧美群妇大交群中文字幕| 欧美日韩精品电影| 日韩亚洲欧美高清| 日韩欧美视频在线| 精品国产99国产精品| 久久香蕉国产线看观看99| 久久综合狠狠综合| 欧美激情资源网| 亚洲欧美日韩国产手机在线| 一区二区三区在线视频免费观看 | 欧美日韩在线免费视频| 欧美少妇性性性| 777a∨成人精品桃花网| 日韩亚洲国产中文字幕欧美| 久久综合久久综合亚洲| 日本一区二区三区在线不卡| 国产精品夫妻自拍| 亚洲成va人在线观看| 蜜臀久久久久久久| 国产电影一区二区三区| 一本久久a久久精品亚洲| 欧美视频一区在线| 欧美二区乱c少妇| 久久久国产午夜精品| 亚洲欧洲av在线| 图片区小说区区亚洲影院| 日韩精品电影在线| 成人一区二区三区在线观看 | 日韩一区二区三区视频在线| 久久综合资源网| 亚洲欧美韩国综合色| 青娱乐精品在线视频| 国产91精品免费| 欧美日韩免费高清一区色橹橹| 精品美女一区二区三区| 亚洲激情自拍偷拍| 久久国产精品区| 91麻豆.com| 精品国产sm最大网站免费看| 亚洲三级在线看| 久久99国产精品久久99果冻传媒| 不卡的看片网站| 日韩三级视频中文字幕| 国产精品色哟哟网站| 美女网站色91| 欧美色男人天堂| 亚洲女人的天堂| 懂色中文一区二区在线播放| 91精品国产品国语在线不卡|