Grizzly开发Echoserver实战

作者:chszs,转载需注明。

博客主页:http://blog.csdn.net/chszs

用Java编写可伸缩的server应用是有难度的。用Java NIO开发、线程管理、为成千上万的用户做server扩展。这些都是难点。Grizzly NIO框架的设计目标就是帮助开发人员非常好地利用Java NIO API。编写出高可扩展性的、功能强大的server,并提高了扩展的框架组件:Web Framework(HTTP/S)、WebSocket、Comet等。

Grizzly 2.3开发Echoserver/client的样例

1、下载grizzly-framework.jar库

Maven依赖

<dependency>
<groupId>org.glassfish.grizzly</groupId>
<artifactId>grizzly-framework</artifactId>
<version>2.3.16</version>
</dependency>

或下载地址: http://central.maven.org/maven2/org/glassfish/grizzly/grizzly-framework/2.3.16/grizzly-framework-2.3.16.jar
2014.10.23日刚公布了2.3.17版,地址见: 
https://maven.java.net/content/repositories/releases/org/glassfish/grizzly/grizzly-framework/2.3.17/grizzly-framework-2.3.17.jar
也能够用这个版本号。

server端:

1)创建Echo过滤器
Echo过滤器负责把接收到的消息(无论其类型)原样返回给Grizzly连接。

import java.io.IOException;

import org.glassfish.grizzly.filterchain.BaseFilter;
import org.glassfish.grizzly.filterchain.FilterChainContext;
import org.glassfish.grizzly.filterchain.NextAction; public class EchoFilter extends BaseFilter{ /**
* 仅处理读操作,当消息到来时进行处理
* @param ctx 处理的上下文
* @return 下一个动作
*/
@Override
public NextAction handleRead(FilterChainContext ctx) throws IOException{
// Peer address用于无连接的UDP连接
final Object peerAddress = ctx.getAddress();
final Object message = ctx.getMessage();
System.out.println("Server received: " + message);
ctx.write(peerAddress, message, null);
return ctx.getStopAction();
}
}

2)server初始化代码
全部的server过滤器链都准备好,開始初始化并启动server。

import java.io.IOException;
import java.nio.charset.Charset;
import java.util.logging.Logger; import org.glassfish.grizzly.filterchain.FilterChainBuilder;
import org.glassfish.grizzly.filterchain.TransportFilter;
import org.glassfish.grizzly.nio.transport.TCPNIOTransport;
import org.glassfish.grizzly.nio.transport.TCPNIOTransportBuilder;
import org.glassfish.grizzly.utils.StringFilter; public class EchoServer {
private static final Logger logger = Logger.getLogger(EchoServer.class.getName());
public static final String HOST = "localhost";
public static final int PORT = 7777; public static void main(String[] args) throws IOException{
// 用FilterChainBuilder创建过滤器链
FilterChainBuilder filterChainBuilder = FilterChainBuilder.stateless(); // 加入TransportFilter。它负责从连接中读数据,并写数据到连接
filterChainBuilder.add(new TransportFilter());
// 字符串过滤器StringFilter负责缓冲和字符串之间的转换
filterChainBuilder.add(new StringFilter(Charset.forName("UTF-8")));
// 过滤器EchoFilter负责把接收到的消息原样返回给连接
filterChainBuilder.add(new EchoFilter()); // 创建TCP传输
final TCPNIOTransport transport = TCPNIOTransportBuilder.newInstance().build();
transport.setProcessor(filterChainBuilder.build());
try{
// 绑定传输,開始对主机+端口进行监听
transport.bind(HOST, PORT);
// 開始传输
transport.start(); logger.info("Press any key to stop the Echo server...");
System.in.read();
} finally{
logger.info("Stopping transport...");
// 停止传输server
transport.shutdown(); logger.info("Stopped transport...");
}
}
}

执行Echoserver:

java -classpath grizzly-framework.jar EchoServer

client:

1)创建client过滤器
client过滤器负责重定向server的响应到标准输出。

要注意,client过滤器须要FilterChainContext消息

