《用Java写一个通用的服务器程序》03 处理新socket

在讲监听器时说过处理的新的socket要尽快返回,监听器调用的是ClientFactory的createPhysicalConnection方法,那么就来看这个方法:

    public boolean createPhysicalConnection(PushClientSocket socket,
            boolean isObserver, ListenerOptions listenerOptions) {
        PhysicalConnectionPool thePhysicalConnectionPool = 
            serverImpl.getPhysicalConnectionPool();
        IOQueue<PhysicalConnection> ioQueue = serverImpl.getIOQueue();

        // 内置了一个PhysicalConnection的对象池,这样可以避免每次都要
        // 创建PhysicalConnection对象,可以加快处理速度
        PhysicalConnection connection = 
            thePhysicalConnectionPool.borrowObject();
        // 把PhysicalConnection对象和socket对象关联起来
        connection.reset(socket, isObserver, listenerOptions);

        // 初始化协议,分配buffer,用来缓存解析请求时的数据
        if (!connection.setUpProtocolContexts()) {
            thePhysicalConnectionPool.returnObject(connection);
            return false;
        }

        Debug.debug("Physical Connection Created for client from: " + 
                socket.getIP());

        // 把连接注册到I/O队列中,这样就可以监听请求
        if (!ioQueue.addSocketContext(socket, connection)) {
            thePhysicalConnectionPool.returnObject(connection);
            //leave socket close to acceptor
            return false;
        }

        Debug.debug("Queue adds client from: " + socket.getIP());

        // 把创建的PhysicalConnection加入pending队列中,此时连接
        // 还不算是真正的已连接状态,要等到第一个请求到达并正确
        // 处理之后才会是已连接状态,并且会创建一个LogicalConnection
        // 和这个PhysicalConnection相关联
        addPhysicalConnection(connection);
        
        // 初始化PhysicalConnection
        serverImpl.getDispatcher().handleInitialize(connection);

        return true;
    }

ClientFactory是PhysicalConnection的管理程序,这个方法的的作用就是创建PhysicalConnection和新的socket相关联,并且把PhysicalConnection加入请求监听的I/O队列。因此来说说IOQueue。

IOQueue本身是一个接口:

public interface IOQueue<T> {

    public boolean create();

    public void free();

    // 从队列中获取事件,默认实现是带有阻塞超时的,即当没有事件
    // 时会阻塞一段时间,超时就会返回null
    public IOEvent<T> getQueuedEvent(boolean isInputEvents);

    // 注册连接,context是关联对象,类似于附件
    public boolean addSocketContext(PushClientSocket socket, T context);

    // 取消注册
    public void deleteSocketContext(PushClientSocket socket);

    // IOQueue的事件监听是一次性,这是为了防止事件在没有被处理之前,这个事件
    // 再次被捕捉到(Java的read/write事件都是这样),因此这个方法会在事件
    // 被处理后调用,再次注册。
    public boolean rearmSocketForWrite(PushClientSocket socket, T context);

    // Read事件代表的就是从客户端有数据过来
    public boolean rearmSocketForRead(PushClientSocket socket, T context);
}

IOQueueImpl是IOQueue的Java NIO版本的实现。IOQueueImpl会内置一个独立线程以及一个Selector,这里关于注册有一点需要说明:

PushClientSocketImpl的registerSelector方法用于注册socket,这里需要调用wakeup方法。因为如果独立线程会调用Selector的select方法等待新的数据,这个时候直接

调用register方法会被阻塞,因此需要先调用wakeup唤醒selector。

    public SelectionKey registerSelector(Selector selector, int ops, 
            Object attachment) throws IOException {
        selector.wakeup(); // To prevent block when calling register method
        return channel.register(selector, ops, attachment);
    }

接着说说独立线程监听事件,因为OP_WRITE的特殊性,这里只监听OP_READ事件。

    private void pollEvents(boolean isOutPoll) {
        Selector selector;
        BlockingQueue<SelectionKey> queue;
        if (isOutPoll) {
            return;
        } else {
            selector = this.inPollSelector;
            queue = this.inPollQueue;
        }
        
        List<SelectionKey> cache = new LinkedList<SelectionKey>();

        while (isPolling) {
            try {
                selector.select();
            } catch (IOException e) {
                continue;
            }
            
            // 这里调用yield释放控制权是为了刚刚提到的register方法能被顺利执行
            Thread.yield();
            
            // Add into cache (Add into the blocking queue directly
            // may block so that the selector cannot release the selection 
            // key in time)
            if (selector.isOpen()) {
                for (SelectionKey key : selector.selectedKeys()) {
                    // 前面提到监听事件是一次性的,因此这里取消监听
                    // 后面再调用rearm方法重新注册
                    key.cancel(); 
                    cache.add(key);
                }
                
                // Clear the keys
                selector.selectedKeys().clear();
                
                // 因为使用了限定长度的BlockingQueue,可能因为队列已满导致阻塞
                // 因此先把事件转移到缓存中,释放Selector
                queue.addAll(cache);
                cache.clear();
            } else {
                break; // selector closed
            }
        }
    }

顺便说说Demultiplexor获取事件调用的getQueuedEvent方法,这里使用BlockingQueue来实现阻塞等待:

    public IOEvent<PhysicalConnection> getQueuedEvent(boolean isInputEvents) {
        final IOEventType type;
        final BlockingQueue<SelectionKey> pollQueue;
        
        if (isInputEvents) {
            type = IOEventType.read;
            pollQueue = inPollQueue;
        } else {
            type = null;
            pollQueue = null;
        }
        
        if (pollQueue == null) {
            return null;
        }

        try {
            // 设置1秒的超时,这样后面关闭时清空I/O队列的时候不会导致
            // Demultiplexor一直被阻塞
            SelectionKey key = pollQueue.poll(1000L, TimeUnit.MILLISECONDS);
            if (key != null) {
                if (key.attachment() instanceof PhysicalConnection) {
                    return new IOEvent<PhysicalConnection>(type, 
                            (PhysicalConnection)(key.attachment()));
                }
            }
        } catch (InterruptedException e) {
            // Ignore
        }
        
        return null;
    }

关于新socket的处理就说这么多吧。

posted @ 2014-10-30 16:58  梧留柒  阅读(1412)  评论(1编辑  收藏  举报