socket INADDR_ANY
linux下的socket INADDR_ANY表示的是一個服務(wù)器上所有的網(wǎng)卡(服務(wù)器可能不止一個網(wǎng)卡)
多個本地ip地址都進行綁定端口號,進行偵聽。
不光是多個網(wǎng)卡的問題.
見如下server listen:
80 0.0.0.0 //INADDR_ANY,外部的client ask 從哪個server的地址近來都可以連接到80端口.
8088 192.168.1.11 //外部的client ask 從server地址192.168.1.11進來才可以連接到8088端口.
8089 192.168.1.12 //外部的client ask 從server地址192.168.1.12進來才可以連接到8089端口.
也就是說0.0.0.0 是指本地的地址(也就是代表了所有本地的地址,同一個網(wǎng)卡上也可能有多個地址).
這點上linux,windows系統(tǒng)都是相同的.
而對于在connect中指定了INADDR_ANY,那么:
1. 在語義上一定是連接到本地地址,不可能是外部地址.
2. INADDR_ANY在語義上有可能是對應(yīng)了幾個本地地址,因此有的系統(tǒng)會根據(jù)缺省規(guī)則連接本地指定的服務(wù),而有的系統(tǒng)則因為不能確定用戶的任意本地地址是哪個而不能有效連接(如linux和windows不同).
INADDR_ANY就是指定地址為0.0.0.0的地址,這個地址事實上表示不確定地址,或“所有地址”、“任意地址”。 一般來說,在各個系統(tǒng)中均定義成為0值。
例如在ubuntu的/usr/include/netinet/in.h定義為:
/* Address to accept any incoming messages. */
#define INADDR_ANY ((in_addr_t) 0x00000000)
一 般情況下,如果你要建立網(wǎng)絡(luò)服務(wù)器應(yīng)用程序,則你要通知服務(wù)器操作系統(tǒng):請在某地址 xxx.xxx.xxx.xxx上的某端口 yyyy上進行偵聽,并且把偵聽到的數(shù)據(jù)包發(fā)送給我。這個過程,你是通過bind()系統(tǒng)調(diào)用完成的。——也就是說,你的程序要綁定服務(wù)器的某地址,或者 說:把服務(wù)器的某地址上的某端口占為已用。服務(wù)器操作系統(tǒng)可以給你這個指定的地址,也可以不給你。
如果你的服務(wù)器有多個網(wǎng)卡(每個網(wǎng)卡上有不同的 IP地址),而你的服務(wù)(不管是在udp端口上偵聽,還是在tcp端口上偵聽),出于某種原因:可能是你的服務(wù)器操作系統(tǒng)可能隨時增減IP地址,也有可能 是為了省去確定服務(wù)器上有什么網(wǎng)絡(luò)端口(網(wǎng)卡)的麻煩 —— 可以要在調(diào)用bind()的時候,告訴操作系統(tǒng):“我需要在 yyyy 端口上偵聽,所以發(fā)送到服務(wù)器的這個端口,不管是哪個網(wǎng)卡/哪個IP地址接收到的數(shù)據(jù),都是我處理的。”這時候,服務(wù)器程序則在0.0.0.0這個地址上 進行偵聽。例如:
Proto Recv-Q Send-Q Local Address Foreign Address (state)
……
udp4 0 0 *.7913 *.*
udp4 0 0 *.7911 *.*
tcp4 0 0 *.ftp *.* LISTEN
……
……
以上這些是網(wǎng)絡(luò)偵聽的情況,其中Local Address 為 “*.ftp”、“*.7911”等,代表了服務(wù)程序綁定了服務(wù)器的所有網(wǎng)卡。
好了,你明白了偵聽INADDR_ANY是什么意思了,那么,我的服務(wù)器有N個IP地址,會不會收到重復的數(shù)據(jù)包?收到數(shù)據(jù)包后,是不是會重復回復客戶端呢?
答案是:不會收到重復的數(shù)據(jù)包,也不會重復發(fā)送數(shù)據(jù)。
為 什么呢?因為路由的關(guān)系,從客戶端來的IP包只可能到達其中一個網(wǎng)卡。同時在服務(wù)器進程發(fā)送數(shù)據(jù)時,操作系統(tǒng)根據(jù)自身維護著的路由表,決定IP數(shù)據(jù)包應(yīng)該 c從哪一個outbound的gateway向目標端發(fā)送。根據(jù)gateway選擇的不同,也就決定了從哪一個網(wǎng)卡/哪個IP地址發(fā)送。
為什么不會接收到重復的數(shù)據(jù)包呢?
答:因為客戶端只向你的服務(wù)器上的唯一一個IP地址發(fā)送數(shù)據(jù)了。
為什么不會重復發(fā)送數(shù)據(jù)包呢?
答:因為發(fā)送數(shù)據(jù)包的路由(路徑)是唯一的。如果服務(wù)器不知道在發(fā)送數(shù)據(jù)的時候應(yīng)該向哪個地址發(fā)送數(shù)據(jù),那么數(shù)據(jù)就會被發(fā)送到“默認網(wǎng)關(guān)”上。
如何選擇發(fā)送數(shù)據(jù)的路徑呢?
答:依照路由表的要求發(fā)送。
如果路由表的記錄有重復/有沖突呢,這時候如何選擇路徑呢?
答:路由表記錄有優(yōu)先級別。一般來說,Windows操作系統(tǒng)的路由表記錄,如果是重復的話,以后來加入的記錄為準,而某些操作系統(tǒng),象linux/FreeBSD是不允許加入重復的路由表記錄的;
如果是專用的路由器,有路由選擇算法,一般來說,到達網(wǎng)絡(luò)上的某一點的路徑是可以有很多條的。路由選擇算法可以確定“最好的一條路徑”,這條路徑要么是延時最小的,要么是通訊費用最低的,要么是帶寬最高的,要么是跳點最小的——究竟是如何選擇,就看路由器的管理員如何配置了。
INADDR_ANY 的具體含義是,綁定到0.0.0.0。此時,對所有的地址都將是有效的,如果系統(tǒng)考慮冗余,采用多個網(wǎng)卡的話,那么使用此種bind,將在所有網(wǎng)卡上進行綁定。在這種情況下,你可以收到發(fā)送到所有有效地址上數(shù)據(jù)包。
例如:
SOCKADDR_IN Local;
Local.sin_addr.s_addr = htonl(INADDR_ANY);
另外一種方式如下:
SOCKADDR_IN Local;
hostent* thisHost = gethostbyname("");
char* ip = inet_ntoa(*(struct in_addr *)*thisHost->h_addr_list);
Local.sin_addr.s_addr = inet_addr(ip);
在這種方式下,將在系統(tǒng)中當前第一個可用地址上進行綁定。在多網(wǎng)卡的環(huán)境下,可能會出問題。
最常見的方式:
const char LocalIP[] = "192.168.0.100";
SOCKADDR_IN Local;
Local.sin_addr.s_addr = inet_addr(LocalIP);
bind(socket, (LPSOCKADDR)&Local, sizeof(SOCKADDR_IN)
bind的安全問題:
如果你在bind時,使用了INADDR_ANY那么,你將可以在所有有效的地址上進行監(jiān)聽,但是Socket有一個特性:可在同一端口上綁定多個Socket。
讓我們看看下面的情況:假設(shè)你的系統(tǒng)只有一個IP:192.168.0.100,你希望bind到4096端口。對于下面的兩種bind,當數(shù)據(jù)包到達時,誰會接收到呢?
Local.sin_addr.s_addr = htonl(INADDR_ANY);
Local.sin_addr.s_addr = inet_addr(“192.168.0.100”);
WinSocke庫是這樣處理的:誰綁定的最明確,誰獲取數(shù)據(jù)包。顯然,第二種bind將獲取到達的數(shù)據(jù)包。如果避免這種情況呢?使用 SO_EXECLUSINEADDRUSE選項。需要注意的是,此選項在Windows NT 4 Service Pack 4以后(包括SP4)才可以使用。
示例代碼:
#ifndef SO_EXECLUSINEADDRUSE
#define SO_EXECLUSINEADDRUSE ((int)(~SO_REUSEADDR))
#endif
SOCKADDR_IN Local;
BOOL val = TRUE;
Local. sin_family = AF_INET;
Local. sin_port = htons(4096);
Local.sin_addr.s_addr = htonl(INADDR_ANY);
setsocketopt(socket,
SOL_SOCKET,
SO_EXECLUSINEADDRUSE,
(char*)&val,
sizeof(val));
bind(socket, (LPSOCKADDR)&Local, sizeof(SOCKADDR_IN)
對 于客戶端如果綁定INADDR_ANY,情況類似。對于TCP而言,在connect()系統(tǒng)調(diào)用時將其綁頂?shù)揭痪唧w的IP地址。選擇的依據(jù)是該地址所在 子網(wǎng)到目標地址是可達的(reachable). 這時通過getsockname()系統(tǒng)調(diào)用就能得知具體使用哪一個地址。對于UDP而言, 情況比較特殊。即使使用connect()系統(tǒng)調(diào)用也不會綁定到一具體地址。這是因為對UDP使用connect()并不會真正向目標地址發(fā)送任何建立連 接的數(shù)據(jù),也不會驗證到目標地址的可達性。它只是將目標地址的信息記錄在內(nèi)部的socket數(shù)據(jù)結(jié)構(gòu)之中,共以后使用。只有當調(diào)用 sendto()/send()時,由系統(tǒng)內(nèi)核根據(jù)路由表決定由哪一個地址(網(wǎng)卡)發(fā)送UDP packet.
SOCKET bind INADDR_LOOPBACK和INADDR_ANY的區(qū)別
今天寫程序時候,服務(wù)器端啟動了,然后客戶端總是連接不上,connect返回錯誤號是10061,服務(wù)器積極拒絕請求。
用telnet連接一下端口,發(fā)現(xiàn)服務(wù)端服務(wù)沒有開啟,但是我程序是啟動的,用netstat -a 命令看服務(wù)器是監(jiān)聽狀態(tài)。
把流程走一遍,發(fā)現(xiàn)bind用的參數(shù)是INADDR_LOOPBACK,改成INADDR_ANY就OK了。
只是有些困惑,因為之前一個程序用的是INADDR_LOOPBACK運行沒有問題,為何這里運行不了?
sa.sin_addr.s_addr = htonl(INADDR_LOOPBACK); // 1
sa.sin_addr.s_addr = htonl(INADDR_ANY ); //2
兩者的區(qū)別
INADDR_ANY是ANY,是綁定地址0.0.0.0上的監(jiān)聽, 能收到任意一塊網(wǎng)卡的連接;
INADDR_LOOPBACK, 也就是綁定地址LOOPBAC, 往往是127.0.0.1, 只能收到127.0.0.1上面的連接請求
總結(jié)
以上是生活随笔為你收集整理的socket INADDR_ANY的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 读写锁 SRWLOCK
- 下一篇: loj124 除数函数求和 1