自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

端口復(fù)用之So_Reuseaddr

網(wǎng)絡(luò) 網(wǎng)絡(luò)管理
端口復(fù)用是網(wǎng)絡(luò)編程里的經(jīng)典問(wèn)題,同時(shí)這里面的知識(shí)點(diǎn)又非常繁瑣,本文通過(guò)代碼簡(jiǎn)單介紹一下 SO_REUSEADDR,但不會(huì)涉及到 SO_REUSEPORT。

端口復(fù)用是網(wǎng)絡(luò)編程里的經(jīng)典問(wèn)題,同時(shí)這里面的知識(shí)點(diǎn)又非常繁瑣,本文通過(guò)代碼簡(jiǎn)單介紹一下 SO_REUSEADDR,但不會(huì)涉及到 SO_REUSEPORT。

長(zhǎng)期以來(lái),我們都有一個(gè)認(rèn)知,就是不能監(jiān)聽(tīng)同一個(gè)端口。比如以下代碼。

server1.listen(8080);
server2.listen(8080);

我們就會(huì)看到 Address already in use 的錯(cuò)誤。但是真的不能綁定到同一個(gè)端口嗎?不一定。

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <unistd.h>
#include <arpa/inet.h>

void start_server(__uint32_t host) {
int listenfd, connfd;
struct sockaddr_in servaddr;

if( (listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1 ){
goto ERROR;
}
memset(&servaddr, 0, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = host;
servaddr.sin_port = htons(6666);

if(bind(listenfd, (struct sockaddr*)&servaddr, sizeof(servaddr)) == -1){
goto ERROR;
}

if(listen(listenfd, 10) == -1) {
goto ERROR;
}
return;
ERROR:
printf("bind socket error: %s(errno: %d)\n", strerror(errno), errno);
}

int main(){
start_server(inet_addr("127.0.0.1"));
start_server(inet_addr("192.168.8.246"));
}

上面的代碼啟動(dòng)了兩個(gè)服務(wù)器,兩個(gè)服務(wù)器都綁定了同一個(gè)端口,編譯執(zhí)行是可以正常跑的,因?yàn)槲抑付瞬煌? IP。由此可見(jiàn),平時(shí)我們認(rèn)為多個(gè)服務(wù)器不能同時(shí)監(jiān)聽(tīng)同一個(gè)端口是因?yàn)槲覀冎恢付硕丝?,而沒(méi)有指定 IP。

const net = require('net');
const server = net.createServer();
server.listen(8080);

執(zhí)行以上代碼,通過(guò) lsof -i:8080 可以看到綁定的地址 *:8080。也就是說(shuō),如果我們沒(méi)有指定 IP,那么系統(tǒng)就會(huì)默認(rèn)監(jiān)聽(tīng)全部 IP。當(dāng)?shù)诙伪O(jiān)聽(tīng)同一個(gè)端口時(shí)就會(huì)報(bào)錯(cuò)。接著看第二種情況。

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <unistd.h>
#include <arpa/inet.h>

void start_server(__uint32_t host) {
int listenfd, connfd;
struct sockaddr_in servaddr;

if( (listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1 ){
goto ERROR;
}
memset(&servaddr, 0, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = host;
servaddr.sin_port = htons(6666);

if(bind(listenfd, (struct sockaddr*)&servaddr, sizeof(servaddr)) == -1){
goto ERROR;
}

if(listen(listenfd, 10) == -1) {
goto ERROR;
}
return;
ERROR:
printf("bind socket error: %s(errno: %d)\n", strerror(errno), errno);
}

int main(){
start_server(htonl(INADDR_ANY));
start_server(inet_addr("127.0.0.1"));
}

上面的代碼執(zhí)行會(huì)報(bào)錯(cuò) Address already in use。為什么改成 INADDR_ANY 就不行了呢?因?yàn)?INADDR_ANY 代表的是全部 IP,這樣默認(rèn)情況下就無(wú)法綁定到其他 IP 了。從邏輯上來(lái)說(shuō)就是當(dāng)操作系統(tǒng)收到這個(gè)127.0.0.1:6666 的數(shù)據(jù)包時(shí),不知道該給誰(shuí)處理,因?yàn)榻壎ǖ膬蓚€(gè)地址都命中了。但是我們可以告訴操作系統(tǒng)把這個(gè)數(shù)據(jù)包給誰(shuí)。

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <unistd.h>
#include <arpa/inet.h>

void start_server(__uint32_t host) {
int listenfd, connfd;
struct sockaddr_in servaddr;

if( (listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1 ){
goto ERROR;
}
int on = 1;
if (setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on))) {
goto ERROR;
}

memset(&servaddr, 0, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = host;
servaddr.sin_port = htons(6666);

if(bind(listenfd, (struct sockaddr*)&servaddr, sizeof(servaddr)) == -1){
goto ERROR;
}

if(listen(listenfd, 10) == -1) {
goto ERROR;
}
return;
ERROR:
printf("bind socket error: %s(errno: %d)\n", strerror(errno), errno);
}

int main(){
start_server(htonl(INADDR_ANY));
start_server(inet_addr("127.0.0.1"));
}

上面代碼加入了 SO_REUSEADDR 的邏輯,編譯執(zhí)行成功。由此可見(jiàn),SO_REUSEADDR 就是告訴操作系統(tǒng)當(dāng)一個(gè)數(shù)據(jù)包命中多個(gè)socket時(shí)應(yīng)該給誰(shuí)處理,操作系統(tǒng)明確了這個(gè)邏輯后,自然也就允許以這種方式監(jiān)聽(tīng)端口了。

責(zé)任編輯:武曉燕 來(lái)源: 編程雜技
相關(guān)推薦

2021-05-31 06:50:47

SelectPoll系統(tǒng)

2010-07-19 10:14:20

云開(kāi)發(fā)

2017-11-08 12:51:12

2015-05-08 09:57:59

綁定端口端口復(fù)用網(wǎng)絡(luò)編程

2009-09-21 17:30:25

組件復(fù)用服務(wù)復(fù)用

2023-04-12 11:36:07

波分復(fù)用WDM

2023-08-24 16:45:16

應(yīng)用開(kāi)發(fā)父自定義組件

2018-08-23 08:58:38

復(fù)用著相代碼

2009-07-15 09:42:56

MyEclipse使用

2022-05-16 13:46:38

Redis高可用Sentinel

2009-07-09 09:38:43

JDK使用圖解

2009-07-15 09:59:36

MyEclipse使用

2009-07-15 09:59:36

MyEclipse使用

2021-04-21 09:55:24

Redis應(yīng)用限流

2010-06-10 13:47:16

2013-01-29 10:41:58

Java復(fù)用HTTP

2020-11-02 08:23:36

shell腳本Linux

2012-03-07 09:02:29

代碼復(fù)用

2010-07-06 15:46:41

UDP協(xié)議

2010-07-20 16:30:44

MySQL內(nèi)存
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)