单线程的server段的主要逻辑如下:
#include <sys/socket.h>
#include <netinet/in.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <iostream>
#define PORT 8888
int main() {
using namespace std;
// 1. 创建socket
// 参数:
// - [in] int af: 地址系列规范, 常用AF_INET(IPV4)、AF_INET6(IPV6)、AF_BTH(蓝牙)等
// - [in] int type: socket类型规范, 常用SOCK_STREAM(流式传输)、SOCK_DGRAM(数据报传输)等
// - [in] int protocol: socket要使用的协议,常用0(默认)、IPPROTO_ICMP(ICMP)、IPPROTO_TCP(TCP)、IPPROTO_UDP(UDP)等
int socket_fd = socket(AF_INET, SOCK_STREAM, 0);
int opt = 1;
setsockopt(socket_fd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(yes));
struct sockaddr_in server_addr;
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
server_addr.sin_port = htons(PORT);
// 将socket和
bind(socket_fd, (struct sockaddr*)&server_addr, sizeof(server_addr));
// 设置为监听状态, 该步不阻塞
listen(socket_fd, 10);
while (1) {
// 等待客户端连接
cout << "Waiting for connection." << endl;
int conn_fd = accept(socket_fd, NULL, NULL);
cout << "Connected." << endl;
const string msg = "hello, world!";
// 发送数据
send(conn_fd, msg.c_str(), msg.length() + 1, 0);
shutdown(conn_fd, SHUT_WR); // 关闭发送
close(conn_fd); // 关闭连接
}
close(socket_fd);
return 0;
}
select函数是跨平台的函数,已验证支持的平台有:
在Linux中, select
函数的定义如下:
int select(int nfds, fd_set *_Nullable restrict readfds,
fd_set *_Nullable restrict writefds,
fd_set *_Nullable restrict exceptfds,
struct timeval *_Nullable restrict timeout);
其中,参数:
nfds
:需要使用 select
委托内核查询的三个集合中的最大fd号+1。或者设置为1024(但是效率略有降低)。__readfds
:大小默认为1024bit,即1024个标志位。
__writefds
:大小默认为1024bit,即1024个标志位。
__exceptfds
:大小默认为1024bit,即1024个标志位。
__timeout
:select函数阻塞的时长,需要注意该结构体有两个成员,分别为秒级成员和微秒级成员。注意两个成员都要初始化。其中:
fd_set
与内核之间的交互如下图组所示:fd_set
提供了如下的操作API:// 清空set集合中所有标志位
void FD_ZERO(fd_set *set);
// 将fd对应的标识符置1
void FD_SET(int fd, fd_set *set);
// 将fd对应的标识符清空
void FD_CLR(int fd, fd_set *set);
// 判断fd对应标识符是否被置1
int FD_ISSET(int fd, fd_set *set);
在Windows中的定义如下:
int WSAAPI select(
[in] int nfds,
[in, out] fd_set *readfds,
[in, out] fd_set *writefds,
[in, out] fd_set *exceptfds,
[in] const timeval *timeout
);
其中,与BSD socket中不一致的参数为:
nfds
参数无效,随意填写即可。通常使用的函数为select和epoll。
poll函数底层使用的数据结构为线性表
epoll函数底层使用的数据结构为红黑树
epoll_create
会将事件注册到红黑树下