IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> muduo中封装fd的Socket类和分发Channel的Acceptor类 -> 正文阅读

[C++知识库]muduo中封装fd的Socket类和分发Channel的Acceptor类

一、Socket类

用于封装fd、bind、listen、accept等操作

1. Socket.h

#include "noncopyable.h"

struct tcp_info;

class InetAddress;

class Socket : noncopyable{
    public:
        explicit Socket(int sockfd)
            : sockfd_(sockfd)
        {}
        ~Socket();

        int fd() const { return sockfd_; }
        void bindAddress(const InetAddress& localaddr);   // 将ip port绑定到listenfd
        void listen();
        int accept(InetAddress* peeraddr);                // 调用accept获取和客户通信的fd
        
        void shutdownWrite();                             // tcp socket是全双工通信,这是关闭监听套接字sockfd_写端

        void setTcpNoDelay(bool on);                      // 数据不进行tcp缓冲
        void setReuseAddr(bool on);
        void setReusePort(bool on);
        void setKeepAlive(bool on);

    private:
        const int sockfd_;  // 这就是服务器用于监听客户端的listenfd
};

2. Socket.cc

shutdown函数
在这里插入图片描述

#include "Socket.h"
#include "Logger.h"
#include "InetAddress.h"

#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/tcp.h>
#include <netinet/in.h>

Socket::~Socket(){
    ::close(sockfd_);
}

// 将服务器本地ip port绑定到listenfd
void Socket::bindAddress(const InetAddress& localaddr){
    if(0 > ::bind(sockfd_, (sockaddr*)localaddr.getSockAddr(), sizeof(sockaddr_in))){
        LOG_FATAL("bind sockfd:%d fail \n", sockfd_);
    }
}

void Socket::listen(){
    if(0 > ::listen(sockfd_, 1024)){
        LOG_FATAL("listen sockfd:%d fail \n", sockfd_);
    }
}

// 调用accept获取和客户通信的fd
// 上层调用accept,传入InetAddress对象peeraddr的地址,并且把客户的addr写入输出参数peeraddr
int Socket::accept(InetAddress* peeraddr){
    sockaddr_in addr;
    socklen_t len;

    ::memset(&addr, 0, sizeof(addr));
    int connfd = ::accept(sockfd_, (sockaddr*)&addr, &len);
    if(connfd >= 0){
        peeraddr->setSockAddr(addr);
    }
    return connfd;
}

// tcp socket是全双工通信,这是关闭监听套接字sockfd_写端
void Socket::shutdownWrite(){
    // 关闭监听套接字的写端
    if(::shutdown(sockfd_, SHUT_WR)){
        LOG_ERROR("Socket::shutdownWrite error");
    }
}            

// 数据不进行tcp缓冲
void Socket::setTcpNoDelay(bool on){
    int optval = on ? 1 : 0;
    // TCP_NODELAY禁用Nagle算法,TCP_NODELAY包含在 <netinet/tcp.h>
    ::setsockopt(sockfd_, IPPROTO_TCP, TCP_NODELAY, &optval, sizeof(optval));
}

void Socket::setReuseAddr(bool on){
    int optval = on ? 1 : 0;
    ::setsockopt(sockfd_, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval));
}

void Socket::setReusePort(bool on){
    int optval = on ? 1 : 0;
    ::setsockopt(sockfd_, SOL_SOCKET, SO_REUSEPORT, &optval, sizeof(optval));
}

void Socket::setKeepAlive(bool on){
    int optval = on ? 1 : 0;
    ::setsockopt(sockfd_, SOL_SOCKET, SO_KEEPALIVE, &optval, sizeof(optval));
}

二、Accept类

在这里插入图片描述

Acceptor工作在mainReactor,用于监听新用户的连接,将与客户端通信的fd打包成Channel,muduo采用轮询算法找一个subloop,将其唤醒,把打包好的Channel给subloop

1. Acceptor.h

在这里插入图片描述

  • 成员变量acceptSocket_:封装了listenfd,用于监听新用户的连接
  • 成员变量acceptChannel_:封装了listenfd,以及感兴趣的事件events和发生的事件revents
  • 成员变量loop_:通过事件循环监听listenfd,也就是mainLoop
  • 成员变量newConnectionCallback_:如果一个客户端连接成功,Acceptor返回一个Channel给TcpServer,TcpServer会通过轮询唤醒一个subLoop,并把新客户端的Channel给subLoop,而这些事情都是交给一个回调函数做的,即newConnectionCallback_

