- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我想做一个 epoll 服务器。但是我的服务器代码丢失了一些连接。
我的客户端生成了 100 个线程,每个线程都发送相同的消息。然后我的服务器应该接收并用计数打印它们。
但是服务器似乎失去了连接,我不知道为什么。
我将 EPOLL_SIZE
从 50 更改为 200,并将 listen()
的积压参数从 5 更改为 1000。但是它们没有用。
1.服务器:
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <sys/epoll.h>
#include <memory>
#include <array>
#define BUF_SIZE 100
#define EPOLL_SIZE 200
void error_handling(const char *buf);
int main(int argc, char *argv[])
{
// Step 1. Initialization
int server_socket, client_socket;
struct sockaddr_in client_addr;
socklen_t addr_size;
int str_len, i;
char buf[BUF_SIZE];
int epfd, event_cnt;
if (argc != 2) {
printf("Usage : %s <port>\n", argv[0]);
exit(1);
}
// Step 2. Creating a socket
server_socket = socket(PF_INET, SOCK_STREAM, 0);
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(atoi(argv[1]));
// Step 3. Binding the server address onto the socket created just right before.
if (bind(server_socket, (struct sockaddr*) &server_addr, sizeof(server_addr)) == -1)
error_handling("bind() error");
// Step 4. Start to listen to the socket.
if (listen(server_socket, 1000) == -1)
error_handling("listen() error");
// Step 5. Create an event poll instance.
epfd = epoll_create(EPOLL_SIZE);
auto epoll_events = (struct epoll_event*) malloc(sizeof(struct epoll_event) * EPOLL_SIZE);
struct epoll_event event;
event.events = EPOLLIN;
event.data.fd = server_socket;
// Step 6. Adding the server socket file descriptor to the event poll's control.
epoll_ctl(epfd, EPOLL_CTL_ADD, server_socket, &event);
int recv_cnt = 0;
while(true)
{
// Step 7. Wait until some event happens
event_cnt = epoll_wait(epfd, epoll_events, EPOLL_SIZE, -1);
if (event_cnt == -1)
{
puts("epoll_wait() error");
break;
}
for (i = 0; i < event_cnt; i++)
{
if (epoll_events[i].data.fd == server_socket)
{
addr_size = sizeof(client_addr);
client_socket = accept(server_socket, (struct sockaddr*)&client_addr, &addr_size);
event.events = EPOLLIN;
event.data.fd = client_socket;
epoll_ctl(epfd, EPOLL_CTL_ADD, client_socket, &event);
//printf("Connected client: %d\n", client_socket);
}
else // client socket?
{
str_len = read(epoll_events[i].data.fd, buf, BUF_SIZE);
if (str_len == 0) // close request!
{
epoll_ctl(epfd, EPOLL_CTL_DEL, epoll_events[i].data.fd, nullptr);
close(epoll_events[i].data.fd);
printf("%d: %s\n", ++recv_cnt, buf);
//printf("closed client: %d \n", epoll_events[i].data.fd);
}
else
{
write(epoll_events[i].data.fd, buf, str_len); // echo!
}
} // end of else()
} // end of for()
} // end of while()
close(server_socket);
close(epfd);
free(epoll_events);
return EXIT_SUCCESS;
}
void error_handling(const char *buf)
{
fputs(buf, stderr);
fputc('\n', stderr);
exit(EXIT_FAILURE);
}
2.客户端:
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <thread>
#include <vector>
#include <algorithm>
#include <mutex>
#define BUF_SIZE 100
#define EPOLL_SIZE 50
void error_handling(const char *buf);
int main(int argc, char *argv[])
{
// Step 1. Initialization
int socketfd;
if (argc != 3) {
printf("Usage : %s <ip address> <port>\n", argv[0], argv[1]);
exit(EXIT_FAILURE);
}
std::vector<std::thread> cli_threads;
std::mutex wlock;
for (int i = 0; i < 100; i++) {
cli_threads.push_back(std::thread([&](const char* szIpAddr, const char* szPort) {
// Step 2. Creating a socket
socketfd = socket(PF_INET, SOCK_STREAM, 0);
struct sockaddr_in server_addr;
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_addr.s_addr = inet_addr(szIpAddr);
server_addr.sin_port = htons(atoi(szPort));
// Step 3. Connecting to the server
if(connect(socketfd, (struct sockaddr*)&server_addr, sizeof(server_addr)) == -1)
error_handling("connect() error");
// Step 4. Writing message to the server
std::string msg = "Hey I'm a client!";
wlock.lock();
auto str_len = write(socketfd, msg.c_str(), msg.size()+1);
wlock.unlock();
close(socketfd);
}, argv[1], argv[2]));
}
std::for_each(cli_threads.begin(), cli_threads.end(),
[](std::thread &t)
{
t.join();
}
);
return EXIT_SUCCESS;
}
void error_handling(const char *buf)
{
fputs(buf, stderr);
fputc('\n', stderr);
exit(EXIT_FAILURE);
}
希望像...
1: Hey I'm a client!
...
100: Hey I'm a client!
但是,结果各不相同,比如...
1: Hey I'm a client!
...
n: Hey I'm a client!
n 小于 100。
最佳答案
由于通过引用线程传递 socketfd
,您有未定义的行为 - std::thread([&](...
。套接字描述符的一个实例正在由所有线程同时修改 - 它导致了问题。每个线程都应该存储自己的描述符。
关于c++ - 我的 epoll 服务器丢失了一些连接。为什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58299409/
我正在使用 epoll 编写媒体服务器。 fds 都设置为非阻塞,我正在使用边缘触发事件。我知道对于 EPOLLIN,我需要循环读取 fd,直到返回 EAGAIN。但是写作呢? 当我想写入时,我将数据
很难说出这里问的是什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或言辞激烈,无法以目前的形式合理回答。如需帮助澄清此问题以便可以重新打开,visit the help center . 8年前关闭
简短的问题,但对我来说很难理解。 为什么 ePoll 的扩展性比 Poll 更好? 最佳答案 poll 系统调用每次都需要将您的文件描述符列表复制到内核。这只会发生一次 epoll_ctl ,但不是每
情况: 多个进程共享同一个文件描述符表。 每个进程监听自己的 epoll 实例。 所有套接字和涉及它们的调用都是非阻塞的。 在这些进程中,只有进程A将监听套接字添加到它的epoll实例。 进程 A 知
我有几个线程,其中一个在循环中调用 epoll_wait,其他线程可以打开需要由第一个线程 epoll 的连接。是否可以只使用 epoll_ctl 添加新套接字,而另一个线程在 epoll_wait
众所周知,我们可以将poll/select与Netmap结合使用:http://info.iet.unipi.it/~luigi/netmap/ 是否可以使用epoll以及将epoll与Netmap一
我正在使用 epoll 和大量 fd 编写应用程序。 为了改进 fd 对应用程序相关 DB 的搜索,我想传递给 epoll 应用程序信息,例如 DB 数组中的索引。 我想过使用data->ptr(ep
我有一个多线程 epoll 服务器。我创建了一个 epoll fd,然后我将让 X 个线程休眠,等待 epoll_wait() 来自 SAME epoll fd 的任何事件。 现在我的问题是:如何唤醒
我有两个线程在 epoll 上运行。一个线程尝试与服务器建立 TCP 连接,使用 EPOLL_CTL_ADD 选项将套接字添加到 epoll-fd。 另一个线程负责等待添加到 epoll-fd 的 S
我正在使用 Cassandra,在启动期间,Netty 会打印一 strip 有堆栈跟踪的警告: Found Netty's native epoll transport in the classpa
epoll_wait、select 和 poll 函数都提供超时。然而使用epoll,其分辨率可达1ms。 Select 和 ppoll 是唯一提供亚毫秒级超时的。 这意味着最好以 1 毫秒的间隔做其
我正在使用 epoll 进行网络编程。我想知道检测用户断开连接的最佳方法。现在,我正在使用带有超时的选择来查看接收缓冲区中是否有信号并且在一定时间内没有信号然后关闭套接字连接。 我认为可能还有其他更适
epoll 是 Linux 系统中常用的多路复用 I/O 组件,一般用于监听 socket 是否能够进行 I/O 操作。那么,epoll 能监听普通文件吗? 我们先通过下面的例子来验证一下,e
我正在使用 epoll 将大消息写入使用 HTTP 协议(protocol)的服务器。 fds 都设置为非阻塞,我正在使用边缘触发事件。我知道对于 EPOLLIN,我需要循环读取 fd,直到返回 EA
我使用python的socket和epoll来制作一个Web服务器。我的操作系统是Linux CentOS 6,我的python版本是python 2.7.8。我的源代码是: # -*-coding:
我的 TCP 服务器中有一个 epoll 事件循环来处理客户端连接并从客户端读取数据。 while(1) { int n, i; n = epoll_wait(efd, events,
我有一个项目,我正在从普通的 select() 转换,但由于我运行的测试有超过 1000 个连接,我决定将其更改为 epoll。 我的代码可以正常工作,但遇到了一个问题,我正在尝试找出解决问题的最佳方
我是 linux 系统编程的新手,所以请多多关照。我必须通过 RS-422 在 8N1 模式下以波特率 921600 从串行端口 (/dev/ttyX) 读取数据,没有奇偶校验。stty 输出信号表示
我从here得到了基础知识关于 epoll。我知道epoll可以监控多个FD并处理。 我的问题是:重事件是否会阻塞服务器,所以我必须使用多线程? 例如,服务器的epoll正在监控2个套接字A和B。现在
我正在使用 epoll 进行网络编程。我想知道如何处理客户端断开连接。我阅读了 api 文档,epoll_wait 有一个超时参数。但是,如果 epoll_wait 函数或正在连接的客户端超时,我会感
我是一名优秀的程序员,十分优秀!