netty 简单应用

java 后端服务

启动类

package netty;

import io.netty.bootstrap.ServerBootstrap;

import io.netty.buffer.ByteBuf;

import io.netty.buffer.Unpooled;

import io.netty.channel.*;

import io.netty.channel.nio.NioEventLoopGroup;

import io.netty.channel.socket.nio.NioServerSocketChannel;

import io.netty.handler.codec.http.*;

import io.netty.handler.codec.http.websocketx.*;

import io.netty.handler.stream.ChunkedWriteHandler;

import io.netty.util.CharsetUtil;

import java.util.Map;

import java.util.concurrent.ConcurrentHashMap;

/**

* Created by yehan on 2018/4/27.

*/

public class WebSocketServer {

private final EventLoopGroupworkerGroup =new NioEventLoopGroup();

    private final EventLoopGroupbossGroup =new NioEventLoopGroup();

    private WebSocketServerHandshakerhandshaker;

    private MapchannelMap =new ConcurrentHashMap();

    public void run() {

ServerBootstrap boot =new ServerBootstrap();

        boot.group(bossGroup, workerGroup)

.channel(NioServerSocketChannel.class)

.childHandler(new ChannelInitializer() {

@Override

                    protected void initChannel(Channel ch)throws Exception {

ChannelPipeline pipeline = ch.pipeline();

                        channelMap.put(ch.id(),ch);

                        pipeline.addLast("http-codec", new HttpServerCodec());

                        pipeline.addLast("aggregator", new HttpObjectAggregator(65536));

                        pipeline.addLast("http-chunked", new ChunkedWriteHandler());

                        pipeline.addLast("handler", new WebSocketServerHandler(WebSocketServer.this));

                    }

});

        try {

Channel ch = boot.bind(2048).sync().channel();

            System.out.println("websocket server start at port:2048");

            ch.closeFuture().sync();

        }catch (InterruptedException e) {

e.printStackTrace();

        }finally {

bossGroup.shutdownGracefully();

            workerGroup.shutdownGracefully();

        }

}

public void handlerWebSocketFrame(ChannelHandlerContext ctx,

                                      WebSocketFrame frame) {

/**

* 判断是否关闭链路的指令

*/

        if (frameinstanceof CloseWebSocketFrame) {

handshaker.close(ctx.channel(),

                    (CloseWebSocketFrame) frame.retain());

return;

        }

/**

* 判断是否ping消息

*/

        if (frameinstanceof PingWebSocketFrame) {

ctx.channel().write(

new PongWebSocketFrame(frame.content().retain()));

return;

        }

/**

* 本例程仅支持文本消息,不支持二进制消息

*/

        if (frameinstanceof BinaryWebSocketFrame) {

throw new UnsupportedOperationException(String.format(

"%s frame types not supported", frame.getClass().getName()));

        }

if(frameinstanceof TextWebSocketFrame){

// 返回应答消息

            String request = ((TextWebSocketFrame) frame).text();

            System.out.println("服务端收到:" + request);

//            ctx.channel().writeAndFlush(new TextWebSocketFrame("服务器收到并返回:"+request));

            for (Channel ch :channelMap.values()) {

if (ctx.channel().equals(ch)) {

ch.writeAndFlush(new TextWebSocketFrame("我发送的:"+request));

continue;

                }

ch.writeAndFlush(new TextWebSocketFrame("收到:"+request));

            }

}

}

public void handleHttpRequest(ChannelHandlerContext ctx,FullHttpRequest req){

if (!req.getDecoderResult().isSuccess()

|| (!"websocket".equals(req.headers().get("Upgrade")))) {

sendHttpResponse(ctx, req, new DefaultFullHttpResponse(

HttpVersion.HTTP_1_1, HttpResponseStatus.BAD_REQUEST));

return;

        }

WebSocketServerHandshakerFactory wsFactory =new WebSocketServerHandshakerFactory(

"ws://localhost:2048/ws", null, false);

        handshaker = wsFactory.newHandshaker(req);

        if (handshaker ==null) {

WebSocketServerHandshakerFactory

.sendUnsupportedWebSocketVersionResponse(ctx.channel());

        }else {

handshaker.handshake(ctx.channel(), req);

        }

}

private static void sendHttpResponse(ChannelHandlerContext ctx,

                                        FullHttpRequest req, DefaultFullHttpResponse res) {

// 返回应答给客户端

        if (res.getStatus().code() !=200) {

ByteBuf buf = Unpooled.copiedBuffer(res.getStatus().toString(),

                    CharsetUtil.UTF_8);

            res.content().writeBytes(buf);

            buf.release();

        }

// 如果是非Keep-Alive,关闭连接

        ChannelFuture f = ctx.channel().writeAndFlush(res);

        if (!isKeepAlive(req) || res.getStatus().code() !=200) {

f.addListener(ChannelFutureListener.CLOSE);

        }

}

private static boolean isKeepAlive(FullHttpRequest req) {

return false;

    }

public static void main(String[] args) {

new WebSocketServer().run();

    }

}


处理类

package netty;

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;

/**

* Created by yehan on 2018/4/27.

*/

public class WebSocketServerHandlerextends SimpleChannelInboundHandler {

private  WebSocketServerwebSocketServer;

    public WebSocketServerHandler(WebSocketServer webSocketServer) {

super();

        this.webSocketServer = webSocketServer;

    }

@Override

    protected void channelRead0(ChannelHandlerContext ctx, Object msg)

throws Exception {

/**

* HTTP接入,WebSocket第一次连接使用HTTP连接,用于握手

*/

        if(msginstanceof FullHttpRequest){

webSocketServer.handleHttpRequest(ctx, (FullHttpRequest)msg);

            System.out.println("http请求");

        }

/**

* Websocket 接入

*/

        else if(msginstanceof WebSocketFrame){

webSocketServer.handlerWebSocketFrame(ctx, (WebSocketFrame)msg);

        }

}

@Override

    public void channelReadComplete(ChannelHandlerContext ctx)throws Exception {

ctx.flush();

    }

}

客户端


©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容

  • netty常用API学习 netty简介 Netty是基于Java NIO的网络应用框架. Netty是一个NIO...
    花丶小伟阅读 6,000评论 0 20
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,646评论 18 139
  • 2018年4月1日 周日 天气晴 今天依旧是明媚的一天。儿子为了今天上午能去图书馆,一早起来完成了剩下的一点儿作业...
    阳光明媚_bf41阅读 208评论 0 3
  • 我不要你的祝福 甜言蜜语只会让人中毒 如果你对我真的在乎 就请好好保重自己别让我为你而哭 我不要你的呵护 风雨再大...
    狼神2019阅读 540评论 21 24
  • 曾经在准备一场重要考试的数月中,我体会到了书中或文献中所不曾具体描述过的真理:把压力化为动力。它似乎很抽象,...
    离默青雪阅读 236评论 0 0