以下是netty4.0的代码,如果是5.0的代码请参考 —>
Netty之——基于Netty5.0高级案例NettyWebsocket
http://blog.csdn.net/l1028386804/article/details/55026558
1、Java代码
package com.test.netty4;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.http.FullHttpRequest;
public interface IHttpService {
void handleHttpRequest(ChannelHandlerContext ctx, FullHttpRequest request);
}
package com.test.netty4;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.http.websocketx.WebSocketFrame;
public interface IWebSocketService {
void handleFrame(ChannelHandlerContext ctx, WebSocketFrame frame);
}
package com.test.netty4;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelId;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.group.ChannelGroup;
import io.netty.channel.group.DefaultChannelGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.http.FullHttpRequest;
import io.netty.handler.codec.http.HttpHeaderNames;
import io.netty.handler.codec.http.HttpHeaders;
import io.netty.handler.codec.http.HttpMethod;
import io.netty.handler.codec.http.HttpObjectAggregator;
import io.netty.handler.codec.http.HttpServerCodec;
import io.netty.handler.codec.http.websocketx.CloseWebSocketFrame;
import io.netty.handler.codec.http.websocketx.PingWebSocketFrame;
import io.netty.handler.codec.http.websocketx.PongWebSocketFrame;
import io.netty.handler.codec.http.websocketx.TextWebSocketFrame;
import io.netty.handler.codec.http.websocketx.WebSocketFrame;
import io.netty.handler.codec.http.websocketx.WebSocketServerHandshaker;
import io.netty.handler.codec.http.websocketx.WebSocketServerHandshakerFactory;
import io.netty.handler.stream.ChunkedWriteHandler;
import io.netty.util.AttributeKey;
import io.netty.util.concurrent.GlobalEventExecutor;
public class WebSocketServer implements IWebSocketService, IHttpService {
public static void main(String[] args) {
new WebSocketServer(9999).start();
}
// ----------------------------static fields -----------------------------
private static final String HN_HTTP_CODEC = "HN_HTTP_CODEC";
private static final String HN_HTTP_AGGREGATOR = "HN_HTTP_AGGREGATOR";
private static final String HN_HTTP_CHUNK = "HN_HTTP_CHUNK";
private static final String HN_SERVER = "HN_LOGIC";
// handshaker attachment key
private static final AttributeKey<WebSocketServerHandshaker> ATTR_HANDSHAKER = AttributeKey.newInstance("ATTR_KEY_CHANNELID");
private static final int MAX_CONTENT_LENGTH = 65536;
private static final String WEBSOCKET_UPGRADE = "websocket";
private static final String WEBSOCKET_CONNECTION = "Upgrade";
private static final String WEBSOCKET_URI_ROOT_PATTERN = "ws://%s:%d";
// ------------------------ member fields -----------------------
private String host; // 绑定的地址
private int port; // 绑定的端口
/**
* 保存所有WebSocket连接
*/
private Map<ChannelId, Channel> channelMap = new ConcurrentHashMap<ChannelId, Channel>();
private ChannelGroup channelGroup = new DefaultChannelGroup(GlobalEventExecutor.INSTANCE);
private final String WEBSOCKET_URI_ROOT;
public WebSocketServer(int port) {
this("localhost", port);
}
public WebSocketServer(String host, int port) {
this.host = host;
this.port = port;
WEBSOCKET_URI_ROOT = String.format(WEBSOCKET_URI_ROOT_PATTERN, host, port);
}
public void start() {
EventLoopGroup bossGroup = new NioEventLoopGroup();
EventLoopGroup workerGroup = new NioEventLoopGroup();
ServerBootstrap b = new ServerBootstrap();
b.group(bossGroup, workerGroup);
b.channel(NioServerSocketChannel.class);
b.childHandler(new ChannelInitializer<Channel>() {
@Override
protected void initChannel(Channel ch) throws Exception {
ChannelPipeline pl = ch.pipeline();
// 保存该Channel的引用
channelGroup.add(ch);
channelMap.put(ch.id(), ch);
System.out.println("new channel {}"+ ch);
ch.closeFuture().addListener(new ChannelFutureListener() {
public void operationComplete(ChannelFuture future) throws Exception {
System.out.println("channel close {}"+ future.channel());
// Channel 关闭后不再引用该Channel
channelMap.remove(future.channel().id());
}
});
pl.addLast(HN_HTTP_CODEC, new HttpServerCodec());
pl.addLast(HN_HTTP_AGGREGATOR, new HttpObjectAggregator(MAX_CONTENT_LENGTH));
pl.addLast(HN_HTTP_CHUNK, new ChunkedWriteHandler());
pl.addLast(HN_SERVER, new WebSocketServerHandler(WebSocketServer.this, WebSocketServer.this));
}
});
try {
// 绑定端口
ChannelFuture future = b.bind(host, port).addListener(new ChannelFutureListener() {
public void operationComplete(ChannelFuture future) throws Exception {
if (future.isSuccess()) {
System.out.println("websocket started.");
}
}
}).sync();
future.channel().closeFuture().addListener(new ChannelFutureListener() {
public void operationComplete(ChannelFuture future) throws Exception {
System.out.println("server channel {} closed."+future.channel());
}
}).sync();
} catch (InterruptedException e) {
e.printStackTrace();
// logger.error(e.toString());
} finally {
bossGroup.shutdownGracefully();
workerGroup.shutdownGracefully();
}
System.out.println("websocket server shutdown");
}
/*
* @see cc.lixiaohui.demo.netty4.websocket.IHttpService#handleHttpRequest(io.netty.handler.codec.http.FullHttpRequest)
*/
public void handleHttpRequest(ChannelHandlerContext ctx, FullHttpRequest req) {
if (isWebSocketUpgrade(req)) { // 该请求是不是websocket upgrade请求
System.out.println("upgrade to websocket protocol");
String subProtocols = req.headers().get(HttpHeaderNames.SEC_WEBSOCKET_PROTOCOL);
WebSocketServerHandshakerFactory factory = new WebSocketServerHandshakerFactory(WEBSOCKET_URI_ROOT, subProtocols, false);
WebSocketServerHandshaker handshaker = factory.newHandshaker(req);
if (handshaker == null) {// 请求头不合法, 导致handshaker没创建成功
WebSocketServerHandshakerFactory.sendUnsupportedVersionResponse(ctx.channel());
} else {
// 响应该请求
handshaker.handshake(ctx.channel(), req);
// 把handshaker 绑定给Channel, 以便后面关闭连接用
ctx.channel().attr(ATTR_HANDSHAKER).set(handshaker);// attach handshaker to this channel
}
return;
}
// TODO 忽略普通http请求
System.out.println("ignoring normal http request");
}
/*
* @see
* cc.lixiaohui.demo.netty4.websocket.IWebSocketService#handleFrame(io.netty
* .channel.Channel, io.netty.handler.codec.http.websocketx.WebSocketFrame)
*/
public void handleFrame(ChannelHandlerContext ctx, WebSocketFrame frame) {
// text frame
if (frame instanceof TextWebSocketFrame) {
String text = ((TextWebSocketFrame) frame).text();
// TextWebSocketFrame rspFrame = new TextWebSocketFrame(text);
System.out.println("recieve TextWebSocketFrame from channel {}"+ ctx.channel());
// 发给其他所有channel
// for (Channel ch : channelMap.values()) {
// if (ctx.channel().equals(ch)) {
// continue;
// }
// TextWebSocketFrame rspFrame = new TextWebSocketFrame(text);
// ch.writeAndFlush(rspFrame);
// System.out.println("write text[{}] to channel {}"+ text+ ch);
// }
TextWebSocketFrame rspFrame = new TextWebSocketFrame(text);
channelGroup.writeAndFlush(rspFrame);
return;
}
// ping frame, 回复pong frame即可
if (frame instanceof PingWebSocketFrame) {
System.out.println("recieve PingWebSocketFrame from channel {}"+ ctx.channel());
ctx.channel().writeAndFlush(new PongWebSocketFrame(frame.content().retain()));
return;
}
if (frame instanceof PongWebSocketFrame) {
System.out.println("recieve PongWebSocketFrame from channel {}"+ ctx.channel());
return;
}
// close frame,
if (frame instanceof CloseWebSocketFrame) {
System.out.println("recieve CloseWebSocketFrame from channel {}"+ ctx.channel());
WebSocketServerHandshaker handshaker = ctx.channel().attr(ATTR_HANDSHAKER).get();
if (handshaker == null) {
System.out.println("channel {} have no HandShaker"+ ctx.channel());
return;
}
handshaker.close(ctx.channel(), (CloseWebSocketFrame) frame.retain());
return;
}
// 剩下的是binary frame, 忽略
System.out.println("unhandle binary frame from channel {}"+ctx.channel());
}
//三者与:1.GET? 2.Upgrade头 包含websocket字符串? 3.Connection头 包含 Upgrade字符串?
private boolean isWebSocketUpgrade(FullHttpRequest req) {
HttpHeaders headers = req.headers();
return req.getMethod().equals(HttpMethod.GET)
&& headers.get(HttpHeaderNames.UPGRADE).contains(WEBSOCKET_UPGRADE)
&& headers.get(HttpHeaderNames.CONNECTION).contains(WEBSOCKET_CONNECTION);
}
}
package com.test.netty4;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;
import io.netty.handler.codec.http.FullHttpRequest;
import io.netty.handler.codec.http.websocketx.WebSocketFrame;
public class WebSocketServerHandler extends SimpleChannelInboundHandler<Object> {
@SuppressWarnings("unused")
private IWebSocketService websocketService;
private IHttpService httpService;
public WebSocketServerHandler(IWebSocketService websocketService, IHttpService httpService) {
super();
this.websocketService = websocketService;
this.httpService = httpService;
}
/*
* @see
* io.netty.channel.SimpleChannelInboundHandler#channelRead0(io.netty.channel
* .ChannelHandlerContext, java.lang.Object)
*/
@Override
protected void channelRead0(ChannelHandlerContext ctx, Object msg) throws Exception {
if (msg instanceof FullHttpRequest) {
httpService.handleHttpRequest(ctx, (FullHttpRequest) msg);
} else if (msg instanceof WebSocketFrame) {
websocketService.handleFrame(ctx, (WebSocketFrame) msg);
}
}
/*
* @see io.netty.channel.ChannelInboundHandlerAdapter#channelReadComplete(io.netty.channel.ChannelHandlerContext)
*/
@Override
public void channelReadComplete(ChannelHandlerContext ctx) throws Exception {
ctx.flush();
}
@Override
public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
}
}
2、html页面代码
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<title></title>
</head>
</head>
<script type="text/javascript">
var socket;
if(!window.WebSocket){
window.WebSocket = window.MozWebSocket;
}
if(window.WebSocket){
socket = new WebSocket("ws://localhost:9999");
socket.onmessage = function(event){
appendln("接收:" + event.data);
};
socket.onopen = function(event){
appendln("WebSocket 连接已建立");
};
socket.onclose = function(event){
appendln("WebSocket 连接已关闭");
};
}else{
alert("浏览器不支持WebSocket协议");
}
function send(message){
if(!window.WebSocket){return;}
if(socket.readyState == WebSocket.OPEN){
socket.send(message);
appendln("发送:" + message);
}else{
alert("WebSocket连接建立失败");
}
}
function appendln(text) {
var ta = document.getElementById('responseText');
ta.value += text + "\r\n";
}
function clear() {
var ta = document.getElementById('responseText');
ta.value = "";
}
</script>
<body>
<form onSubmit="return false;">
<input type = "text" name="message" value="你好啊"/>
<br/><br/>
<input type="button" value="发送 WebSocket 请求消息" onClick="send(this.form.message.value)"/>
<hr/>
<h3>服务端返回的应答消息</h3>
<textarea id="responseText" style="width: 800px;height: 300px;"></textarea>
</form>
</body>
</html>
3、启动 WebSocketServer 的main方法,然后打开多个html页面,即可实现聊天互通。更
此处是基于netty-all-4.1.12.Final.jar实现的。更详细的介绍可自行网络搜索。