由于,本人是主修java的,所以以下內(nèi)容可能不是很精通,各位看完后盡可評論。
以下皆是在linux的描述
第一步,通過socket拿到fd
Socket()函數(shù):創(chuàng)建用于通信的端點(diǎn)并返回描述符。
int fd = socket(AF_INET, SOCK_STREAM, 0);
它的第一個參數(shù)便是通信域,這里我舉出常用的。
AF_INET是ipv4的。
AF_INET6是ipv6的。
它的第二個參數(shù)便是套接字具有指定的類型,指定通信語義
套接字類型與協(xié)議族所相關(guān)。
常用的套接字類型有:
SOCK_STREAM:提供有序的、可靠的、雙向的、基于連接的字節(jié)流。一種帶外數(shù)據(jù)傳輸裝置
主義可能得到支持。
SOCK_DGRAM:支持?jǐn)?shù)據(jù)報(bào)(無連接、最大長度固定的不可靠消息)。
看上去是不是感覺就是,一個是tcp,另外一個是udp呢0.0.
它的第三個參數(shù)是協(xié)議族的標(biāo)識符。這里的0表示使用默認(rèn)的協(xié)議族。
該協(xié)議指定套接字使用的特定協(xié)議。通常只存在一個協(xié)議來支持
給定協(xié)議族中的特定套接字類型,在這種情況下,可以將protocol指定為0。然而,這是可能的
可能存在許多協(xié)議,在這種情況下,必須以這種方式指定特定的協(xié)議。
第二步,設(shè)置套接字選項(xiàng)
getsockopt, setsockopt 函數(shù):?get and set options on sockets
int val = 1;
setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &val, sizeof(val));
-
fd
:上文拿到的文件描述符。 -
SOL_SOCKET
:這是一個預(yù)定義的常量,表示我們要修改的是套接字級別的選項(xiàng)。 -
SO_REUSEADDR
:這是一個預(yù)定義的常量,表示我們要修改的是“地址重用”選項(xiàng)。當(dāng)一個套接字關(guān)閉后,操作系統(tǒng)通常會保留該套接字的地址,以便將來重新使用。通過將此選項(xiàng)設(shè)置為1,我們可以告訴操作系統(tǒng)在套接字關(guān)閉后立即釋放其地址,而不是等待一段時間。 -
&val
:這是一個指向整數(shù)變量val
的指針,該變量的值被設(shè)置為1。這是因?yàn)?code>setsockopt()函數(shù)的最后一個參數(shù)需要一個指向整數(shù)的指針。 -
sizeof(val)
:這是val
變量的大?。ㄒ宰止?jié)為單位)。在這種情況下,它是4,因?yàn)?code>int通常占用4個字節(jié)。
第三步進(jìn)行bind和listen
// bind
struct sockaddr_in addr = {};
addr.sin_family = AF_INET;
addr.sin_port = ntohs(1234);
addr.sin_addr.s_addr = ntohl(0); // wildcard address 0.0.0.0
int rv = bind(fd, (const sockaddr *)&addr, sizeof(addr));
if (rv) {
die("bind()");
}
// listen
rv = listen(fd, SOMAXCONN);
if (rv) {
die("listen()");
}
這個sockaddr_in結(jié)構(gòu)體:
linux上是這個:#include <sys/socket.h>
windows上是這個:#include <winsock.h>
struct sockaddr_in {
short sin_family;
u_short sin_port;
struct in_addr sin_addr;
char sin_zero[8];
};
ntohs()函數(shù)的作用是:將無符號短整數(shù)netshort從網(wǎng)絡(luò)字節(jié)順序轉(zhuǎn)換為主機(jī)字節(jié)順序。
說白了,addr
結(jié)構(gòu)體的sin_port
字段,表示要綁定的端口號。ntohs
函數(shù)將主機(jī)字節(jié)序的端口號轉(zhuǎn)換為網(wǎng)絡(luò)字節(jié)序。在這里,端口號被設(shè)置為1234。
ntohl
函數(shù)將主機(jī)字節(jié)序的IP地址轉(zhuǎn)換為網(wǎng)絡(luò)字節(jié)序。在這里,IP地址被設(shè)置為通配符地址0.0.0.0,表示允許任何IP地址連接到該套接字。
int rv = bind(fd, (const sockaddr *)&addr, sizeof(addr));
這行代碼調(diào)用了bind
函數(shù),將套接字fd
綁定到指定的網(wǎng)絡(luò)地址和端口上。bind
函數(shù)的第一個參數(shù)是要綁定的套接字的文件描述符,第二個參數(shù)是指向網(wǎng)絡(luò)地址結(jié)構(gòu)的指針,第三個參數(shù)是網(wǎng)絡(luò)地址結(jié)構(gòu)的大小。在這里,fd
是要綁定的套接字的文件描述符,&addr
是指向網(wǎng)絡(luò)地址結(jié)構(gòu)的指針,sizeof(addr)
是網(wǎng)絡(luò)地址結(jié)構(gòu)的大小。
最后,bind
函數(shù)返回一個整數(shù)值,表示操作的結(jié)果。文章來源:http://www.zghlxwxcb.cn/news/detail-794213.html
listen(fd, SOMAXCONN);
是調(diào)用 listen()
函數(shù),該函數(shù)用于使套接字進(jìn)入監(jiān)聽狀態(tài),等待客戶端的連接請求。其中,fd
是之前創(chuàng)建的套接字文件描述符,SOMAXCONN
是一個系統(tǒng)定義的最大連接數(shù)。文章來源地址http://www.zghlxwxcb.cn/news/detail-794213.html
第四步,循環(huán)接受并處理請求
while (true) {
// accept
struct sockaddr_in client_addr = {};
socklen_t socklen = sizeof(client_addr);
int connfd = accept(fd, (struct sockaddr *)&client_addr, &socklen);
if (connfd < 0) {
continue; // error
}
while (true) {
// here the server only serves one client connection at once
int32_t err = one_request(connfd);
if (err) {
break;
}
}
close(connfd);
}
到了這里,關(guān)于Redis的實(shí)現(xiàn)一:c、c++的網(wǎng)絡(luò)通信編程技術(shù),先實(shí)現(xiàn)server和client的通信的文章就介紹完了。如果您還想了解更多內(nèi)容,請?jiān)谟疑辖撬阉鱐OY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!