import java.io.IOException;

import org.glassfish.grizzly.filterchain.BaseFilter;
import org.glassfish.grizzly.filterchain.FilterChainContext;
import org.glassfish.grizzly.filterchain.NextAction; public class ClientFilter extends BaseFilter{ /**
* 仅处理读操作。当消息到来时进行处理
* @param ctx 处理的上下文
* @return 下一个动作
*/
@Override
public NextAction handleRead(final FilterChainContext ctx) throws IOException{
// 从上下文得到字符串消息,过滤器链仅仅使用了字符串过滤器StringFilter
final String serverResponse = ctx.getMessage();
System.out.println("Server echo: " + serverResponse);
return ctx.getStopAction();
}
}

2)client代码
简单的client,向Echoserver发送消息并等待响应。

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.nio.charset.Charset;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.logging.Logger; import org.glassfish.grizzly.Connection;
import org.glassfish.grizzly.Grizzly;
import org.glassfish.grizzly.filterchain.FilterChainBuilder;
import org.glassfish.grizzly.filterchain.TransportFilter;
import org.glassfish.grizzly.nio.transport.TCPNIOTransport;
import org.glassfish.grizzly.nio.transport.TCPNIOTransportBuilder;
import org.glassfish.grizzly.utils.StringFilter; import ch.echo.server.EchoServer; public class EchoClient {
private static final Logger logger = Grizzly.logger(EchoClient.class); public static void main(String[] args) throws IOException,
ExecutionException, InterruptedException, TimeoutException{
Connection connection = null; // 用FilterChainBuilder类创建过滤器链
FilterChainBuilder filterChainBuilder = FilterChainBuilder.stateless();
// 加入传输过滤器,它负责从连接读数据并向连接写数据
filterChainBuilder.add(new TransportFilter());
// 加入字符串过滤器,它负责缓冲和字符串之间的转换
filterChainBuilder.add(new StringFilter(Charset.forName("UTF-8")));
// 加入client过滤器,他负责把server响应重定向到标准输出
filterChainBuilder.add(new ClientFilter()); // 创建TCP传输
final TCPNIOTransport transport = TCPNIOTransportBuilder.newInstance().build();
transport.setProcessor(filterChainBuilder.build()); try{
// 启动传输
transport.start(); // 异步执行。连接到server
Future<Connection> future = transport.connect(EchoServer.HOST, EchoServer.PORT);
// 等待连接操作的完毕
connection = future.get(10, TimeUnit.SECONDS); assert connection!=null;
System.out.println("Ready...(\"q\" to exit");
final BufferedReader inReader = new BufferedReader(new InputStreamReader(System.in));
do{
final String userInput = inReader.readLine();
if(userInput==null || "q".equals(userInput))
break; connection.write(userInput);
} while(true);
} finally{
// 关闭client连接
if(connection!=null)
connection.close(); // 停止传输
transport.shutdownNow();
}
}
}

执行Echoclient:

java -classpath grizzly-framework.jar EchoClient

已经通过測试,程序执行的非常完美。

