sylar高性能服务器-日志(P67-P68)内容记录

2024-03-13 03:52

本文主要是介绍sylar高性能服务器-日志(P67-P68)内容记录,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • P68-P69TcpServer封装
      • class TcpServer
        • 成员变量
        • 构造函数
        • bind(绑定地址以及监听)
        • startAccept
        • Start
        • Stop
    • echo_server

P68-P69TcpServer封装

这一节的内容对于之前学过网络编程的同学看起来比较简单一些,也没啥新的内容,都是使用之前完成的模块搭建了一个简单的echo服务器。

class TcpServer

成员变量
// 多监听,多网卡
std::vector<Socket::ptr> m_socks;
// 新连接的Socket工作的调度器, IOManager就是线程池
IOManager* m_worker;
// 服务器Socket接收连接的调度器 
IOManager* m_acceptWorker;
// 接收超时时间
uint64_t m_recvTimeout;
// 服务器名称
std::string m_name;
// 服务是否停止
bool m_isStop; 
构造函数
static sylar::ConfigVar<uint64_t>::ptr g_tcp_server_read_timeout =sylar::Config::Lookup("tcp_server.read_timeout", (uint64_t)(60 * 1000 * 2),"tcp server read timeout");TcpServer::TcpServer(sylar::IOManager* worker, sylar::IOManager* accept_worker):m_worker(worker),m_acceptWorker(accept_worker),m_recvTimeout(g_tcp_server_read_timeout->getValue()),m_name("sylar/1.0.0"),m_isStop(true) {}
bind(绑定地址以及监听)
// 绑定单个地址
bool TcpServer::bind(sylar::Address::ptr addr) {// 绑定成功的地池容器std::vector<Address::ptr> addrs;// 绑定失败的地址容器std::vector<Address::ptr> fail;addrs.push_back(addr);return bind(addrs, fail);
}// 绑定多个地址容器
bool TcpServer::bind(const std::vector<Address::ptr>& addrs, std::vector<Address::ptr>& fails) {//  m_ssl = ssl;// 遍历传入的地址容器for(auto& addr : addrs) {// Socket::ptr sock = ssl ? SSLSocket::CreateTCP(addr) : Socket::CreateTCP(addr);// 调用之前封装好的hook函数,创建一个TCP连接Socket::ptr sock = Socket::CreateTCP(addr);// 绑定失败if(!sock->bind(addr)) {SYLAR_LOG_ERROR(g_logger) << "bind fail errno="<< errno << " errstr=" << strerror(errno)<< " addr=[" << addr->toString() << "]";// 记录当前失败的地址fails.push_back(addr);// 继续下一个地址continue;}// 监听失败if(!sock->listen()) {SYLAR_LOG_ERROR(g_logger) << "listen fail errno="<< errno << " errstr=" << strerror(errno)<< " addr=[" << addr->toString() << "]";fails.push_back(addr);continue;}// 绑定、监听都成功m_socks.push_back(sock);}// 如果绑定失败的地址容器不为空,bind调用函数返回false,清空所有Socketif(!fails.empty()) {m_socks.clear();return false;}// 终端打印出绑定好的地址for(auto& i : m_socks) {// SYLAR_LOG_INFO(g_logger) << "type=" << m_type//     << " name=" << m_name//     << " ssl=" << m_sslSYLAR_LOG_INFO(g_logger)  << " server bind success: " << *i;}return true;
}
startAccept
// 服务器不断监听传入的Socket对象,等待并接收客户端的连接
void TcpServer::startAccept(Socket::ptr sock) {// 如果服务器没有停止while(!m_isStop) {// 接受一个新的客户端连接Socket::ptr client = sock->accept();// 成功接收了一个连接if(client) {// 设置服务器等待客户端发送数据的最大时间client->setRecvTimeout(m_recvTimeout);// m_ioWorker->schedule(std::bind(&TcpServer::handleClient,//             shared_from_this(), client));// 将handleClient加入到工作线程队列m_worker中m_worker->schedule(std::bind(&TcpServer::handleClient,shared_from_this(), client));} else {SYLAR_LOG_ERROR(g_logger) << "accept errno=" << errno<< " errstr=" << strerror(errno);}}
}
Start
bool TcpServer::start() {if(!m_isStop) {return true;}m_isStop = false;for(auto& sock : m_socks) {// 异步执行startAcceptm_acceptWorker->schedule(std::bind(&TcpServer::startAccept,shared_from_this(), sock));}return true;
}
Stop
void TcpServer::stop() {// 标记服务器为停止状态m_isStop = true;// 使用shared_from_this()获取当前TcpServer对象的共享指针,并将其存储在局部变量self中。// 这是为了确保在异步任务执行期间TcpServer对象不会被销毁。auto self = shared_from_this();// 这个lambda表达式捕获了this指针(即当前对象的指针)和self(即TcpServer的共享指针)。m_acceptWorker->schedule([this, self]() {for(auto& sock : m_socks) {sock->cancelAll();sock->close();}m_socks.clear();});
}

echo_server

#include "sylar/tcp_server.h"
#include "sylar/log.h"
#include "sylar/iomanager.h"
#include "sylar/bytearray.h"
#include "sylar/address.h"static sylar::Logger::ptr g_logger = SYLAR_LOG_ROOT();// EchoServer继承于TcpServer
class EchoServer : public sylar::TcpServer {
public:EchoServer(int type);// 重写void handleClient(sylar::Socket::ptr client);private:// 控制服务器行为:是否以文本或二进制形式输出接收到的数据int m_type = 0;
};EchoServer::EchoServer(int type):m_type(type) {
}// 处理客户端连接
void EchoServer::handleClient(sylar::Socket::ptr client) {// 日志打印正在处理的来自某个客户端的连接SYLAR_LOG_INFO(g_logger) << "handleClient " << *client;   // 存储从客户端接收到的数据sylar::ByteArray::ptr ba(new sylar::ByteArray);// 循环接收数据while(true) {// 清空缓冲区ba->clear();/* struct iovec{void *iov_base;	 Pointer to data.  size_t iov_len;	 Length of data.  };*/std::vector<iovec> iovs;// 用于接收客户端发送的数,这里指定了每个iovec的大小为1024字节。ba->getWriteBuffers(iovs, 1024);// 接收数据, rt保存实际接收到的字节数int rt = client->recv(&iovs[0], iovs.size());if(rt == 0) {SYLAR_LOG_INFO(g_logger) << "client close: " << *client;break;} else if(rt < 0) {SYLAR_LOG_INFO(g_logger) << "client error rt=" << rt<< " errno=" << errno << " errstr=" << strerror(errno);break;}// 移动位置指针在接收到的数据后面,不能删除,防止新接收的数据覆盖之前的数据ba->setPosition(ba->getPosition() + rt);// 设置为0方便从头读取或处理数据ba->setPosition(0);// SYLAR_LOG_INFO(g_logger) << "recv rt=" << rt << " data=" << std::string((char*)iovs[0].iov_base, rt);if(m_type == 1) {//text std::cout << ba->toString() << std::endl;// SYLAR_LOG_INFO(g_logger) << ba->toString();} else {std::cout << ba->toHexString() << std::endl;// SYLAR_LOG_INFO(g_logger) << ba->toHexString();}std::cout.flush();}
}int type = 1;void run() {SYLAR_LOG_INFO(g_logger) << "server type=" << type;EchoServer::ptr es(new EchoServer(type));// 解析传入的字符串地址 "0.0.0.0"是一个特殊的IP地址,表示服务器应该监听所有可用的网络接口。auto addr = sylar::Address::LookupAny("0.0.0.0:8020");while(!es->bind(addr)) {sleep(2);}es->start();
}int main(int argc, char** argv) {if(argc < 2) {SYLAR_LOG_INFO(g_logger) << "used as[" << argv[0] << " -t] or [" << argv[0] << " -b]";return 0;}if(!strcmp(argv[1], "-b")) {type = 2;}sylar::IOManager iom(2);iom.schedule(run);return 0;
}

服务器

image-20240312201551374

客户端

image-20240312201615264

这篇关于sylar高性能服务器-日志(P67-P68)内容记录的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/803567

相关文章

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

解读GC日志中的各项指标用法

《解读GC日志中的各项指标用法》:本文主要介绍GC日志中的各项指标用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、基础 GC 日志格式(以 G1 为例)1. Minor GC 日志2. Full GC 日志二、关键指标解析1. GC 类型与触发原因2. 堆

mysql中的服务器架构详解

《mysql中的服务器架构详解》:本文主要介绍mysql中的服务器架构,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、mysql服务器架构解释3、总结1、背景简单理解一下mysqphpl的服务器架构。2、mysjsql服务器架构解释mysql的架

在Spring Boot中集成RabbitMQ的实战记录

《在SpringBoot中集成RabbitMQ的实战记录》本文介绍SpringBoot集成RabbitMQ的步骤,涵盖配置连接、消息发送与接收,并对比两种定义Exchange与队列的方式:手动声明(... 目录前言准备工作1. 安装 RabbitMQ2. 消息发送者(Producer)配置1. 创建 Spr

Linux如何快速检查服务器的硬件配置和性能指标

《Linux如何快速检查服务器的硬件配置和性能指标》在运维和开发工作中,我们经常需要快速检查Linux服务器的硬件配置和性能指标,本文将以CentOS为例,介绍如何通过命令行快速获取这些关键信息,... 目录引言一、查询CPU核心数编程(几C?)1. 使用 nproc(最简单)2. 使用 lscpu(详细信

k8s上运行的mysql、mariadb数据库的备份记录(支持x86和arm两种架构)

《k8s上运行的mysql、mariadb数据库的备份记录(支持x86和arm两种架构)》本文记录在K8s上运行的MySQL/MariaDB备份方案,通过工具容器执行mysqldump,结合定时任务实... 目录前言一、获取需要备份的数据库的信息二、备份步骤1.准备工作(X86)1.准备工作(arm)2.手

MySQL 打开binlog日志的方法及注意事项

《MySQL打开binlog日志的方法及注意事项》本文给大家介绍MySQL打开binlog日志的方法及注意事项,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录一、默认状态二、如何检查 binlog 状态三、如何开启 binlog3.1 临时开启(重启后失效)

Java实现删除文件中的指定内容

《Java实现删除文件中的指定内容》在日常开发中,经常需要对文本文件进行批量处理,其中,删除文件中指定内容是最常见的需求之一,下面我们就来看看如何使用java实现删除文件中的指定内容吧... 目录1. 项目背景详细介绍2. 项目需求详细介绍2.1 功能需求2.2 非功能需求3. 相关技术详细介绍3.1 Ja

SpringBoot3应用中集成和使用Spring Retry的实践记录

《SpringBoot3应用中集成和使用SpringRetry的实践记录》SpringRetry为SpringBoot3提供重试机制,支持注解和编程式两种方式,可配置重试策略与监听器,适用于临时性故... 目录1. 简介2. 环境准备3. 使用方式3.1 注解方式 基础使用自定义重试策略失败恢复机制注意事项

MySQL MCP 服务器安装配置最佳实践

《MySQLMCP服务器安装配置最佳实践》本文介绍MySQLMCP服务器的安装配置方法,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下... 目录mysql MCP 服务器安装配置指南简介功能特点安装方法数据库配置使用MCP Inspector进行调试开发指