设为首页 收藏本站
查看: 876|回复: 0

[经验分享] Apache Mina 文档翻译

[复制链接]

尚未签到

发表于 2017-1-12 08:40:14 | 显示全部楼层 |阅读模式
第五章 - 过滤器
  IoFilter是MINA的重要组件之一, 它可以过滤在IoService和IoHandler的所有IO事件和请求。 如果你有Web应用程序开发经验, 你会发现它和Servlet的过滤器十分相似。MINA提供了很多可以直接使用的过滤器,这大大简化了网络应用程序的开发,例如:

    LoggingFilter:记录所有事件和请求
    ProtocolCodecFilter:把接收到的ByteBuffer里的数据转换成Java对象,会反之。      CompressionFilter:压缩数据
    SSLFilter: 添加SSL - TLS - StartTLS 支持.
    还有很多!

在本章中我们通过一个现实世界的例子来学习如何实现一个过滤器。实现一个通用的过滤器很容易,但最好还是了解一下其中的原理。所以我们还会介绍其内部的属性。



既存的过滤器

过滤器 类 描述
BlacklistBlacklistFilter阻止黑名单中的地址访问服务器
Buffered WriteBufferedWriteFilter像BufferedOutputStream一样处理发送的请求
CompressionCompressionFilter压缩输入输出的数据
ConnectionThrottleConnectionThrottleFilter阻止过度频繁的访问
ErrorGeneratingErrorGeneratingFilter随机的向输入输出流中插入一些字节照成输入输出错误。
ExecutorExecutorFilter把IO事件处理指派到指定的线程模型
FileRegionWriteFileRegionWriteFilter把FileRegion对象写到ByteBuffer
KeepAliveKeepAliveFilter在会话空闲时发送一个keep-alive请求或当客户端有keep-alive请求发过来是给予相应。
LoggingLoggingFilter把事件的消息写入日志,例如接收到消息, 发送消息, 会话建立
MDCMdcInjectionFilter把IoSession属性添加到MDC
NoopNoopFilter什么都不做的过滤器,一般用于测试
ProfilerProfilerTimerFilter对收发消息,会话建立等事件进行Profile。
ProtocolCodecProtocolCodecFilter编码解码消息
ProxyProxyFilter当使用ProxyConnector时自己被放入过滤器链,发送初始握手消息和响应后续的握手消息
Reference countingReferenceCountingFilter可以记录当前过滤器被引用了多次
RequestResponseRequestResponseFilter实现请求-响应模式的过滤器
SessionAttributeInitializingSessionAttributeInitializingFilter当会话建立是初始化其属性
StreamWriteStreamWriteFilter当把InputStream之间传给Session.write方法时,把InputStream里的内容直接发送给对方。
SslFilterSslFilter添加SSL - TLS - StartTLS 支持
WriteRequestWriteRequestFilter一个抽象类用来方便IoFilter实现
   
   

有选择的重写事件
  你可以继承IoAdapter而不是直接实现IoFilter接口。除非你重写了某个方法,对应的事件会被直接指派到下一个过滤器。



public class MyFilter extends IoFilterAdapter {
@Override
public void sessionOpened(NextFilter nextFilter, IoSession session) throws Exception {
// Some logic here...
nextFilter.sessionOpened(session);
// Some other logic here...
}
}

转换写请求
  如果你想通过IoSession.write()对请求进行转换,这需要点小技巧。例如你想在IoSession.write()传递HighLevelMessage时把HighLevelMessage转换成LowLevelMessage。你可以把转换代码放在过滤器的filterWrite(),认为这就OK了,其实你还需要重写messageSent事件处理,因为IoHandler和你的过滤器后面的其他任何过滤器会希望messageSent被调用时传递的是HighLevelMessage。这是因为如果调用者传递的是HighLevelMessage但是实际发送的是LowLevelMessage是不合理的。所以你在处理转换时必须同时重写filterWrite()和messageSent()方法。

