首页 > 系统相关 >【Linux网络编程】基于 EPOLL 的 SOCKET 通信

【Linux网络编程】基于 EPOLL 的 SOCKET 通信

时间:2024-08-28 20:03:46浏览次数:9  
标签:return SOCKET EPOLL int fd Linux include Epoller events

【Linux网络编程】基于 EPOLL 的 SOCKET 通信

epoller.h

#ifndef EPOLLER_H
#define EPOLLER_H

#include <sys/epoll.h>
#include <fcntl.h>
#include <unistd.h>
#include <assert.h>
#include <errno.h>
#include <vector>

class Epoller {
public:
    explicit Epoller(int maxEvent = 1024);
    ~Epoller();
    bool AddFd(int fd, uint32_t events);
    bool ModFd(int fd, uint32_t events);
    bool DelFd(int fd);
    int Wait(int timeoutMs = -1);
    int GetEventFd(size_t i) const;
    uint32_t GetEvents(size_t i) const;

private:
    int epollFd_;  // epoll_create()创建一个epoll对象,返回值为epollFd
    std::vector<struct epoll_event> events_;  // 检测到的事件的集合
};

#endif  // EPOLLER_H

epoller.cpp

#include "epoller.h"

// 创建epoll对象 epoll_create(512)
Epoller::Epoller(int maxEvents) : epollFd_(epoll_create(512)), events_(maxEvents) {
    assert(epollFd_ >= 0 && events_.size() > 0);
}

Epoller::~Epoller() {
    close(epollFd_);
}

bool Epoller::AddFd(int fd, uint32_t events) {
    if (fd < 0) return false;
    epoll_event ev = {0};
    ev.data.fd = fd;
    ev.events = events;
    return 0 == epoll_ctl(epollFd_, EPOLL_CTL_ADD, fd, &ev);
}

bool Epoller::ModFd(int fd, uint32_t events) {
    if (fd < 0) return false;
    epoll_event ev = {0};
    ev.data.fd = fd;
    ev.events = events;
    return 0 == epoll_ctl(epollFd_, EPOLL_CTL_MOD, fd, &ev);
}

bool Epoller::DelFd(int fd) {
    if (fd < 0) return 0;
    epoll_event ev = {0};
    return 0 == epoll_ctl(epollFd_, EPOLL_CTL_DEL, fd, &ev);
}

int Epoller::Wait(int timeoutMs) {
    return epoll_wait(epollFd_, &events_[0], static_cast<int>(events_.size()), timeoutMs);
}

int Epoller::GetEventFd(size_t i) const {
    assert(i < events_.size() && i >= 0);
    return events_[i].data.fd;
}

uint32_t Epoller::GetEvents(size_t i) const {
    assert(i < events_.size() && i >= 0);
    return events_[i].events;
}

webserver.h

#ifndef WEBSERVER_H
#define WEBSERVER_H

#include <stdio.h>
#include <ctype.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <memory>
#include "epoller.h"

class WebServer {
public:
    WebServer(int port, int timeoutMs);
    ~WebServer() {}
    // void Start();
private:
    bool InitSocket_();
    int timeoutMs_;
    int port_;
    int listenFd_;
    std::unique_ptr<Epoller> epoller_;
};

#endif  // WEBSERVER_H

webserver.cpp

#include "webserver.h"

WebServer::WebServer(int port, int timeoutMs) : port_(port), timeoutMs_(timeoutMs),
                    epoller_(new Epoller()) {
    InitSocket_();
}

