1、IoSession与底层的传输层类型无关,表示通信双端的连接。提供用户自定义属性,可以用于在过滤器和处理器之间交换用户自定义协议相关信 息。每个会话都由一个Service来提供服务,同时有一个Handler负责此会话的I/O事件处理。最重要的两个方法就是read和write,这两 个方法都是异步执行,如要真正完成必须在其结果上进行等待。关闭会话的方法close也是异步执行的,也就是应等待返回的CloseFuture,此外, 还有另一种关闭方式closeOnFlush,它和close的区别是会先flush掉写请求队列中的请求数据,但同样是异步的。会话的读写类型是可配置 的,在运行中可设置此端是否可读写。
一个会话主要包括两方面的数据:属性映射图和写请求队列,这里使用工厂模式来为新创建的会话提供这些数据结构,定义如下:
public interface IoSessionDataStructureFactory { //返回属性 IoSessionAttributeMap getAttributeMap(IoSession session) throws Exception; //返回写请求队列 WriteRequestQueue getWriteRequestQueue(IoSession session) throws Exception; }
2、IoSessionConfig表示会话的配置信息,主要包括:读缓冲区大小,会话数据吞吐量,计算吞吐量的时间间隔,指定会话段的空闲时间,写请求操作超时时间等。这个里面有两个方法需要注意,如下:
/* * 只有在IoSession的read方法可用的时候返回true。 如果可用,受到的消息 * 保存在BlockingQueue这样对那个客户端应用程序来说更方便取到消息。开 * 启这个对服务器没什么好处,并且可能造成内存漏洞,默认是不开启的。 */ boolean isUseReadOperation(); /* * 打开或关闭IoSession的read方法。 */ void setUseReadOperation(boolean useReadOperation);
3、IoSessionInitializer定义了一个回调函数,用于把用户自定义的会话初始化行为剥离出来:
public interface IoSessionInitializer<T extends IoFuture> { void initializeSession(IoSession session, T future); }
4、IoSessionRecycler为一个无连接的传输服务提供回收现有会话的服务:
public interface IoSessionRecycler { /** * 一个虚假的recycler(并不回收任何session)。但是用这个可以使得所有session * 的生命周期事件被fired */ static IoSessionRecycler NOOP = new IoSessionRecycler() { public void put(IoSession session) {} public IoSession recycle(SocketAddress localAddress,SocketAddress remoteAddress) { return null; } public void remove(IoSession session) {} }; /* * 创建或写Iossion的时候被调用。 */ void put(IoSession session); /* * 尝试获取一个被回收了的IoSession。 */ IoSession recycle(SocketAddress localAddress, SocketAddress remoteAddress); /* * 会话被关闭的时候调用。 */ void remove(IoSession session); }
ExpiringSessionRecycler是IoSessionRecycler的一个实现,用来回收超时失效的会话:
private ExpiringMap<Object, IoSession> sessionMap;//待处理的会话 private ExpiringMap<Object, IoSession>.Expirer mapExpirer;//负责具体的回收工作
下面来看Key是什么样的:
private Object generateKey(SocketAddress localAddress,SocketAddress remoteAddress) { List<SocketAddress> key = new ArrayList<SocketAddress>(2); key.add(remoteAddress); key.add(localAddress); return key; }
ExpiringMap中保存了超过限制的对象和该对象的监听器,如下:
public class ExpiringMap<K, V> implements Map<K, V> { public static final int DEFAULT_TIME_TO_LIVE = 60; public static final int DEFAULT_EXPIRATION_INTERVAL = 1; private static volatile int expirerCount = 1; private final ConcurrentHashMap<K, ExpiringObject> delegate; private final CopyOnWriteArrayList<ExpirationListener<V>> expirationListeners; private final Expirer expirer; }
其中的ExpiringObject表示一个超过限制的对象,是ExpiringMap的一个内部类,如下:
private class ExpiringObject { private K key; private V value; private long lastAccessTime;//上次访问时间 private final ReadWriteLock lastAccessTimeLock = new ReentrantReadWriteLock(); }
mapExpirer用来移除sessionMap上超过临界值的项,关键代码如下:
private void processExpires() { long timeNow = System.currentTimeMillis();//当前时间 for (ExpiringObject o : delegate.values()) { if (timeToLiveMillis <= 0) { continue; } long timeIdle = timeNow - o.getLastAccessTime(); if (timeIdle >= timeToLiveMillis) { delegate.remove(o.getKey());//移除 for (ExpirationListener<V> listener : expirationListeners) { listener.expired(o.getValue());//终止监听? } } } }
启动关闭该县城都需要进行封锁机制。
5、Mina中的I/O事件类型如下:
public enum IoEventType { SESSION_CREATED, SESSION_OPENED, SESSION_CLOSED, MESSAGE_RECEIVED, MESSAGE_SENT, SESSION_IDLE, EXCEPTION_CAUGHT, WRITE, CLOSE, }
IoEvent表示一个I/O事件或者一个I/O请求,包括时间类型、所属会话、时间参数:
public class IoEvent implements Runnable { private final IoEventType type; private final IoSession session; private final Object parameter; public IoEvent(IoEventType type, IoSession session, Object parameter) { //... } //根据事件类型向会话的过滤链上的众多监听者发出事件到来的信号。 public void fire() { switch (getType()) { case MESSAGE_RECEIVED: getSession().getFilterChain().fireMessageReceived(getParameter());break; case MESSAGE_SENT: getSession().getFilterChain().fireMessageSent((WriteRequest) getParameter());break; case WRITE: getSession().getFilterChain().fireFilterWrite((WriteRequest) getParameter());break; case CLOSE: getSession().getFilterChain().fireFilterClose();break; case EXCEPTION_CAUGHT: getSession().getFilterChain().fireExceptionCaught((Throwable) getParameter());break; case SESSION_IDLE: getSession().getFilterChain().fireSessionIdle((IdleStatus) getParameter());break; case SESSION_OPENED: getSession().getFilterChain().fireSessionOpened(); break; case SESSION_CREATED: getSession().getFilterChain().fireSessionCreated(); break; case SESSION_CLOSED: getSession().getFilterChain().fireSessionClosed(); break; default: throw new IllegalArgumentException("Unknown event type: " + getType()); } } public String toString() { if (getParameter() == null) { return "[" + getSession() + "] " + getType().name(); } return "[" + getSession() + "] " + getType().name() + ": "+ getParameter(); } }
Mina的会话中,有三种类型的闲置状态:READER_IDLE读端空闲、WRITER_IDLE写端空闲、BOTH_IDLE读写都空闲。为了节省会话资源可以让用户设置当空闲超过一定时间后关闭会话,因为此会话可能在一段出现问题,从而导致另一端空闲超过太长时间。
6、DefaultIoSessionDataStructureFactory是IoSessionDataStructureFactory的一个默认的实现:
public class DefaultIoSessionDataStructureFactory implements IoSessionDataStructureFactory { public IoSessionAttributeMap getAttributeMap(IoSession session) throws Exception { return new DefaultIoSessionAttributeMap(); } public WriteRequestQueue getWriteRequestQueue(IoSession session) throws Exception { return new DefaultWriteRequestQueue(); } private static class DefaultIoSessionAttributeMap implements IoSessionAttributeMap { private final Map<Object, Object> attributes = Collections.synchronizedMap(new HashMap<Object, Object>(4)); public DefaultIoSessionAttributeMap() { super(); } public Object getAttribute(IoSession session, Object key, Object defaultValue) { if (key == null) { throw new NullPointerException("key"); } Object answer = attributes.get(key); if (answer == null) { return defaultValue; } return answer; } public Object setAttribute(IoSession session, Object key, Object value) { if (key == null) { throw new NullPointerException("key"); } if (value == null) { return attributes.remove(key); } return attributes.put(key, value); } public Object setAttributeIfAbsent(IoSession session, Object key, Object value) { if (key == null) { throw new NullPointerException("key"); } if (value == null) { return null; } Object oldValue; synchronized (attributes) { oldValue = attributes.get(key); if (oldValue == null) { attributes.put(key, value); } } return oldValue; } public Object removeAttribute(IoSession session, Object key) { if (key == null) { throw new NullPointerException("key"); } return attributes.remove(key); } public boolean removeAttribute(IoSession session, Object key, Object value) { if (key == null) { throw new NullPointerException("key"); } if (value == null) { return false; } synchronized (attributes) { if (value.equals(attributes.get(key))) { attributes.remove(key); return true; } } return false; } public boolean replaceAttribute(IoSession session, Object key, Object oldValue, Object newValue) { synchronized (attributes) { Object actualOldValue = attributes.get(key); if (actualOldValue == null) { return false; } if (actualOldValue.equals(oldValue)) { attributes.put(key, newValue); return true; } return false; } } public boolean containsAttribute(IoSession session, Object key) { return attributes.containsKey(key); } public Set<Object> getAttributeKeys(IoSession session) { synchronized (attributes) { return new HashSet<Object>(attributes.keySet()); } } public void dispose(IoSession session) throws Exception {} } private static class DefaultWriteRequestQueue implements WriteRequestQueue { //一个队列存储传入的写请求 private final Queue<WriteRequest> q = new CircularQueue<WriteRequest>(16); public DefaultWriteRequestQueue() { super(); } //... } }
7、AbstractIoSession是IoSession的一个抽象实现类,如下:
private IoSessionAttributeMap attributes;//会话属性 private WriteRequestQueue writeRequestQueue;//写请求队列 private WriteRequest currentWriteRequest;//当前写请求
下面是关闭的时候涉及的成员:
//要结束当前会话时发送写请求CLOSE_REQUEST private static final WriteRequest CLOSE_REQUEST = new DefaultWriteRequest(new Object()); //在连接关闭时状态被设置为closed private final CloseFuture closeFuture = new DefaultCloseFuture(this); //关闭的监听器 private static final IoFutureListener<CloseFuture> SCHEDULED_COUNTER_RESETTER = new IoFutureListener<CloseFuture>() { public void operationComplete(CloseFuture future) { AbstractIoSession session = (AbstractIoSession) future.getSession(); session.scheduledWriteBytes.set(0); session.scheduledWriteMessages.set(0); session.readBytesThroughput = 0; session.readMessagesThroughput = 0; session.writtenBytesThroughput = 0; session.writtenMessagesThroughput = 0; } };
close和closeOnFlush都是异步操作的,区别是前者立即关闭连接,后者是在写请求队列中放入一个CLOSE_REQUEST并将其即时刷新储蓄,若要真正等到关闭完成,需要调用方法在返回的CloseFuture等待。下面是close的代码:
public final CloseFuture close() { synchronized (lock) { if (isClosing()) { return closeFuture; } closing = true; } getFilterChain().fireFilterClose();//fire出关闭事件 return closeFuture; }
下面是closeOnFlush的代码:
private final CloseFuture closeOnFlush() { getWriteRequestQueue().offer(this, CLOSE_REQUEST); getProcessor().flush(this); return closeFuture; }
对于读的情况,下面是取得读的数据队列:
//返回可被读取数据队列 private Queue<ReadFuture> getReadyReadFutures() { Queue<ReadFuture> readyReadFutures = (Queue<ReadFuture>) getAttribute(READY_READ_FUTURES_KEY); //如果是第一次读取数据 if (readyReadFutures == null) { //构造一个新的读数据队列 readyReadFutures = new CircularQueue<ReadFuture>(); Queue<ReadFuture> oldReadyReadFutures = (Queue<ReadFuture>) setAttributeIfAbsent(READY_READ_FUTURES_KEY, readyReadFutures); if (oldReadyReadFutures != null) { readyReadFutures = oldReadyReadFutures; } } return readyReadFutures; }
读数据的过程:
public final ReadFuture read() { //配置不允许读数据 if (!getConfig().isUseReadOperation()) { throw new IllegalStateException("useReadOperation is not enabled."); } //获得已经被许可的可被读数据队列 Queue<ReadFuture> readyReadFutures = getReadyReadFutures(); ReadFuture future; synchronized (readyReadFutures) { future = readyReadFutures.poll(); if (future != null) { //如果关联的会话已关闭,通知读者 if (future.isClosed()) { readyReadFutures.offer(future); } } else { future = new DefaultReadFuture(this); //将数据出入等待读的队列 getWaitingReadFutures().offer(future); } } return future; }
写数据的过程:
//IoBuffer、文件、文件部分区域 public WriteFuture write(Object message, SocketAddress remoteAddress) { if (message == null) { throw new NullPointerException("message"); } //如果没有远端地址 if (!getTransportMetadata().isConnectionless() && remoteAddress != null) { throw new UnsupportedOperationException(); } //如果会话已被关闭 if (isClosing() || !isConnected()) { WriteFuture future = new DefaultWriteFuture(this); WriteRequest request = new DefaultWriteRequest(message, future, remoteAddress); WriteException writeException = new WriteToClosedSessionException(request); future.setException(writeException); return future; } FileChannel openedFileChannel = null; try { if (message instanceof IoBuffer && !((IoBuffer) message).hasRemaining()) {//如果是空消息 throw new IllegalArgumentException("message is empty. Forgot to call flip()?"); } else if (message instanceof FileChannel) {//文件的某一区域 FileChannel fileChannel = (FileChannel) message; message = new DefaultFileRegion(fileChannel, 0, fileChannel.size()); } else if (message instanceof File) {//要发送的是文件 File file = (File) message; openedFileChannel = new FileInputStream(file).getChannel();//打开文件通道 message = new DefaultFileRegion(openedFileChannel, 0, openedFileChannel.size()); } } catch (IOException e) { ExceptionMonitor.getInstance().exceptionCaught(e); return DefaultWriteFuture.newNotWrittenFuture(this, e); } //构造写请求,通过过滤器链发送出去 WriteFuture writeFuture = new DefaultWriteFuture(this); WriteRequest writeRequest = new DefaultWriteRequest(message, writeFuture, remoteAddress); IoFilterChain filterChain = getFilterChain(); filterChain.fireFilterWrite(writeRequest); //如果打开文件通道应该在完成时关闭通道 if (openedFileChannel != null) { final FileChannel finalChannel = openedFileChannel; writeFuture.addListener(new IoFutureListener<WriteFuture>() { public void operationComplete(WriteFuture future) { try { finalChannel.close(); } catch (IOException e) { ExceptionMonitor.getInstance().exceptionCaught(e); } } }); } return writeFuture; }