此处TcpServer流程待修改

#pragma once

#include "noncopyable.h"
#include "Socket.h"
#include "Channel.h"
#include "EventLoop.h"

#include <functional>

class EventLoop;
class InetAddress;

class Acceptor : noncopyable{
    public:
        using NewConnetionCallback = std::function<void(int sockfd, const InetAddress&)>;
        
        Acceptor(EventLoop* loop, const InetAddress& listenAddr, bool reuseport);
        ~Acceptor();

        void setNewConnectionCallback(const NewConnetionCallback& cb){
            newConnetionCallback_ = std::move(cb);
        }

        bool listening(){ return listening_; }
        void listen();

    private:
        void handleRead();

        EventLoop* loop_;
        Socket acceptSocket_;
        Channel acceptChannel_;
        NewConnetionCallback newConnetionCallback_;
        bool listening_;
};
  • 成员函数handleRead:服务器的listenfd发生读事件调用,即新用户连接

2. Acceptor.cc

socket函数第二个参数type
在这里插入图片描述
SOCK_NONBLOCK:非阻塞形式,没有事件发生立刻返回
SOCK_CLOEXEC:fork之后子进程会继承父进程所有的文件描述符,此时子进程会关闭这些继承来的文件描述符

#include "Acceptor.h"
#include "Logger.h"
#include "InetAddress.h"

#include <sys/types.h>
#include <sys/socket.h>
#include <unistd.h>
#include <netinet/in.h>
#include <errno.h>

static int createNonblocking(){
    // 创建listenfd
    int sockfd = ::socket(AF_INET, SOCK_STREAM | SOCK_NONBLOCK | SOCK_CLOEXEC, IPPROTO_TCP);
    if(sockfd <= 0){
        LOG_FATAL("%s:%s:%d listen socket crete err:%d \n", __FILE__, __FUNCTION__, __LINE__, errno);
    }
    return sockfd;
}

Acceptor::Acceptor(EventLoop* loop, const InetAddress& listenAddr, bool reuseport)
    : loop_(loop)
    , acceptSocket_(createNonblocking())             // Socket的构造函数需要一个int参数,createNonblocking()返回值就是int
    , acceptChannel_(loop, acceptSocket_.fd())       // 第一个参数就是Channel所属的EventLoop
    , listening_(false)
{
    acceptSocket_.setReuseAddr(true);
    acceptSocket_.setReusePort(true);
    acceptSocket_.bindAddress(listenAddr);        // bind
    // 当我们TcpServer调用start方法时,就会启动Acceptor的listen方法
    // 有新用户连接时,要执行一个回调,这个方法会将和用户连接的fd打包成Channel,然后交给subloop
    // 下面就是注册包装了listenfd的Channel发生读事件后,需要执行的回调函数
    acceptChannel_.setReadCallBack(std::bind(&Acceptor::handleRead, this));
}

Acceptor::~Acceptor(){
    acceptChannel_.disableAll();
    acceptChannel_.remove();
}

void Acceptor::listen(){
    listening_ = true;              // 开始监听
    acceptSocket_.listen();         // 调用底层的::listen
    acceptChannel_.enableReading(); // 将acceptChannel_注册到Poller,或者在Poller更新自己感兴趣的事件 
}

// 服务器的listenfd发生读事件调用,即新用户连接
void Acceptor::handleRead(){
    InetAddress peerAddr;  // 客户端地址
    int connfd = acceptSocket_.accept(&peerAddr);
    if(connfd >= 0){
        if(newConnetionCallback_){
            newConnetionCallback_(connfd, peerAddr);  // 轮询找到subloop,唤醒并分发封装connfd的Channel
        }else{
            // 如果没有设置新用户连接的回调操作,就关闭连接
            ::close(connfd);
        }
    }else{
        // accept出错
        LOG_ERROR("%s:%s:%d accept err:%d \n", __FILE__, __FUNCTION__, __LINE__, errno);
        if(errno == EMFILE){
            // 打开的fd达到上限
            LOG_ERROR("%s:%s:%d the open fd exceeding the resource limit\n", __FILE__, __FUNCTION__, __LINE__);
        }
    }
}
  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-05-01 15:31:12  更:2022-05-01 15:33:01 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/11 3:52:11-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码