Skip to content
forked from aixi/blaze

A multi-threading Ractor network library in Linux

Notifications You must be signed in to change notification settings

xinlzhang/blaze

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

blaze: A multi-thread Linux C++ network library

简介

blaze 是仿照 muduo[1] 实现的一个基于 Reactor 模式的多线程 C++ 网络库。改动的部分如下:

  • 去掉了 muduo/base 中的基础库,利用 C++11 的新特性而非自己实现。
  • 多线程依赖于 C++11 提供的 std::thread,而不是重新封装 POSIX thread API。
  • 原子操作使用 C++11 提供的 std::atomic。
  • 重新实现了 BlockingQueue、BoundedBlockingQueue、CountDownLatch 等线程安全容器
  • 新增了ThreadGuard类,防止joinable thread对象析构时调用 std::terminate

Requirements

  • C++17 (required 用到了std::any和std::string_view)
  • CMake (required)
  • Linux (required 如果你要使用epoll)
  • Google Protobuf 3.0 or newer (optional)

需要什么预备知识

  • 理解网络编程的基础知识,了解TCP协议,了解 socket API
  • 了解多线程编程,至少能够正确地使用 mutex和condition_variable
  • 理解C++中的RAII资源管理手法以及智能指针

推荐的书

  • Effective C++, C++沉思录, Effective Modern C++
  • Linux高性能服务器编程, Effective TCP/IP programming
  • unix网络编程和unix环境高级编程等书适合查阅,如果从头开始看很容易丧失兴趣
  • 陈硕的Linux多线程服务器端编程
  • 网络编程的关键不是socket API怎么用,而是理解网络背后的机制,到底发了什么包,收了什么包

示例

一个简单的echo服务器如下:

class EchoServer : public noncopyable
{
public:
  EchoServer(EventLoop* loop, const InetAddress& addr)
    : loop_(loop),
      server_(loop, addr),
  {
    // set echo callback
    server_.SetMessageCallback(std::bind(&EchoServer::OnMessage, this, _1, _2, _3));
  }
  void Start() 
  { 
    server_.Start(); 
  }
  
  void onMessage(const TcpConnectionPtr& conn, Buffer* buf, Timestamp time)
  {
    // echo message
    conn->Send(buf);
  }
private:
  EventLoop* loop_;
  TcpServer server_;
};

这个实现非常简单,只需关注OnMessage回调函数,它将收到消息发回客户端。然而,该实现有一个问题:若客户端只发送而不接收数据(即只调用write而不调用read),则TCP的流量控制(flow control)会导致数据堆积在服务端,最终会耗尽服务端内存。为解决该问题我们引入高/低水位回调:

class EchoServer : public noncopyable
{
public:
  ...
  void OnConnection(const TcpConnectionPtr& conn)
  {
    if (conn->Connected())
      conn->SetHighWaterMarkCallback(
            std::bind(&EchoServer::OnHighWaterMark, this, _1, _2), 1024);
  }
  void OnHighWaterMark(const TcpConnectionPtr& conn, size_t mark)
  {
    LOG_INFO << "high water mark " << mark << " bytes, stop read";
    conn->StopRead();
  }
  
  void OnWriteComplete(const TcpConnectionPtr& conn)
  {
    if (!conn->IsReading()) {
      LOG_INFO << "write complete, start read";
      conn->StartRead();
    }
  }
  ...
};

新增了3个回调:OnConnectionOnHighWaterMarkOnWriteComplete。当TCP连接建立时OnConnection会设置高水位回调值(high water mark);当 output buffer 达到该值时,OnHighWaterMark会停止读socket;当 Output buffer全部写入内核时,OnWriteComplete会重新开始读socket。

然后,我们给服务器加上多线程功能。实现起来非常简单,只需加一行代码即可:

void EchoServer::SetThreadNum(size_t n)
{
  // set thread num here
  server_.SetNumThread(n);
}

最后,main函数如下:

int main()
{
  EventLoop loop;
  // listen address localhost:9877
  InetAddress server_addr(9877);
  // echo server with 4 threads and timeout of 10 seconds
  EchoServer server(&loop, server_addr);
  server.SetThreadNum(4)
  server.Start();
  // quit after 1 minute
  loop.RunAfter(60, [&](){ loop.quit(); });
  // loop main thread
  loop.Loop();
}

blaze还提供了定时器功能:
利用std::set管理定时器,std::set提供了lower_bound功能,可以较快地找到已到期的定时器:

void PrintTid()
{
    std::stringstream ss;
    ss << std::this_thread::get_id();
    printf("pid = %d, tid = %s\n", getpid(), ss.str().data());
}

void Print(const char* msg)
{
    printf("msg %s %s\n", Timestamp::Now().ToString().c_str(), msg);
    if (++cnt == 20)
    {
        g_loop->Quit();
    }
}

void Cancel(TimerId timer)
{
    g_loop->CancelTimer(timer);
    // 自注销
    printf("cancelled timer at %s\n", Timestamp::Now().ToString().c_str());
}

int main()
{
    PrintTid();
    sleep(1);
    {
        EventLoop loop;
        g_loop = &loop;
        Print("main");
        loop.RunAfter(1, [](){Print("once1");});
        loop.RunAfter(1.5, [](){Print("once1.5");});
        loop.RunAfter(2.5, [](){Print("once2.5");});
        loop.RunAfter(3.5, [](){Print("once3.5");});
        TimerId t45 = loop.RunAfter(4.5, [](){Print("once4.5");});
        loop.RunAfter(4.2, [t45](){Cancel(t45);});
        loop.RunAfter(4.8, [t45](){Cancel(t45);});
        loop.RunEvery(2, [](){Print("every2");});
        TimerId t3 = loop.RunEvery(3, [](){Print("every3");});
        loop.RunAfter(9.001, [&t3](){Cancel(t3);});
        loop.Loop();
        printf("main loop exits");
    }
    sleep(1);
    {
        EventLoopThread loop_thread;
        EventLoop* loop = loop_thread.StartLoop();
        loop->RunAfter(2, PrintTid);
        sleep(3);
        printf("thread loop exits");
    }
}

简单的HTTP Server

这个简单的HTTP Server并未完全符合HTTP 1.1的标准,仅仅能做到与HttpClient及Web browser通信. 在HttpServer_test.cc中提供了几个简单的response,如hello、favicon图片等,你可以修改HttpServer_test.cc中的OnRequest函数

利用ab命令进行测试

  • ab -n 500000 -r -c 20000 -k 0.0.0.0:8000/hello
  • http server 设置为4线程

测试结果:总请求数500000,并发请求数20000,90%的请求都在0.897 s内得到响应
Percentage of the requests served within a certain time (ms)
50% 548
66% 589
75% 612
80% 629
90% 897
95% 975
98% 12411
99% 12435
100% 14004 (longest request)

参考

[1] Muduo is a multithreaded C++ network library based on the reactor pattern.

About

A multi-threading Ractor network library in Linux

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • C++ 97.1%
  • CMake 2.9%