`
eoasis
  • 浏览: 76759 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

netty和mina Socket框架(转)

 
阅读更多
(转自:http://blog.163.com/chen_scor/blog/static/178805061201182615318929/)

比较:http://sandaobusi.iteye.com/blog/961175

从Jetty、Tomcat和Mina中提炼NIO构架Server的经典模式:

从Jetty、Tomcat和Mina中提炼NIO构架网络服务器的经典模式(一)
从Jetty、Tomcat和Mina中提炼NIO构架网络服务器的经典模式(二)
从Jetty、Tomcat和Mina中提炼NIO构架网络服务器的经典模式(三)

0.mina比netty出现的早,都是Trustin Lee的作品;


1.mina将内核和一些特性的联系过于紧密,使得用户在不需要这些特性的时候无法脱离,相比下性能会有所下降;netty解决了这个设计问题;


2.netty的文档更清晰,很多mina的特性在netty里都有;


3.netty更新周期更短,新版本的发布比较快;


4.它们的架构差别不大,mina靠apache生存,而netty靠jboss,和jboss的结合度非常高,netty有对google protocal buf的支持,有更完整的ioc容器支持(spring,guice,jbossmc和osgi);


5.netty比mina使用起来更简单,netty里你可以自定义的处理upstream events 或/和 downstream events,可以使用decoder和encoder来解码和编码发送内容;


6.netty和mina在处理UDP时有一些不同,netty将UDP无连接的特性暴露出来;而mina对UDP进行了高级层次的抽象,可以把UDP当成"面向连接"的协议,而要netty做到这一点比较困难。



运用spring注解实现netty服务器端udp应用程序

http://blog.csdn.net/tanrenzong1986/article/details/6404256

  netty是JBOSS针对网络开发的一套应用框架,它也是在NIO的基础上发展起来的。netty基于异步的事件驱动,具有高性能、高扩展性等特性,它提供了统一的底层协议接口,使得开发者从底层的网络协议(比如 TCP/IP、UDP)中解脱出来。就使用来说,开发者只要参考 Netty提供的若干例子和它的指南文档,就可以放手开发基于Netty的服务端程序了。

   netty有几个比较重要的概念,在此,仅做介绍,详细可以参考netty文档或源码。

   1). channelBuffer: Bootstrap 是一个设置服务的帮助类。你甚至可以在这个服务中直接设置一个 Channel 通道。

  现在以实现一个UDP协议下的服务器应用程序为例,演示netty通过spring注解开发服务器端。(在此以maven工具管理项目开发)

  首先,是导入关联jar的POM文件:


+ expand sourceview plaincopy to clipboardprint?
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
  xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> 
  <modelVersion>4.0.0</modelVersion> 
  <groupId>com.byd.example</groupId> 
  <artifactId>nettyTest</artifactId> 
  <version>0.0.1-SNAPSHOT</version> 
  <packaging>jar</packaging> 
  <name>nettyTest</name> 
  <url>http://maven.apache.org</url> 
  <properties> 
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> 
  </properties> 
  <dependencies> 
    <dependency> 
      <groupId>junit</groupId> 
      <artifactId>junit</artifactId> 
      <version>3.8.1</version> 
      <scope>test</scope> 
    </dependency> 
    <dependency> 
     <groupId>slf4j</groupId> 
     <artifactId>slf4j-jdk14</artifactId>    
     <version>1.6.1</version> 
    </dependency> 
    <dependency> 
     <groupId>log4j</groupId> 
     <artifactId>log4j</artifactId>   
     <version>1.2.9</version>  
    </dependency> 
    <dependency> 
     <groupId>org.jboss.netty</groupId>    
     <artifactId>netty</artifactId> 
     <version>3.2.0.BETA1</version> 
    </dependency> 
    <dependency> 
     <groupId>org.springframework</groupId>  
     <artifactId>spring-context</artifactId>   
     <version>3.0.2.RELEASE</version> 
    </dependency> 
    <dependency> 
     <groupId>org.springframework</groupId>  
     <artifactId>spring-expression</artifactId>   
     <version>3.0.2.RELEASE</version> 
    </dependency> 
    <dependency> 
     <groupId>org.springframework</groupId>  
     <artifactId>spring-tx</artifactId>   
     <version>3.0.2.RELEASE</version> 
    </dependency> 
    <dependency> 
     <groupId>org.springframework</groupId>  
     <artifactId>spring-context</artifactId>   
     <version>3.0.2.RELEASE</version> 
    </dependency> 
  </dependencies> 
  <repositories>   
     <repository>   
       <id>repository.jboss.org</id>   
          <url>http://repository.jboss.org/nexus/content/groups/public/</url>   
         <snapshots>   
           <enabled>false</enabled>   
         </snapshots>   
       </repository>   
    </repositories>   
    <build> 
    
    </build> 
</project> 

接下来是应用程序代码,在本人机器上可以运行并正常工作,如有需要可以试试。

I、Server接口:


view plaincopy to clipboardprint?
package com.byd.example.netty; 
public interface IServer  

    /**
     * 启动服务器
     */ 
    public void start(); 
    /**
     * 重启程序
     */ 
    public void restart(); 
     
    /**
     * 停止程序运行
     */ 
    public void stop(); 
}  

II、ChannelHandler扩张类(继承SimpleChannelHandler):




view plaincopy to clipboardprint?
package com.byd.example.netty; 
import java.util.Random; 
import org.apache.log4j.Logger; 
import org.jboss.netty.buffer.ChannelBuffer; 
import org.jboss.netty.buffer.DynamicChannelBuffer; 
import org.jboss.netty.channel.ChannelFutureListener; 
import org.jboss.netty.channel.ChannelHandlerContext; 
import org.jboss.netty.channel.ExceptionEvent; 
import org.jboss.netty.channel.MessageEvent; 
import org.jboss.netty.channel.SimpleChannelHandler; 
import org.springframework.stereotype.Component; 
@Component("receiverHandler") 
public class ReceiverHandler extends SimpleChannelHandler 

    private static final Logger     logger=Logger.getLogger(ReceiverHandler.class.getName()); 
     
    @Override 
    public void messageReceived(ChannelHandlerContext ctx,MessageEvent e) throws Exception 
    { 
        ChannelBuffer buffer=(ChannelBuffer)e.getMessage(); 
        byte[] recByte=buffer.copy().toByteBuffer().array(); 
        String recMsg=new String(recByte); 
        logger.info("server received:"+recMsg.trim()); 
        Random random=new Random(); 
        int backWord=random.nextInt(10000); 
        ChannelBuffer responseBuffer=new DynamicChannelBuffer(4); 
        responseBuffer.readBytes(backWord); 
        e.getChannel().write(responseBuffer); 
    } 
     
    @Override 
    public void exceptionCaught(ChannelHandlerContext ctx,ExceptionEvent e) 
    { 
        logger.error(e.getCause()); 
        if(e.getChannel() !=null) 
        { 
            e.getChannel().close().addListener(ChannelFutureListener.CLOSE); 
        } 
    } 
}  

III、ChannelPipelineFactory实现类,包装ChannelHandler,处理I/O事件。


view plaincopy to clipboardprint?
package com.byd.example.netty; 
import org.jboss.netty.channel.ChannelPipeline; 
import org.jboss.netty.channel.ChannelPipelineFactory; 
import org.jboss.netty.channel.Channels; 
import org.springframework.beans.factory.annotation.Autowired; 
import org.springframework.beans.factory.annotation.Qualifier; 
import org.springframework.stereotype.Component; 
@Component("serverChannelPipelineFactory") 
public class ServerChannelPipelineFactory implements ChannelPipelineFactory 

    @Autowired 
    @Qualifier("receiverHandler") 
    private ReceiverHandler handler; 
    @Override 
    public ChannelPipeline getPipeline() throws Exception  
    { 
        ChannelPipeline pipeline=Channels.pipeline(); 
        pipeline.addLast("handler", this.handler); 
        return pipeline; 
    } 
    public void setHandler(ReceiverHandler handler) { 
        this.handler = handler; 
    } 


IV、Iserver接口的实现类。


+ expand sourceview plaincopy to clipboardprint?
package com.byd.example.netty; 
import java.net.InetSocketAddress; 
import java.net.SocketAddress; 
import java.util.concurrent.Executors; 
import org.apache.log4j.Logger; 
import org.jboss.netty.bootstrap.ConnectionlessBootstrap; 
import org.jboss.netty.channel.Channel; 
import org.jboss.netty.channel.ChannelFutureListener; 
import org.jboss.netty.channel.socket.DatagramChannelFactory; 
import org.jboss.netty.channel.socket.nio.NioDatagramChannelFactory; 
import org.springframework.beans.factory.annotation.Autowired; 
import org.springframework.beans.factory.annotation.Qualifier; 
import org.springframework.stereotype.Component; 
@Component("serverNettyImpl") 
public class ServerNettyImpl implements IServer { 
    @Autowired 
    @Qualifier("serverChannelPipelineFactory") 
    private ServerChannelPipelineFactory pipelineFactory; 
    private Channel channel; 
    private static final Logger logger = Logger.getLogger(ServerNettyImpl.class 
            .getName()); 
    @Override 
    public void start() { 
        DatagramChannelFactory udpChannelFactory = new NioDatagramChannelFactory( 
                Executors.newCachedThreadPool()); 
        ConnectionlessBootstrap bootstrap = new ConnectionlessBootstrap(udpChannelFactory); 
        bootstrap.setOption("reuseAddress", false); 
        bootstrap.setOption("child.reuseAddress", false); 
        bootstrap.setOption("readBufferSize", 1024); 
        bootstrap.setOption("writeBufferSize", 1024); 
        bootstrap.setPipelineFactory(this.pipelineFactory); 
        SocketAddress serverAddress = new InetSocketAddress(5000); 
        this.channel = bootstrap.bind(serverAddress); 
        logger.info("server start on " + serverAddress); 
    } 
    @Override 
    public void restart() { 
        this.stop(); 
        this.start(); 
    } 
    @Override 
    public void stop() { 
        if (this.channel != null) { 
            this.channel.close().addListener(ChannelFutureListener.CLOSE); 
        } 
    } 
}  

V、应用程序入口:


+ expand sourceview plaincopy to clipboardprint?
package com.byd.example; 
import org.springframework.context.ApplicationContext; 
import org.springframework.context.support.ClassPathXmlApplicationContext; 
import com.byd.example.netty.IServer; 
public class NettyTestRun  

    public static void main( String[] args ) 
    { 
        ApplicationContext context=new ClassPathXmlApplicationContext("classpath*:nettyTest-context.xml"); 
        IServer server=(IServer)context.getBean("serverNettyImpl"); 
        server.start(); 
    } 


VI、context配置文件:


+ expand sourceview plaincopy to clipboardprint?
<?xml version="1.0" encoding="UTF-8" ?> 
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
    xmlns:p="http://www.springframework.org/schema/p" xmlns:aop="http://www.springframework.org/schema/aop" 
    xmlns:tx="http://www.springframework.org/schema/tx" xmlns:context="http://www.springframework.org/schema/context" 
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd 
    http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd 
    http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd 
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> 
    <context:component-scan base-package="com.byd.example"/> 
    <context:annotation-config/> 
</beans> 

VII、日志配置文件:


+ expand sourceview plaincopy to clipboardprint?
<?xml version="1.0" encoding="UTF-8"?> 
<!DOCTYPE log4j:configuration SYSTEM "log4j.dtd" > 
<log4j:configuration xmlns:log4j='http://jakarta.apache.org/log4j/' >       
    <appender name="CONSOLE" class="org.apache.log4j.ConsoleAppender"> 
    <param name="target" value="System.out"/> 
    <layout class="org.apache.log4j.PatternLayout"> 
        <param name="ConversionPattern" value="%d{yyyy-MM-dd HH:mm:ss}-%m%n" /> 
    </layout> 
    <!--过滤器设置输出的级别--> 
    <filter class="org.apache.log4j.varia.LevelRangeFilter"> 
        <param name="levelMin" value="debug" /> 
        <param name="levelMax" value="warn" /> 
        <param name="AcceptOnMatch" value="true" /> 
    </filter> 
</appender>            
    <appender name="FILE" class="org.apache.log4j.RollingFileAppender">    
        <!-- 设置日志输出文件名 -->       
        <param name="File" value="./target/output.log" />   
        <!-- 设置是否在重新启动服务时,在原有日志的基础添加新日志 -->     
        <param name="Append" value="true" />     
        <param name="MaxBackupIndex" value="10" />     
        <param name="encoding" value="UTF-8"/> 
        <layout class="org.apache.log4j.PatternLayout">     
            <param name="ConversionPattern" value="%d{yyyy-MM-dd HH:mm:ss}-%m%n" />     
        </layout>     
    </appender>     
    <appender name="activexAppender" class="org.apache.log4j.DailyRollingFileAppender">     
        <param name="File" value="./target/activex.log" />       
        <param name="DatePattern" value="'.'yyyy-MM-dd'.log'" />       
        <layout class="org.apache.log4j.PatternLayout">     
         <param name="ConversionPattern" value="%d{yyyy-MM-dd HH:mm:ss}-%m%n" />     
        </layout>       
    </appender>     
           
<!-- 
    <logger name="com.runway.bssp.activeXdemo" additivity="false">     
        <priority value ="info"/>  
        <param name="Level" value="DEBUG"/>      
        <appender-ref ref="activexAppender" />       
    </logger> --> 
    
     
    <!-- 根logger的设置-->     
    <root>     
        <priority value ="debug"/>     
        <appender-ref ref="CONSOLE"/>     
        <appender-ref ref="FILE"/>        
    </root>     
</log4j:configuration>    

整理上面的代码,运行起来,可以看到服务器侦听在5000端口并接受客户端的信息并返回一个int随机数。

后记:netty当然也可以应用在TCP和客户端的程序,具体的使用可以参考netty API文档。
分享到:
评论
1 楼 asialee 2013-02-20  
博主这块写的挺好的,我完全看了,不过我也写了一些关于编解码器的,请批评指正http://asialee.iteye.com/blog/1769508

相关推荐

    MinaServer:学习使用mina框架

    改用网络通信框架、目前比较流行的又netty、mina;由于主要作者是同一个人所以差别不是很大、这边选择mina学习; minaserver为采用mina搭建的基本服务器;这边使用了自带的简单TextLineCodesFactory;也自定义了了一...

    轻量级网络通信框架nSocket.zip

    nSocket与mina、netty、xSocket这类高性能的设计不同之处在于,nSocket将直接使用java7中的异步通道构建异步操作,并在后期将弱化C/S模式,逐步过渡到P2P模式。 编写目的: 简化NIO开发过程,将业务与实现...

    Netty原理和使用

    支持HTTP、WebSocket、Protobuf、BinaryTCP|和UDP,Netty已经被很多高性能项目作为其Socket底层基础,如HornetQInfinispanVert.xPlayFrameworkFinangle和Cassandra。其竞争对手是:ApacheMINA和Grizzly。传统堵塞的...

    Java Netty技术研究

    提供异步的、事件驱动网络应用程序框架和工具 优点:提供异步的、事件驱动网络应用程序框架和工具 优点:提供异步的、事件驱动网络应用程序框架和工具 优点:提供异步的、事件驱动网络应用程序框架和工具 优点:提供...

    Hprose 全名是高性能远程对象服务引擎.rar

    基于 TCP 协议的(通常会借助 Mina、Netty 等高性能网络框架) RPC (远程过程调用) 是什么 简单的说,RPC 就是从一台机器(客户端)上通过参数传递的方式调用另一台机器(服务器)上的一个函数或方法(可以统称为...

    一个Java码农的Node之旅

    3.长连接吧方面是WebSocket标准,亦或Netty、Mina系的框架。4.连接断了需要实现下重连机制吧?服务器端写完了还有客户端,也需要帮助指导下实现,问题诸多。如果有现成的轮子偷懒的话,socket.io比较适合:1.轻量级...

    JAVA上百实例源码以及开源项目

     Java绘制图片火焰效果,源代码相关注释:前景和背景Image对象、Applet和绘制火焰的效果的Image对象、Applet和绘制火焰的效果的Graphics对象、火焰效果的线程、Applet的高度,图片到图片装载器、绘制火焰效果的X坐标...

    JAVA上百实例源码以及开源项目源代码

    在有状态SessionBean中,用累加器,以对话状态存储起来,创建EJB对象,并将当前的计数器初始化,调用每一个EJB对象的count()方法,保证Bean正常被激活和钝化,EJB对象是用完毕,从内存中清除…… Java Socket 聊天...

Global site tag (gtag.js) - Google Analytics