仿muduo库实现高并发服务器--日志的书写和套接字Socket的实现

目录

一、日志的书写

二、套接字Socket的实现


一、日志的书写

我们直接使用宏来实现日志

我们将日志的等级分为三个等级 : INF DBG ERR

首先设置什么等级的日志可以打印,以便区别测试版本和上线版本

使用time函数来获取时间戳 ,local 将时间戳转化成当前时间,struct tm 储存拆分后的结构体,strftime按照对应的格式转化成字符串,fprintf向标准输出打印 stdout 标准输出,%s = ts %s = FILE %s = LINE format:是日志宏的自定义格式参数(比如你传的 "这是一条日志,数字:% d");"\n":换行符,让每条日志单独一行。##__VA_ARGS__:GCC 扩展语法,代表「可变参数列表」(比如你传的 123, "test"),## 是为了处理没有可变参数的情况(避免编译错误)。

二、套接字Socket的实现

成员变量:

文件描述符

成员函数:

创建套接字

cpp 复制代码
 bool Creat()
    {
        _sockfd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
        if (_sockfd < 0)
        {
            ERR_LOG("CREAT SOCKET FAILED");
            return false;
        }
        return true;
    }

绑定套接字

cpp 复制代码
 bool Bind(const std::string &ip, uint16_t port)
    {
        struct sockaddr_in addr;
        addr.sin_family = AF_INET;//设置为ipv4
        addr.sin_port = htons(port);//转化为大端口
        addr.sin_addr.s_addr = inet_addr(ip.c_str());//将点分十进制转化为32字节的整数
        socklen_t len = sizeof(struct sockaddr_in);
        int ret = bind(_sockfd, (struct sockaddr *)&addr, len);
        if (ret < 0)
        {
            ERR_LOG("BIND FAILED");
            return false;
        }
        return true;
    }

监听

cpp 复制代码
#define MAX_LISTEN 10
    bool Listen(int backlog = MAX_LISTEN)
    {
        int ret = listen(_sockfd, backlog);
        if (ret < 0)
        {
            ERR_LOG("LISTEN FAILED");
            return false;
        }
        return true;
    }

链接

cpp 复制代码
 bool Connect(const std::string &ip, uint16_t port)
    {
        struct sockaddr_in addr;
        addr.sin_family = AF_INET;
        addr.sin_port = htons(port);
        addr.sin_addr.s_addr = inet_addr(ip.c_str());
        socklen_t len = sizeof(struct sockaddr_in);
        int ret = connect(_sockfd, (struct sockaddr *)&addr, len);
        if (ret < 0)
        {
            ERR_LOG("Connect FAILED");
            return false;
        }
        return true;
    }

接收链接

cpp 复制代码
int Accpet()
    {
        int newfd = accept(_sockfd, NULL, NULL);
        if (newfd < 0)
        {
            ERR_LOG("ACCEPT FAILE");
            return -1;
        }
        return newfd;
    }

读取数据

cpp 复制代码
ssize_t Recv(void *buf, size_t len, int flag = 0)
    {
        ssize_t ret = recv(_sockfd, buf, len, flag);
        if (ret < 0)
        {
            if (errno == EAGAIN || errno == EINTR)
            {
                return 0;
            }
            ERR_LOG("RECV FAILED");
            return -1;
        }
        return ret;
    }

读取设置为非阻塞

cpp 复制代码
 ssize_t NonBlockRecv(void *buf, size_t len) //
    {
        return Recv(buf, len, MSG_DONTWAIT); // MSG_DONTWAIT为非阻塞
    }

发送数据

cpp 复制代码
 ssize_t Send(const void *buf, size_t len, int flag = 0)
    {
        ssize_t ret = send(_sockfd, buf, len, flag);
        if (ret < 0)
        {
            if (errno == EAGAIN || errno == EINTR)
            {
                return 0;
            }
            ERR_LOG("SEND FAILED");
            return -1;
        }
        return ret;
    }

发送设置为非阻塞

cpp 复制代码
 ssize_t NonBlockSend(void *buf, size_t len)
    {
        if (len == 0)
            return 0;
        return Send(buf, len, MSG_DONTWAIT);
    }

关闭链接

cpp 复制代码
 void Close()
    {
        if (_sockfd != -1)
        {
            close(_sockfd);
        }
        _sockfd = -1;
    }

开启地址重用

cpp 复制代码
void ReuseAddress()
    {
        int val = 1;
        // 端口
        setsockopt(_sockfd, SOL_SOCKET, SO_REUSEADDR, (void *)&val, sizeof(int));
        val = 1;
        // 接口
        setsockopt(_sockfd, SOL_SOCKET, SO_REUSEPORT, (void *)&val, sizeof(int));
    }

套接字设置为非阻塞

cpp 复制代码
  void NonBlock()
    {
        int flag = fcntl(_sockfd, F_GETFL, 0); // 不能覆盖
        fcntl(_sockfd, F_SETFL, flag | O_NONBLOCK);
    }

创建服务端

cpp 复制代码
 bool CreatServer(uint16_t port, const std::string &ip = "0.0.0.0")
    {
        if (Creat() == false)
            return false;
        ReuseAddress();
        NonBlock();
        if (Bind(ip, port) == false)
            return false;
        if (Listen() == false)
            return false;

        return true;
    }

创建客户端

cpp 复制代码
 bool CreatClient(uint16_t port, const std::string &ip = "0.0.0.0")
    {
        if (Creat() == false)
            return false;
        if (Connect(ip, port) == false)
            return false;
        return true;
    }

整体代码