bool WebServer::InitSocket_() {
    // 创建socket
    listenFd_= socket(AF_INET, SOCK_STREAM, 0);
    if (listenFd_ == -1) {
        perror("socket");
        return -1;
    }

    // 设置端口复用
    int optval = 1;
    setsockopt(listenFd_, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval));

    struct sockaddr_in saddr;
    saddr.sin_family = AF_INET;
    saddr.sin_addr.s_addr = INADDR_ANY;
    saddr.sin_port = htons(port_);
    // 绑定
    int ret = bind(listenFd_, (struct sockaddr*)&saddr, sizeof(saddr));
    if (ret == -1) {
        perror("bind");
        return -1;
    }

    // 监听
    ret == listen(listenFd_, 128);
    if (ret == -1) {
        perror("listen");
        return -1;
    }

    // 将listenFd_注册至epoll事件表中
    epoller_->AddFd(listenFd_, EPOLLIN);

    while (1) {
        int num = epoller_->Wait(timeoutMs_);
        if (num == -1) {
            perror("epoll_wait");
            exit(-1);
        }
        printf("num = %d\n", num);
        for (int i = 0; i < num; ++i) {
            int curfd = epoller_->GetEventFd(i);
            if (curfd == listenFd_) {  // 有客户端连接进来
                struct sockaddr_in cliaddr;
                socklen_t len = sizeof(cliaddr);
                int cfd = accept(listenFd_, (struct sockaddr*)&cliaddr, &len);

                // 获取客户端信息
                char cliIp[16];
                inet_ntop(AF_INET, &cliaddr.sin_addr.s_addr, cliIp, sizeof(cliIp));
                unsigned short cliPort = ntohs(cliaddr.sin_port);
                // 输出客户端的信息
                printf("client's ip is %s, and port is %d\n", cliIp, cliPort );

                int flag = fcntl(cfd, F_GETFL);
                flag |= O_NONBLOCK;  // 设置文件描述符非阻塞
                fcntl(cfd, F_SETFL, flag);

                epoller_->AddFd(cfd, EPOLLIN | EPOLLET);  // cfd注册至内核事件表中,并设置ET模式
            } else {  // 有客户端发送数据
                if (epoller_->GetEvents(i) & EPOLLOUT) {
                    continue;
                }
                char buf[5];
                int len = 0;
                while ((len = read(curfd, buf, sizeof(buf))) > 0) {
                    write(STDOUT_FILENO, buf, len);
                    write(curfd, buf, len);
                }
                if (len == 0) {
                    printf("client close...\n");
                } else if (len == -1) {
                    if (errno == EAGAIN) {
                        printf("read over ...\n");
                    } else {
                        perror("read\n");
                        exit(-1);
                    }
                }
            }
        }
    }
    close(listenFd_);
    return 0;
}

main.cpp

#include "../server/webserver.h"

int main() {
    WebServer webserver(1316, -1);  // 设置端口, 超时时间
    // webserver.Start();
    return 0;
}

client.cpp

#include <stdio.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>

int main() {
    int fd = socket(AF_INET, SOCK_STREAM, 0);
    if (fd == -1) {
        perror("socket");
        return -1;
    }

    struct sockaddr_in seraddr;
    inet_pton(AF_INET, "127.0.0.1", &seraddr.sin_addr.s_addr);
    seraddr.sin_family = AF_INET;
    seraddr.sin_port = htons(1316);
    // 连接服务器
    int ret = connect(fd, (struct sockaddr*)&seraddr, sizeof(seraddr));
    if (ret == -1) {
        perror("connect");
        return -1;
    }

    char sendBuf[1024] = {0};
    while (1) {
        fgets(sendBuf, sizeof(sendBuf), stdin);
        write(fd, sendBuf, strlen(sendBuf) + 1);
        int len = read(fd, sendBuf, sizeof(sendBuf));
        if (len == -1) {
            perror("read");
            return -1;
        } else if (len > 0) {
            printf("read buf = %s\n", sendBuf);
        } else {
            printf("server disconnect...\n");
            break;
        }
    }

    close(fd);
    return 0;
}

标签:return,SOCKET,EPOLL,int,fd,Linux,include,Epoller,events
From: https://www.cnblogs.com/yangxuanzhi/p/18385476

