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

相关文章

pycharm远程连接服务器运行pytorch的过程详解

《pycharm远程连接服务器运行pytorch的过程详解》:本文主要介绍在Linux环境下使用Anaconda管理不同版本的Python环境,并通过PyCharm远程连接服务器来运行PyTorc... 目录linux部署pytorch背景介绍Anaconda安装Linux安装pytorch虚拟环境安装cu

SpringBoot项目注入 traceId 追踪整个请求的日志链路(过程详解)

《SpringBoot项目注入traceId追踪整个请求的日志链路(过程详解)》本文介绍了如何在单体SpringBoot项目中通过手动实现过滤器或拦截器来注入traceId,以追踪整个请求的日志链... SpringBoot项目注入 traceId 来追踪整个请求的日志链路,有了 traceId, 我们在排

MySQL 中的服务器配置和状态详解(MySQL Server Configuration and Status)

《MySQL中的服务器配置和状态详解(MySQLServerConfigurationandStatus)》MySQL服务器配置和状态设置包括服务器选项、系统变量和状态变量三个方面,可以通过... 目录mysql 之服务器配置和状态1 MySQL 架构和性能优化1.1 服务器配置和状态1.1.1 服务器选项

ElasticSearch+Kibana通过Docker部署到Linux服务器中操作方法

《ElasticSearch+Kibana通过Docker部署到Linux服务器中操作方法》本文介绍了Elasticsearch的基本概念,包括文档和字段、索引和映射,还详细描述了如何通过Docker... 目录1、ElasticSearch概念2、ElasticSearch、Kibana和IK分词器部署

部署Vue项目到服务器后404错误的原因及解决方案

《部署Vue项目到服务器后404错误的原因及解决方案》文章介绍了Vue项目部署步骤以及404错误的解决方案,部署步骤包括构建项目、上传文件、配置Web服务器、重启Nginx和访问域名,404错误通常是... 目录一、vue项目部署步骤二、404错误原因及解决方案错误场景原因分析解决方案一、Vue项目部署步骤

Linux流媒体服务器部署流程

《Linux流媒体服务器部署流程》文章详细介绍了流媒体服务器的部署步骤,包括更新系统、安装依赖组件、编译安装Nginx和RTMP模块、配置Nginx和FFmpeg,以及测试流媒体服务器的搭建... 目录流媒体服务器部署部署安装1.更新系统2.安装依赖组件3.解压4.编译安装(添加RTMP和openssl模块

如何解决Pycharm编辑内容时有光标的问题

《如何解决Pycharm编辑内容时有光标的问题》文章介绍了如何在PyCharm中配置VimEmulator插件,包括检查插件是否已安装、下载插件以及安装IdeaVim插件的步骤... 目录Pycharm编辑内容时有光标1.如果Vim Emulator前面有对勾2.www.chinasem.cn如果tools工

关于Spring @Bean 相同加载顺序不同结果不同的问题记录

《关于Spring@Bean相同加载顺序不同结果不同的问题记录》本文主要探讨了在Spring5.1.3.RELEASE版本下,当有两个全注解类定义相同类型的Bean时,由于加载顺序不同,最终生成的... 目录问题说明测试输出1测试输出2@Bean注解的BeanDefiChina编程nition加入时机总结问题说明

Spring Boot整合log4j2日志配置的详细教程

《SpringBoot整合log4j2日志配置的详细教程》:本文主要介绍SpringBoot项目中整合Log4j2日志框架的步骤和配置,包括常用日志框架的比较、配置参数介绍、Log4j2配置详解... 目录前言一、常用日志框架二、配置参数介绍1. 日志级别2. 输出形式3. 日志格式3.1 PatternL

JavaWeb-WebSocket浏览器服务器双向通信方式

《JavaWeb-WebSocket浏览器服务器双向通信方式》文章介绍了WebSocket协议的工作原理和应用场景,包括与HTTP的对比,接着,详细介绍了如何在Java中使用WebSocket,包括配... 目录一、概述二、入门2.1 POM依赖2.2 编写配置类2.3 编写WebSocket服务2.4 浏