cpp 复制代码
class Socket
{
private:
    int _sockfd;

public:
    Socket() : _sockfd(-1)
    {
    }
    Socket(int sockfd) : _sockfd(sockfd)
    {
    }
    int Fd()
    {
        return _sockfd;
    }
    ~Socket()
    {
        Close();
    }
    bool Creat()
    {
        _sockfd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
        if (_sockfd < 0)
        {
            ERR_LOG("CREAT SOCKET FAILED");
            return false;
        }
        return true;
    }
    bool Bind(const std::string &ip, uint16_t port)
    {
        struct sockaddr_in addr;
        addr.sin_family = AF_INET;
        addr.sin_port = htons(port);
        addr.sin_addr.s_addr = inet_addr(ip.c_str());
        socklen_t len = sizeof(struct sockaddr_in);
        int ret = bind(_sockfd, (struct sockaddr *)&addr, len);
        if (ret < 0)
        {
            ERR_LOG("BIND FAILED");
            return false;
        }
        return true;
    }
#define MAX_LISTEN 10
    bool Listen(int backlog = MAX_LISTEN)
    {
        int ret = listen(_sockfd, backlog);
        if (ret < 0)
        {
            ERR_LOG("LISTEN FAILED");
            return false;
        }
        return true;
    }
    bool Connect(const std::string &ip, uint16_t port)
    {
        struct sockaddr_in addr;
        addr.sin_family = AF_INET;
        addr.sin_port = htons(port);
        addr.sin_addr.s_addr = inet_addr(ip.c_str());
        socklen_t len = sizeof(struct sockaddr_in);
        int ret = connect(_sockfd, (struct sockaddr *)&addr, len);
        if (ret < 0)
        {
            ERR_LOG("Connect FAILED");
            return false;
        }
        return true;
    }
    int Accpet()
    {
        int newfd = accept(_sockfd, NULL, NULL);
        if (newfd < 0)
        {
            ERR_LOG("ACCEPT FAILE");
            return -1;
        }
        return newfd;
    }
    ssize_t Recv(void *buf, size_t len, int flag = 0)
    {
        ssize_t ret = recv(_sockfd, buf, len, flag);
        if (ret < 0)
        {
            if (errno == EAGAIN || errno == EINTR)
            {
                return 0;
            }
            ERR_LOG("RECV FAILED");
            return -1;
        }
        return ret;
    }
    ssize_t NonBlockRecv(void *buf, size_t len) //
    {
        return Recv(buf, len, MSG_DONTWAIT); // MSG_DONTWAIT为非阻塞
    }
    ssize_t Send(const void *buf, size_t len, int flag = 0)
    {
        ssize_t ret = send(_sockfd, buf, len, flag);
        if (ret < 0)
        {
            if (errno == EAGAIN || errno == EINTR)
            {
                return 0;
            }
            ERR_LOG("SEND FAILED");
            return -1;
        }
        return ret;
    }
    ssize_t NonBlockSend(void *buf, size_t len)
    {
        if (len == 0)
            return 0;
        return Send(buf, len, MSG_DONTWAIT);
    }

    void Close()
    {
        if (_sockfd != -1)
        {
            close(_sockfd);
        }
        _sockfd = -1;
    }
    bool CreatServer(uint16_t port, const std::string &ip = "0.0.0.0")
    {
        if (Creat() == false)
            return false;
        ReuseAddress();
        NonBlock();
        if (Bind(ip, port) == false)
            return false;
        if (Listen() == false)
            return false;

        return true;
    }
    bool CreatClient(uint16_t port, const std::string &ip = "0.0.0.0")
    {
        if (Creat() == false)
            return false;
        if (Connect(ip, port) == false)
            return false;
        return true;
    }
    // 设置套接字选项,开启地址端口重用
    void ReuseAddress()
    {
        int val = 1;
        // 端口
        setsockopt(_sockfd, SOL_SOCKET, SO_REUSEADDR, (void *)&val, sizeof(int));
        val = 1;
        // 接口
        setsockopt(_sockfd, SOL_SOCKET, SO_REUSEPORT, (void *)&val, sizeof(int));
    }
    void NonBlock()
    {
        int flag = fcntl(_sockfd, F_GETFL, 0); // 不能覆盖
        fcntl(_sockfd, F_SETFL, flag | O_NONBLOCK);
    }
};
相关推荐
Irene19911 小时前
ElementPlus 与成熟后台框架对比:vue-element-plus-admin、vue-pure-admin等
前端·ui·框架·vue3
尘中客5 小时前
放弃 Echarts?前端直接渲染后端高精度 SVG 矢量图流的踩坑记录
前端·javascript·echarts·前端开发·svg矢量图·echarts避坑
FreeBuf_5 小时前
Chrome 0Day漏洞遭野外利用
前端·chrome
小彭努力中5 小时前
199.Vue3 + OpenLayers 实现:点击 / 拖动地图播放音频
前端·vue.js·音视频·openlayers·animate
2501_916007475 小时前
网站爬虫原理,基于浏览器点击行为还原可接口请求
前端·javascript·爬虫·ios·小程序·uni-app·iphone
前端大波6 小时前
Sentry 每日错误巡检自动化:设计思路与上手实战
前端·自动化·sentry
ZC跨境爬虫7 小时前
使用Claude Code开发校园交友平台前端UI全记录(含架构、坑点、登录逻辑及算法)
前端·ui·架构
慧一居士7 小时前
Vue项目中,何时使用布局、子组件嵌套、插槽 对应的使用场景,和完整的使用示例
前端·vue.js
Можно7 小时前
uni.request 和 axios 的区别?前端请求库全面对比
前端·uni-app
M ? A8 小时前
解决 VuReact 中 ESLint 规则冲突的完整指南
前端·react.js·前端框架