相关文章

  • 重头开始嵌入式第二十九天(Linux系统编程 网络通信 tcp)
    目录1.常见网络模型1.bs2.p2p3.cs2.网络编程之TCP(传输控制协议)1.TCP模型2.服务器端:1.socket();2、bind();3、listen();4、accept();5、接受函数:/发送函数:6、close()  ===>关闭指定的套接字id;3.客户端:1.connect();2、send()3、客户端信息获取4、客户端的信息bin......
  • Java 入门指南:Java Socket 网络通信编程
    SocketSocket(套接字)是用于网络通信的编程接口、网络通信的基础,通过它可以实现不同计算机之间的数据传输,应用程序可以通过它发送或接收数据;就像操作文件那样可以打开、读写和关闭。它提供了一种机制,使得计算机之间可以进行数据的发送和接收。套接字允许应用程序将I/O应用......
  • socket,TCP/IP的理解
    socket,TCP/IP的理解TCP/IP要想理解socket首先得熟悉一下TCP/IP协议族, TCP/IP(TransmissionControlProtocol/InternetProtocol)即传输控制协议/网间协议,定义了主机如何连入因特网及数据如何再它们之间传输的标准,从字面意思来看TCP/IP是TCP和IP协议的合称,但实际上TCP/IP协议......
  • SOCKET和MODBUS的区别
    SOCKET和MODBUS的区别  Socket和Modbus是两种常用的网络协议,它们在网络通信中发挥着重要作用。这两种协议有着许多共同的特征,但也存在一些显著差异。本文将介绍Socket和Modbus协议的基本概念,以及它们之间的区别。Socket协议  Socket是一种用于实现网络通信的应用层协议......
  • 【Linux网络编程】I/O 多路复用技术
    【Linux网络编程】I/O多路复用技术什么是I/O多路复用?为什么需要I/O多路复用最简单的socket网络模型,就是单线程模型,一个同时进行监听、处理,然而,单线程模型同时只能服务一个客户端,当线程发生阻塞的时候,其他客户端只能排队等待,甚至连接失败。为了能够同时服务更多的客户端,......
  • Linux基础命令
    Linux的目录结构/,根目录是最顶级的目录了Linux只有一个顶级目录:/路径描述的层次关系同样适用/来表示/home/itheima/a.txt,表示根目录下的home文件夹内有itheima文件夹,内有a.txtls命令功能:列出文件夹信息语法:ls[-l-h-a][参数]参数:被查看的文件夹,不提供参数,表示查看......
  • 【Linux网络编程】字节序
    【Linux网络编程】字节序字节序字节序就是字节在内存中存储的顺序,如32位整数0x01234567,在内存中存储时,有如下两种顺序:大端序将数值的高位存储在低位地址中,小端序则相反。网络字节序网络中传输数据均采用大端序。Linux字节序转换函数在#include<netinet/in.h>中提供了4......
  • 【Linux网络编程】Socket Api函数
    【Linux网络编程】SocketApi函数TCP/IP协议族TCP/IP协议族有sockaddr_in和sockaddr_in6两个专用的socket地址结构体,它们分别用于IPv4和IPv6,在此只将IPv4,如下为structsockaddr_in:structsockaddr_in{sa_family_tsin_family;//地址族:AF_INETu_int......
  • 协议汇总 TCP、UDP、Http、Socket、Web Scoket、Web Service、WCF、API
    TCP:(1)位于OSI传输层,基于soap(信封)协议;(2)数据格式是xml、Json;(3)是面向连接的,需要先建立连接;(4)TCP协议是一个可靠的传输协议,它可以保证传输的一个正确性,保证我们的不丢包不重复,而且数据是按顺序到达的,保证不丢包(握手需要三次,挥手却要四次);(5)典型的TCP/IP之上的协议有FTP、......
  • Linux 软件目录迁移
    背景:很多软件默认安装到了/root盘下的/var/lib/目录下,导致/root盘很快被占满,如docker。为了释放/root盘,让系统能正常工作,就需要把这些特别大磁盘占用量的目录迁移到挂载了其他磁盘的目录,比如:/home,下面是具体步骤:1. 停止Docker服务systemctlstopdocker 2. 复制 /var/l......