C语言如何实现异步Socket
创作时间:
作者:
@小白创作中心
C语言如何实现异步Socket
引用
1
来源
1.
https://docs.pingcode.com/baike/1531899
异步Socket编程是网络编程中的一个重要概念,它允许程序在等待I/O操作完成时继续执行其他任务,从而提高程序的并发性和响应性能。本文将详细介绍如何在C语言中实现异步Socket,重点讲解多路复用的方法。
一、非阻塞I/O
非阻塞I/O是实现异步Socket的基础。通过设置Socket为非阻塞模式,I/O操作将在数据尚未准备好时立即返回,而不是阻塞等待。
1. 设置非阻塞模式
使用fcntl函数设置Socket为非阻塞模式:
#include <fcntl.h>
#include <unistd.h>
int set_nonblocking(int sockfd) {
int flags = fcntl(sockfd, F_GETFL, 0);
if (flags == -1) {
return -1;
}
return fcntl(sockfd, F_SETFL, flags | O_NONBLOCK);
}
2. 非阻塞读写
在非阻塞模式下,读写操作可能返回EAGAIN或EWOULDBLOCK,表示数据尚未准备好。需要处理这些错误并继续操作。
ssize_t nonblocking_read(int sockfd, void *buf, size_t len) {
ssize_t n;
while ((n = read(sockfd, buf, len)) == -1 && errno == EAGAIN) {
// 数据尚未准备好,继续尝试读取
}
return n;
}
ssize_t nonblocking_write(int sockfd, const void *buf, size_t len) {
ssize_t n;
while ((n = write(sockfd, buf, len)) == -1 && errno == EAGAIN) {
// 数据尚未准备好,继续尝试写入
}
return n;
}
二、使用多线程
通过多线程可以在每个线程中处理一个Socket连接,从而实现异步I/O。然而,多线程编程复杂且开销大,不适合大量并发连接的场景。
1. 创建线程
使用pthread库创建线程来处理每个Socket连接:
#include <pthread.h>
void *handle_connection(void *arg) {
int sockfd = *(int *)arg;
char buf[1024];
ssize_t n;
// 处理连接
while ((n = read(sockfd, buf, sizeof(buf))) > 0) {
// 处理数据
write(sockfd, buf, n);
}
close(sockfd);
return NULL;
}
void start_server(int listen_sock) {
pthread_t tid;
int conn_sock;
while ((conn_sock = accept(listen_sock, NULL, NULL)) != -1) {
pthread_create(&tid, NULL, handle_connection, &conn_sock);
pthread_detach(tid);
}
}
三、使用多路复用
多路复用是处理大量并发连接的高效方法。常用的多路复用机制有select、poll和epoll。下面将详细介绍如何使用select和epoll实现异步Socket。
1. 使用select
select是一个标准的多路复用函数,适用于小规模并发连接。
#include <sys/select.h>
#include <unistd.h>
void start_server_select(int listen_sock) {
fd_set readfds;
int maxfd, conn_sock;
struct sockaddr_in client_addr;
socklen_t client_len = sizeof(client_addr);
char buf[1024];
ssize_t n;
// 初始化
FD_ZERO(&readfds);
FD_SET(listen_sock, &readfds);
maxfd = listen_sock;
while (1) {
fd_set tempfds = readfds;
if (select(maxfd + 1, &tempfds, NULL, NULL, NULL) == -1) {
perror("select");
break;
}
// 处理新连接
if (FD_ISSET(listen_sock, &tempfds)) {
conn_sock = accept(listen_sock, (struct sockaddr *)&client_addr, &client_len);
if (conn_sock == -1) {
perror("accept");
continue;
}
set_nonblocking(conn_sock);
FD_SET(conn_sock, &readfds);
if (conn_sock > maxfd) {
maxfd = conn_sock;
}
}
// 处理现有连接
for (int i = 0; i <= maxfd; ++i) {
if (FD_ISSET(i, &tempfds)) {
n = read(i, buf, sizeof(buf));
if (n <= 0) {
// 连接关闭或出错
close(i);
FD_CLR(i, &readfds);
} else {
// 处理数据
write(i, buf, n);
}
}
}
}
}
2. 使用epoll
epoll是Linux特有的多路复用机制,适用于大规模并发连接。
#include <sys/epoll.h>
#include <unistd.h>
void start_server_epoll(int listen_sock) {
int epfd, conn_sock;
struct epoll_event ev, events[1024];
struct sockaddr_in client_addr;
socklen_t client_len = sizeof(client_addr);
char buf[1024];
ssize_t n;
epfd = epoll_create1(0);
if (epfd == -1) {
perror("epoll_create1");
return;
}
// 注册监听Socket
ev.events = EPOLLIN;
ev.data.fd = listen_sock;
if (epoll_ctl(epfd, EPOLL_CTL_ADD, listen_sock, &ev) == -1) {
perror("epoll_ctl");
close(epfd);
return;
}
while (1) {
int nfds = epoll_wait(epfd, events, 1024, -1);
if (nfds == -1) {
perror("epoll_wait");
break;
}
for (int i = 0; i < nfds; ++i) {
if (events[i].data.fd == listen_sock) {
// 处理新连接
conn_sock = accept(listen_sock, (struct sockaddr *)&client_addr, &client_len);
if (conn_sock == -1) {
perror("accept");
continue;
}
set_nonblocking(conn_sock);
ev.events = EPOLLIN | EPOLLET;
ev.data.fd = conn_sock;
if (epoll_ctl(epfd, EPOLL_CTL_ADD, conn_sock, &ev) == -1) {
perror("epoll_ctl");
close(conn_sock);
continue;
}
} else {
// 处理现有连接
n = read(events[i].data.fd, buf, sizeof(buf));
if (n <= 0) {
// 连接关闭或出错
close(events[i].data.fd);
} else {
// 处理数据
write(events[i].data.fd, buf, n);
}
}
}
}
close(epfd);
}
四、总结
通过以上介绍,可以发现实现异步Socket的关键在于非阻塞I/O、使用多线程、使用多路复用。其中,多路复用(如select和epoll)是处理大量并发连接的高效方法。选择具体方法时,需要根据应用场景和性能需求进行权衡。
热门推荐
每天认识一味中草药——百合
NBA球队总市值排名及其经济影响分析
中国古代严苛对待女性的刑法探析
【西班牙美食】6个经典传统必吃美食
成语“格格不入”:贬义之中的启示
大学绩点评估标准和方法详解
轻松应对特殊病种报销:流程、材料及注意事项
企业应对职场“内卷”:强制下班,以人为本的新策略?
年金保险的优点和缺点有哪些方面
飞牛NAS配置小雅资源教程:可行但需谨慎
农村宅基地的变动对农民有何影响?
奥迪氏括约肌功能障碍:诊断、治疗与生活调适指南
宋朝时期的中国经济对世界产生重要影响
营养均衡,助力运动表现:运动达人的饮食哲学
从模糊到超清,提升电脑显示效果的方法(调整电脑显示设置)
糖尿病人为什么要检测糖化血红蛋白
养娃成本分析:家庭财务规划的重要性
杭州银行副行长陈岚去年薪酬251.23万 该行上半年共计被罚310万
股息政策是什么
嫦娥奔月:工笔画与水墨写意的结合~千古传奇的浪漫与启示
空腹有氧运动全攻略:5种运动项目及注意事项
二十四节气陈列的花卉有哪些?不同节气应选择哪些花卉?
小学数学概念教学:从创设情境到应用实践
五十岁之后,夫妻关系再好,也要懂得“五五定律”
深圳办理回乡证需要什么资料和手续?
一文看懂香港回乡证的历史演变
如何制定股市投资策略?这些策略有哪些成功案例和失败教训?
色彩缤纷又营养丰富!教你做五彩鸡蛋蔬菜卷!
刘邦的抉择:废长立幼背后的权力逻辑
汉朝的终结与更替:一个时代的消逝与原因探究