摘要:是一个持久化的协议,相对于这种非持久的协议来说。最大的特点就是实现全双工通信客户端能够实时推送消息给服务端,服务端也能够实时推送消息给客户端。参考链接知乎问题原理原理知乎问题编码什么用如果文章有错的地方欢迎指正,大家互相交流。
前言
今天在慕课网上看到了Java的新教程(Netty入门之WebSocket初体验):https://www.imooc.com/learn/941
WebSocket我是听得很多,没有真正使用过的技术。我之前也去了解过了WebSocket究竟是什么东西,不过一直没有去实践过。
我在写监听器博文的时候,在线人数功能用监听器的是来做,在评论有说使用WebSocket的方式会更加好。
那么,我们就来探究一下WebSocket究竟是什么东西,顺便了解一下Netty!
WebSocket介绍 什么是WebSocketWebSocket是一个协议,归属于IETF。
HTTP是运行在TCP协议传输层上的应用协议,而WebSocket是通过HTTP协议协商如何连接,然后独立运行在TCP协议传输层上的应用协议。
Websocket是一个持久化的协议,相对于HTTP这种非持久的协议来说。
websocket约定了一个通信的规范,通过一个握手的机制,客户端和服务器之间能建立一个类似tcp的连接,从而方便它们之间的通信
为什么需要WebSocket添加WebSocket特性,是为了更好、更灵活,轻量的与服务器通讯。因为WebSocket提供了简单的消息规范,可以更快的适应长连接的环境,其实现在HTTP协议自身就可以做,但是不太轻便。
WebSocket最大的特点就是实现全双工通信:客户端能够实时推送消息给服务端,服务端也能够实时推送消息给客户端。
WebSocket可以做聊天室,股票实时价格显示等应用
纠正WebSocket误区WebSocket是一种应用协议,而我们常常看到了HTML5 WebSocket是API,不要将其进行混淆。
广义上的 HTML5 里面包含的是 WebSocket API,并不是 WebSocket。简单的说,可以把 WebSocket 当成 HTTP,WebSocket API 当成 Ajax。
Netty介绍 什么是Netty知乎的@郭无心总结得很好,我下面就摘抄一下了(链接在下方):
Netty是什么?
1)本质:JBoss做的一个Jar包
2)目的:快速开发高性能、高可靠性的网络服务器和客户端程序
3)优点:提供异步的、事件驱动的网络应用程序框架和工具
通俗的说:一个好使的处理Socket的东东
如果没有Netty?
远古:java.net + java.io
近代:java.nio其他:Mina,Grizzly
简单来说:
你想写个tomcat一样的Server,可以用netty。
你想写一个即时通讯的应用,可以用netty。
你想实现一个高性能Rpc框架,可以用netty。
Netty优势Netty优势:API简单,性能高,入门门槛低,成熟稳健,修复了很多原生NIO的bug
回到课程中来课程是以Netty实现WebSocket来进行讲解的,也就上边所说的:用Netty来实现即时通信的应用
源码下载地址:https://img.mukewang.com/down/5a6e804c0001970d00000000.zip
首先创建了一个全局配置类,WebSocket是全双工通信的,它是通过通道来进行通信,因此配置了系统通道组,管理所有的通道
/** * 存储整个工程的全局配置 * @author liuyazhuang * */ public class NettyConfig { /** * 存储每一个客户端接入进来时的channel对象 */ public static ChannelGroup group = new DefaultChannelGroup(GlobalEventExecutor.INSTANCE); }
配置一些通道的信息(可以理解成Servlet时配置request对象的charset、response对象的缓存)
/** * 初始化连接时候的各个组件 * @author liuyazhuang * */ public class MyWebSocketChannelHandler extends ChannelInitializer{ //配置通道的一些编码格式、数据大小、处理器(交由谁处理) @Override protected void initChannel(SocketChannel e) throws Exception { e.pipeline().addLast("http-codec", new HttpServerCodec()); e.pipeline().addLast("aggregator", new HttpObjectAggregator(65536)); e.pipeline().addLast("http-chunked", new ChunkedWriteHandler()); e.pipeline().addLast("handler", new MyWebSocketHandler()); } }
Netty接收请求,分别处理HTTP请求和WebSocket请求,此部分在视频中单单只是代码编写,并没有做过多的介绍。下面我就整理一下:
该类是用于处理请求的核心业务类
最重要的方法是:messageReceived()方法,主要判断是HTTP请求还是WebSocket请求
是HTTP请求时,就handHttpRequest()来进行处理,该方法判断是否有握手的倾向,
如果不是WebSocket握手请求消息,那么直接返回HTTP 400 BAD REQUEST 响应给客户端,应答消息,并关闭链接。
如果是握手请求,那么就进行握手,将WebSocket相关的编码和解码类动态添加到ChannelPipeline中
是websocket则群发,服务端向每个连接上来的客户端群发消息
package com.imooc.netty; import io.netty.buffer.ByteBuf; import io.netty.buffer.Unpooled; import io.netty.channel.ChannelFuture; import io.netty.channel.ChannelFutureListener; import io.netty.channel.ChannelHandlerContext; import io.netty.channel.SimpleChannelInboundHandler; import io.netty.handler.codec.http.DefaultFullHttpResponse; import io.netty.handler.codec.http.FullHttpRequest; import io.netty.handler.codec.http.HttpResponseStatus; import io.netty.handler.codec.http.HttpVersion; import io.netty.handler.codec.http.websocketx.*; import io.netty.util.CharsetUtil; import java.util.Date; /** * 接收/处理/响应客户端websocket请求的核心业务处理类 * * @author liuyazhuang */ public class MyWebSocketHandler extends SimpleChannelInboundHandler
最后,编写入口程序:启动WebSocket服务
package com.imooc.netty; import io.netty.bootstrap.ServerBootstrap; import io.netty.channel.Channel; import io.netty.channel.EventLoopGroup; import io.netty.channel.nio.NioEventLoopGroup; import io.netty.channel.socket.nio.NioServerSocketChannel; /** * 程序的入口,负责启动应用 * @author liuyazhuang * */ public class Main { public static void main(String[] args) { EventLoopGroup bossGroup = new NioEventLoopGroup(); EventLoopGroup workGroup = new NioEventLoopGroup(); try { ServerBootstrap b = new ServerBootstrap(); b.group(bossGroup, workGroup); b.channel(NioServerSocketChannel.class); b.childHandler(new MyWebSocketChannelHandler()); System.out.println("服务端开启等待客户端连接...."); Channel ch = b.bind(8888).sync().channel(); ch.closeFuture().sync(); } catch (Exception e) { e.printStackTrace(); }finally{ //优雅的退出程序 bossGroup.shutdownGracefully(); workGroup.shutdownGracefully(); } } }
客户端代码:
实际使用WebSocketWebSocket客户端
上面的例子讲解了Netty实现WebSocket,一般我们使用WebSocket不会自己来实现,都是用现成的工具包来进行实现。
我查到的常用的方式有两种:
Tomcat实现WebSocket
整合Spring实现WebSocket
这一部分我就不再赘述了,等我用到的时候再补教程吧,先mark下相关的博客:
Tomcat实现
Spring整合和Tomcat实现
基于Java实现
总结WebSocket最大的特点就是长连接,能够实时推送数据。
参考链接:
WebSocket知乎问题
WebSocket原理
WebSocket原理
Netty知乎问题
chunk编码什么用
如果文章有错的地方欢迎指正,大家互相交流。习惯在微信看技术文章的同学,想要获取更多的Java资源的同学,可以关注微信公众号:Java3y
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/68553.html
摘要:服务端确认协议版本,升级为协议。自己写了一个例子,服务端在开始连接后,利用定时器主动向客户端发送随机数,客户端也可以发给服务器消息,然后服务器返回这条消息给客户端。做的事情就是给页面的元素绑定事件。 写在前面webSocket是一项可以让服务器将数据主动推送给客户端的技术。前几天写了一个日志功能,日志数据需要实时更新。正好项目中有封装好的WebSocket组件,且接口支持webSock...
摘要:什么是是一种在单个连接上进行全双工通信的协议。短轮询配段代码,静态服务中间件用来返回静态文件当前价格是元获取最新价格接口客户端不停的发送请求,去服务端获取最新价格。它通过连接到一个服务器,以格式接收事件不关闭连接。 什么是WebSocket? WebSocket是一种在单个TCP连接上进行全双工通信的协议。这里我们发现了一个有趣的词:全双工,那我们就来简单了解下通信方式有哪些! 单工 ...
摘要:什么是是一种在单个连接上进行全双工通信的协议。短轮询配段代码,静态服务中间件用来返回静态文件当前价格是元获取最新价格接口客户端不停的发送请求,去服务端获取最新价格。它通过连接到一个服务器,以格式接收事件不关闭连接。 什么是WebSocket? WebSocket是一种在单个TCP连接上进行全双工通信的协议。这里我们发现了一个有趣的词:全双工,那我们就来简单了解下通信方式有哪些! 单工 ...
摘要:本系列文章旨在总结前端技术栈中的一些通用技术本文主要总结之前在之前,请求的主要模式是客户端发起请求,服务端负责接收,并返回数据。一种比较常见的方式是服务端携带参数,访问某个指定的前端页面,例如后端主动访问。协议是基于的一种新的网络协议。 本系列文章旨在总结前端技术栈中的一些通用技术本文主要总结websocket websocket WebSocket之前 在WebSocket之前,ht...
阅读 2639·2021-11-23 09:51
阅读 1628·2021-11-22 13:54
阅读 2762·2021-11-18 10:02
阅读 920·2021-08-16 10:57
阅读 3527·2021-08-03 14:03
阅读 1859·2019-08-30 15:54
阅读 3510·2019-08-23 14:39
阅读 571·2019-08-23 14:26