Grizzly开发Echoserver实战的更多相关文章

  1. 前端开发css实战:使用css制作网页中的多级菜单

    前端开发css实战:使用css制作网页中的多级菜单 在日常工作中,大家都会遇到一些显示隐藏类菜单,比如页头导航.二维码显示隐藏.文本提示等等......而这些效果都是可以使用纯css实现的(而且非常简 ...

  2. 用grunt搭建自动化的web前端开发环境实战教程(详细步骤)

    用grunt搭建自动化的web前端开发环境实战教程(详细步骤) jQuery在使用grunt,bootstrap在使用grunt,百度UEditor在使用grunt,你没有理由不学.不用!前端自动化, ...

  3. NET 分布式架构开发项目实战

    .NET 分布式架构开发项目实战 从头到尾,一步一步讲述一个真实的项目实战,关注点主要是架构的思考和实现,以及如何解决平时项目遇到的一些问题. 同时也司公布源代码. 如何构建高性能,稳定SOA应用之- ...

  4. 《PyQt5快速开发与实战了》正式发售 !!!

    <PyQt5快速开发与实战>正式出售了,该书是国内第一本介绍PyQt5的书籍.是两位一线工程师耗费一年的心血.本书github网址:https://github.com/cxinping/ ...

  5. C# Xamarin移动开发项目实战篇

    一.课程介绍 在前面阿笨的<C# Xamarin移动开发基础进修篇>课程中,大家已经熟悉和了解了Xamarin移动App开发的基础知识和原理.本次分享课<C# Xamarin移动开发 ...

  6. 【Android开发VR实战】二.播放360&#176;全景视频

    转载请注明出处:http://blog.csdn.net/linglongxin24/article/details/53924006 本文出自[DylanAndroid的博客] [Android开发 ...

  7. Hbuilder开发app实战-识岁03-文件上传

    前言 做app不得不谈的问题就是文件上传.用hbuilder开发app让上传变的非常easy. Uploader Uploader模块管理网络上传任务,用于从本地上传各种文件到server,并支持跨域 ...

  8. jQuery:[2]百度地图开发平台实战

    jQuery:[2]百度地图开发平台实战 原文链接:   http://blog.csdn.net/moniteryao/article/details/51078779 快速开始 开发平台地址 ht ...

  9. 【Android开发VR实战】三.开发一个寻宝类VR游戏TreasureHunt

    转载请注明出处:http://blog.csdn.net/linglongxin24/article/details/53939303 本文出自[DylanAndroid的博客] [Android开发 ...

随机推荐

  1. IOS开发之微博的设计与实现

    // // main.m // Microblog // #import <Foundation/Foundation.h> #import "Person.h" #i ...

  2. 金典 SQL笔记(4)

    由于在本地笔记上写的.CSDN markdown编辑器仅仅支持.md格式导入, 图片没办法直接导进去.写的多了懒的一张一张图片切图上传; 直接整个文章切成图片上传上去了. watermark/2/te ...

  3. VS2010中属性页中,C/C++ -->预处理器定义

    如上图中,在这里,WIN32._DEBUGE._UNICODE等其实是一些宏定义,在这里写上这些,相当于在本工程所有的文件中都写上了: #define WIN32 #define _DEBUG#def ...

  4. Windows server 2008系统下FTP服务器的安装

    一.在 Windows 服务器上安装 FTP 服务 1. 在"开始"菜单上,单击"管理工具",然后单击"服务器管理器". 2. 在" ...

  5. 理解JavaScript中作用域链的关系

    javascript里的关系又多又乱.作用域链是一种单向的链式关系,还算简单清晰:this机制的调用关系,稍微有些复杂:而关于原型,则是prototype.proto和constructor的三角关系 ...

  6. (转).NET平台开源JSON库LitJSON的使用方法

    一个简单示例: String str = "{’name’:’cyf’,’id’:10,’items’:[{’itemid’:1001,’itemname’:’hello’},{’itemi ...

  7. C#获取当前路径的几种方法

    C#获取当前路径的方法如下: 1. System.Diagnostics.Process.GetCurrentProcess().MainModule.FileName -获取模块的完整路径. 2. ...

  8. Activity一共有以下四种launchMode

    1. standard: 无论什么情况都会生成一个新的Activity实例,并且放于栈顶. 2. singleTop:如果Activity纯在但是不位于栈顶,就重新生成一个Activity实例. 3. ...

  9. 详解CSS设置默认字体样式

    浏览器默认的样式往往在不同的浏览器.不同的语言版本甚至不同的系统版本都有不同的设置,这就导致如 果直接利用默认样式的页面在各个浏览器下显示非常不一致,于是就有了类似YUI的reset之类用来尽量重写浏 ...

  10. HTML&CSS基础学习笔记1.15-合并单元格

    合并单元格 之前的文章中,我们已经能够创建一个简单地表格了,如果我们需要把横向的某两个相邻单元格<td>或者纵向的某两个相邻单元格<td>合并,我们该怎么做呢?我们要知道的知识 ...