还需要注意的是即使你的输入和输出类型是一致的情况下你也需要实现同样的机制(例如CompressionFilter)。因为IoSession.write()的调用者需要它的messageSent()方法被调用时得到的类型是完全一致的。

假设你有一个过滤器可以把字符串转换为char数组。你的过滤器的filterWrite()方法是这样:



public void filterWrite(NextFilter nextFilter, IoSession session, WriteRequest request) {
nextFilter.filterWrite(
session, new DefaultWriteRequest(
((String) request.getMessage()).toCharArray(), request.getFuture(), request.getDestination()));
}
  

那么你的messageSent()就应该是反过来的:



public void messageSent(NextFilter nextFilter, IoSession session, Object message) {
nextFilter.messageSent(session, new String((char[]) message));
}
  

那么把字符串转换为ByteBuffer时呢?这种情况下会更有效率一些,因为我们不需要重建原始消息(String)。但是看上去要比前面的例子复杂一些。



public void filterWrite(NextFilter nextFilter, IoSession session, WriteRequest request) {
String m = (String) request.getMessage();
ByteBuffer newBuffer = new MyByteBuffer(m, ByteBuffer.wrap(m.getBytes());
nextFilter.filterWrite(
session, new WriteRequest(newBuffer, request.getFuture(), request.getDestination()));
}
public void messageSent(NextFilter nextFilter, IoSession session, Object message) {
if (message instanceof MyByteBuffer) {
nextFilter.messageSent(session, ((MyByteBuffer) message).originalValue);
} else {
nextFilter.messageSent(session, message);
}
}
private static class MyByteBuffer extends ByteBufferProxy {
private final Object originalValue;
private MyByteBuffer(Object originalValue, ByteBuffer encodedValue) {
super(encodedValue);
this.originalValue = originalValue;
}
}

当处理sessionCreated事件时要小心
  建立会话是一个特殊的事件,必须在I/O processor的线程里进行。千万不要把sessionCreated的处理放到其他的线程里。



public void sessionCreated(NextFilter nextFilter, IoSession session) throws Exception {
// ...
nextFilter.sessionCreated(session);
}
// DON'T DO THIS!
public void sessionCreated(final NextFilter nextFilter, final IoSession session) throws Exception {
Executor executor = ...;
executor.execute(new Runnable() {
nextFilter.sessionCreated(session);
});
}


小心空的缓冲区
  MINA在内部使用空的Buffer来标识几种情况。 但空Buffer有些时候会造成一些问题,比如IndexOutOfBoundsException。 下面介绍如何避免这些情况。

ProtocolCodecFilter利用空Buffer(buf.hasRemaining() = 0)来表示消息的结束。 如果你的过滤器被放到ProtocolCodecFilter之前,并且在Buffer是空的时候会抛出异常。 请保证把这个空Buffer指派到下一个过滤器。



public void messageSent(NextFilter nextFilter, IoSession session, Object message) {
if (message instanceof ByteBuffer && !((ByteBuffer) message).hasRemaining()) {
nextFilter.messageSent(nextFilter, session, message);
return;
}
...
}
public void filterWrite(NextFilter nextFilter, IoSession session, WriteRequest request) {
Object message = request.getMessage();
if (message instanceof ByteBuffer && !((ByteBuffer) message).hasRemaining()) {
nextFilter.filterWrite(nextFilter, session, request);
return;
}
...
}
  

我们需要在每一个过滤器里又加入上面的if语句吗? 幸运的是我们不必。这里有一个针对于空Buffer的黄金法则:

    如果即时Buffer为空,你的过滤器也能正常工作就不用添加上面的if语句。
    如果你的过滤器在ProtocolCodecFilter之后被添加, 就不用添加上面的if语句。
    其他情况需要添加。

即使你需要上面这样的if语句,也不用和上面一模一样, 你可以自己定义如何检查空Buffer,只要你能保证你的Filter不会抛出非期望的异常。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-327234-1-1.html 上篇帖子: byte[] 转 int int 转byte apache mina源码 下篇帖子: Apache httpd 2.4.4 与老版本的差异
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表