cellnet是一个组件化、高扩展性、高性能的开源服务器网络库。它的主要概念如下


队列
队列使用NewEventQueue创建,使用.StartLoop()开启队列事件处理循环,所有投递到队列中的函数回调会在队列自由的goroutine中被调用,逻辑在此时被处理

一般在main goroutine中调用queue.Wait阻塞等待队列结束。

队列在cellnet中使用cellnet.Queue接口, 底层由带缓冲的channel实现

端(Peer)
peer是客户端和服务端的封装
cellnet使用Acceptor接收多个连接,Acceptor是一种Peer(端),连接到Acceptor的Peer叫做Connector。

一个Peer拥有很多属性(名称,地址,队列),peer.NewGenericPeer函数封装了属性的设置过程。

peer.NewGenericPeer创建好的Peer不会产生任何socket操作,对于Acceptor来说,调用Acceptor的Start方法后,才会真正开始socket的侦听

使用如下代码创建一个接受器(Acceptor):

  queue := cellnet.NewEventQueue()

    // NewGenericPeer参数依次是: peer类型, peer名称(日志中方便查看), 侦听地址,事件队列
    peerIns := peer.NewGenericPeer("tcp.Acceptor", "server", "127.0.0.1:8801", queue)

    peerIns.Start()

Connector也是一种Peer,与Acceptor很很多类似的地方,因此创建过程也是类似的。

使用如下代码创建一个连接器(Connector):

queue := cellnet.NewEventQueue()

    peerIns := peer.NewGenericPeer("tcp.Connector", "client", "127.0.0.1:8801", queue)

    peerIns.Start("127.0.0.1:8801")

处理器(proc)
cellnet使用Processor处理消息的收发过程。

使用proc.BindProcessorHandler函数,将一个Peer绑定到某个Processor上,且设置用户消息处理回调。

下面代码尝试将peerIns的Peer,绑定"tcp.ltv"处理器,回调函数为func(ev cellnet.Event) { ... }

proc.BindProcessorHandler(peerIns, "tcp.ltv", func(ev cellnet.Event) {

    switch msg := ev.Message().(type) {
    // 有新的连接连到8801端口
    case *cellnet.SessionAccepted:
        log.Debugln("server accepted")
    // 有连接从8801端口断开
    case *cellnet.SessionClosed:
        log.Debugln("session closed: ", ev.Session().ID())
    // 收到某个连接的ChatREQ消息
    case *proto.ChatREQ:

        // 准备回应的消息
        ack := proto.ChatACK{
            Content: msg.Content,       // 聊天内容
            Id:      ev.Session().ID(), // 使用会话ID作为发送内容的ID
        }

        // 在Peer上查询SessionAccessor接口,并遍历Peer上的所有连接,并发送回应消息(即广播消息)
        p.(cellnet.SessionAccessor).VisitSession(func(ses cellnet.Session) bool {

            ses.Send(&ack)

            return true
        })

    }

})

callnet 通过这三种封装将网络连接和处理消息解耦开,下篇文章将要介绍callnet关于peer源码的解析,不同的协议(http,tcp, protobuf)都有对应的peer我将取其中(tcp)解析,其它的协议下的内容都差不多。