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

相关文章

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

Window Server创建2台服务器的故障转移群集的图文教程

《WindowServer创建2台服务器的故障转移群集的图文教程》本文主要介绍了在WindowsServer系统上创建一个包含两台成员服务器的故障转移群集,文中通过图文示例介绍的非常详细,对大家的... 目录一、 准备条件二、在ServerB安装故障转移群集三、在ServerC安装故障转移群集,操作与Ser

NFS实现多服务器文件的共享的方法步骤

《NFS实现多服务器文件的共享的方法步骤》NFS允许网络中的计算机之间共享资源,客户端可以透明地读写远端NFS服务器上的文件,本文就来介绍一下NFS实现多服务器文件的共享的方法步骤,感兴趣的可以了解一... 目录一、简介二、部署1、准备1、服务端和客户端:安装nfs-utils2、服务端:创建共享目录3、服

Python MySQL如何通过Binlog获取变更记录恢复数据

《PythonMySQL如何通过Binlog获取变更记录恢复数据》本文介绍了如何使用Python和pymysqlreplication库通过MySQL的二进制日志(Binlog)获取数据库的变更记录... 目录python mysql通过Binlog获取变更记录恢复数据1.安装pymysqlreplicat

Go语言使用Buffer实现高性能处理字节和字符

《Go语言使用Buffer实现高性能处理字节和字符》在Go中,bytes.Buffer是一个非常高效的类型,用于处理字节数据的读写操作,本文将详细介绍一下如何使用Buffer实现高性能处理字节和... 目录1. bytes.Buffer 的基本用法1.1. 创建和初始化 Buffer1.2. 使用 Writ

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

SpringBoot如何使用TraceId日志链路追踪

《SpringBoot如何使用TraceId日志链路追踪》文章介绍了如何使用TraceId进行日志链路追踪,通过在日志中添加TraceId关键字,可以将同一次业务调用链上的日志串起来,本文通过实例代码... 目录项目场景:实现步骤1、pom.XML 依赖2、整合logback,打印日志,logback-sp

Python项目打包部署到服务器的实现

《Python项目打包部署到服务器的实现》本文主要介绍了PyCharm和Ubuntu服务器部署Python项目,包括打包、上传、安装和设置自启动服务的步骤,具有一定的参考价值,感兴趣的可以了解一下... 目录一、准备工作二、项目打包三、部署到服务器四、设置服务自启动一、准备工作开发环境:本文以PyChar

Apache Tomcat服务器版本号隐藏的几种方法

《ApacheTomcat服务器版本号隐藏的几种方法》本文主要介绍了ApacheTomcat服务器版本号隐藏的几种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需... 目录1. 隐藏HTTP响应头中的Server信息编辑 server.XML 文件2. 修China编程改错误

如何在一台服务器上使用docker运行kafka集群

《如何在一台服务器上使用docker运行kafka集群》文章详细介绍了如何在一台服务器上使用Docker运行Kafka集群,包括拉取镜像、创建网络、启动Kafka容器、检查运行状态、编写启动和关闭脚本... 目录1.拉取镜像2.创建集群之间通信的网络3.将zookeeper加入到网络中4.启动kafka集群