如何開發(fā)基于Netty的HTTP/HTTPS應(yīng)用程序
一、通過 SSL/TLS 保護(hù)應(yīng)用程序
SSL 和 TLS 安全協(xié)議層疊在其他協(xié)議之上,用以實(shí)現(xiàn)數(shù)據(jù)安全。為了支持 SSL/TLS,Java 提供了 javax.net.ssl 包,它的 SSLContext 和 SSLEngine 類使得實(shí)現(xiàn)解密和加密變得相當(dāng)簡(jiǎn)單。Netty 通過一個(gè)名為 SsLHandler 的 ChannelHandler 實(shí)現(xiàn)了這個(gè) API,其中 SSLHandler 在內(nèi)部使用 SSLEngine 來完成實(shí)際工作
Netty 還提供了基于 OpenSSL 工具包的 SSLEngine 實(shí)現(xiàn),比 JDK 提供的 SSLEngine 具有更好的性能。如果 OpenSSL 可用,可以將 Netty 應(yīng)用程序配置為默認(rèn)使用 OpenSSLEngine。如果不可用,Netty 將會(huì)退回到 JDK 實(shí)現(xiàn)
下述代碼展示了如何使用 ChannelInitializer 來將 SslHandler 添加到 ChannelPipeline 中
public class SslChannelInitializer extends ChannelInitializer<Channel> { private final SslContext context; private final boolean startTls; public SslChannelInitializer(SslContext context, boolean startTls) { this.context = context; this.startTls = startTls; } @Override protected void initChannel(Channel ch) throws Exception { SSLEngine engine = context.newEngine(ch.alloc()); ch.pipeline().addFirst("ssl", new SslHandler(engine, startTls)); } }
大多數(shù)情況下,Sslhandler 將是 ChannelPipeline 中的第一個(gè) ChannelHandler,這確保了只有在所有其他的 ChannelHandler 將它們的邏輯應(yīng)用到數(shù)據(jù)之后,才會(huì)進(jìn)行加密
SSLHandler 具有一些有用的方法,如表所示,例如,在握手階段,兩個(gè)節(jié)點(diǎn)將相互驗(yàn)證并且商定一種加密方式,你可以通過配置 SslHandler 來修改它的行為,或者在 SSL/TLS 握手一旦完成之后提供通知,握手階段之后,所有的數(shù)據(jù)都將會(huì)被加密
方法名稱 | 描述 |
---|---|
setHandshakeTimeout(long, TimeUnit) setHandshakeTimeoutMillis(long) getHandshakeTimeoutMillis() |
設(shè)置和獲取超時(shí)時(shí)間,超時(shí)之后,握手 ChannelFuture 將會(huì)被通知失敗 |
setCloseNotifyTimeout(long, TimeUnit) setCloseNotifyTimeoutMillis(long) getCloseNotifyTimeoutMillis() |
設(shè)置和獲取超時(shí)時(shí)間,超時(shí)之后,將會(huì)觸發(fā)一個(gè)關(guān)閉通知并關(guān)閉連接,這也會(huì)導(dǎo)致通知該 ChannelFuture 失敗 |
handshakeFuture() | 返回一個(gè)在握手完成后將會(huì)得到通知的 ChannelFuture,如果握手先前已經(jīng)執(zhí)行過,則返回一個(gè)包含了先前握手結(jié)果的 ChannelFuture |
close() close(ChannelPipeline) close(ChannelHandlerContext, ChannelPromise) |
發(fā)送 close_notify 以請(qǐng)求關(guān)閉并銷毀底層的 SslEngine |
二、HTTP 編解碼器
HTTP 是基于請(qǐng)求/響應(yīng)模式的,客戶端向服務(wù)器發(fā)送一個(gè) HTTP 請(qǐng)求,然后服務(wù)器將會(huì)返回一個(gè) HTTP 響應(yīng),Netty 提供了多種多種編碼器和解碼器以簡(jiǎn)化對(duì)這個(gè)協(xié)議的使用
下圖分別展示了生產(chǎn)和消費(fèi) HTTP 請(qǐng)求和 HTTP 響應(yīng)的方法
如圖所示,一個(gè) HTTP 請(qǐng)求/響應(yīng)可能由多個(gè)數(shù)據(jù)部分組成,并且總以一個(gè) LastHttpContent 部分作為結(jié)束
下表概要地介紹了處理和生成這些消息的 HTTP 解碼器和編碼器
名稱 | 描述 |
---|---|
HttpRequestEncoder | 將 HTTPRequest、HttpContent 和 LastHttpContent 消息編碼為字節(jié) |
HttpResponseEncoder | 將 HTTPResponse、HttpContent 和 LastHttpContent 消息編碼為字節(jié) |
HttpRequestDecoder | 將字節(jié)編碼為 HTTPRequest、HttpContent 和 LastHttpContent 消息 |
HttpResponseDecoder | 將字節(jié)編碼為 HTTPResponse、HttpContent 和 LastHttpContent 消息 |
下述代碼中的 HttpPipelineInitializer 類展示了將 HTTP 支持添加到你的應(yīng)用程序是多么簡(jiǎn)單 —— 只需要將正確的 ChannelHandler 添加到 ChannelPipeline 中
public class HttpPipelineInitializer extends ChannelInitializer<Channel> { private final boolean client; public HttpPipelineInitializer(boolean client) { this.client = client; } @Override protected void initChannel(Channel ch) throws Exception { ChannelPipeline pipeline = ch.pipeline(); if (client) { // 如果是客戶端,則添加 HttpResponseDecoder 處理來自服務(wù)器的響應(yīng) pipeline.addLast("decoder", new HttpResponseDecoder()); // 如果是客戶端,則添加 HttpRequestEncoder 向服務(wù)器發(fā)送請(qǐng)求 pipeline.addLast("encoder", new HttpRequestEncoder()); } else { // 如果是服務(wù)端,則添加 HttpRequestDecoder 處理來自客戶端的請(qǐng)求 pipeline.addLast("decoder", new HttpRequestDecoder()); // 如果是客戶端,則添加 HttpResponseEncoder 向客戶端發(fā)送響應(yīng) pipeline.addLast("encoder", new HttpResponseEncoder()); } } }
三、聚合 HTTP 消息
在 ChannelInitializer 將 ChannelHandler 安裝到 ChannelPipeline 中之后,你就可以處理不同類型的 HTTPObject 消息了。但由于 HTTP 請(qǐng)求和響應(yīng)可能由許多部分組成,因此你需要聚合它們以形成完整的消息。Netty 提供了一個(gè)聚合器,它可以將多個(gè)消息部分合并為 FullHttpRequest 或者 FullHttpResponse 消息
由于消息分段需要被緩沖,直到可以轉(zhuǎn)發(fā)下一個(gè)完整的消息給下一個(gè) ChannelInboundHandler,所以這個(gè)操作有輕微的開銷,其所帶來的好處就是你可以不必關(guān)心消息碎片了
引入這種自動(dòng)聚合機(jī)制只不過是向 ChannelPipeline 中添加另外一個(gè) ChannelHandler 罷了,下述代碼展示了如何做到這一點(diǎn):
public class HttpAggregatorInitializer extends ChannelInitializer<Channel> { private final boolean isClient; public HttpAggregatorInitializer(boolean isClient) { this.isClient = isClient; } @Override protected void initChannel(Channel ch) throws Exception { ChannelPipeline pipeline = ch.pipeline(); if (isClient) { // 如果是客戶端,則添加 HttpClientCodec pipeline.addLast("codec", new HttpClientCodec()); } else { // 如果是服務(wù)器,則添加 HttpServerCodec pipeline.addLast("codec", new HttpServerCodec()); } // 將最大的消息大小為 512KB 的 HTTPObjectAggregator 添加到 ChannelPipeline pipeline.addLast("aggregator", new HttpObjectAggregator(512 * 1024)); } }
四、HTTP 壓縮
當(dāng)使用 HTTP 時(shí),建議開啟壓縮功能以盡可能多地減小傳輸數(shù)據(jù)的大小。雖然壓縮會(huì)帶來一些消耗,但通常來說它都是一個(gè)好主意,尤其是對(duì)于文本數(shù)據(jù)而言
Netty 為壓縮和解壓都提供了 ChannelHandler 實(shí)現(xiàn),它們同時(shí)支持 gzip 和 deflate 編碼
客戶端可以通過提供以下頭部信息來指示服務(wù)器它所支持的壓縮格式
GET /encrypted-area HTTP/1.1
Host: www.example.com
Accept-Encoding: gzip, deflate
然而,需要注意的是,服務(wù)器沒有義務(wù)壓縮它所發(fā)送的數(shù)據(jù)
public class HttpCompressionInitializer extends ChannelInitializer<Channel> { private final boolean isClient; public HttpCompressionInitializer(boolean isClient) { this.isClient = isClient; } @Override protected void initChannel(Channel ch) throws Exception { ChannelPipeline pipeline = ch.pipeline(); if (isClient) { // 如果是客戶端,則添加 HTTPClientCodec pipeline.addLast("codec", new HttpClientCodec()); // 如果是客戶端,則添加 HttpContentDecompressor 以處理來自服務(wù)器的壓縮內(nèi)容 pipeline.addLast("decompressor", new HttpContentDecompressor()); } else { // 如果是服務(wù)端,則添加 HttpServerCodec pipeline.addLast("codec", new HttpServerCodec()); // 如果是服務(wù)器,則添加 HttpContentDecompressor 來壓縮數(shù)據(jù) pipeline.addLast("decompressor", new HttpContentDecompressor()); } } }
五、HTTPS
啟用 HTTPS 只需要將 SslHandler 添加到 ChannelPipeline 的 ChannelHandler 組合中
public class HttpsCodecInitializer extends ChannelInitializer<Channel> { private final SslContext context; private final boolean isClient; public HttpsCodecInitializer(SslContext context, boolean isClient) { this.context = context; this.isClient = isClient; } @Override protected void initChannel(Channel ch) throws Exception { ChannelPipeline pipeline = ch.pipeline(); SSLEngine engine = context.newEngine(ch.alloc()); pipeline.addLast("ssl", new SslHandler(engine)); if (isClient) { pipeline.addLast("codec", new HttpClientCodec()); } else { pipeline.addLast("codec", new HttpServerCodec()); } } }
六、WebSocket
WebSocket 解決了一個(gè)長(zhǎng)期存在的問題:既然底層協(xié)議(HTTP)是一個(gè)請(qǐng)求/響應(yīng)模式的交互序列,那么如何實(shí)時(shí)地發(fā)布信息呢?AJAX一定程度上解決了這個(gè)問題,但數(shù)據(jù)流仍然是由客戶端所發(fā)送的請(qǐng)求驅(qū)動(dòng)的
WebSocket 提供了在單個(gè) TCP 連接上提供雙向的通信,它為網(wǎng)頁和遠(yuǎn)程服務(wù)器之間的雙向通信提供了一種替代 HTTP 輪詢的方案
要想向你的應(yīng)用程序添加對(duì)于 WebSocket 的支持,你需要將適當(dāng)?shù)目蛻舳嘶蛘叻?wù)器 WebSocketChannelHandler 添加到 ChannelPipeline 中。這個(gè)類將處理由 WebSocket 定義的稱為幀的特殊消息類型,如表所示,WebSocketFrame 可以被歸類為數(shù)據(jù)幀或者控制幀
名稱 | 描述 |
---|---|
BinaryWebSocketFrame | 數(shù)據(jù)幀:二進(jìn)制數(shù)據(jù) |
TextWebSocketFrame | 數(shù)據(jù)幀:文本數(shù)據(jù) |
ContinuationWebSocketFrame | 數(shù)據(jù)幀:屬于上一個(gè) BinaryWebSocketFrame 或者 TextWebSocketFrame 的文本或者二進(jìn)制的數(shù)據(jù) |
CloseWebSocketFrame | 控制幀:一個(gè) CLOSE 請(qǐng)求,關(guān)閉的狀態(tài)碼以及關(guān)閉的原因 |
PingWebSocketFrame | 控制幀:請(qǐng)求一個(gè) PongWebSocketFrame |
PongWebSocketFrame | 控制幀:對(duì) PingWebSocketFrame 請(qǐng)求的響應(yīng) |
因?yàn)?Netty 主要是一種服務(wù)器端技術(shù),所以我們重點(diǎn)創(chuàng)建 WebSocket 服務(wù)器。下述代碼展示了使用 WebSocketChannelHandler 的簡(jiǎn)單示例,這個(gè)類會(huì)處理協(xié)議升級(jí)握手,以及三種控制幀 —— Close、Ping 和 Pong,Text 和 Binary 數(shù)據(jù)幀將會(huì)被傳遞給下一個(gè) ChannelHandler 進(jìn)行處理
public class WebSocketServerInitializer extends ChannelInitializer<Channel> { @Override protected void initChannel(Channel ch) throws Exception { ch.pipeline().addLast( new HttpServerCodec(), new HttpObjectAggregator(65536), // 如果被請(qǐng)求的端點(diǎn)是 /websocket,則處理該升級(jí)握手 new WebSocketServerProtocolHandler("/websocket"), // TextFrameHandler 處理 TextWebSocketFrame new TextFrameHandler(), // BinaryFrameHandler 處理 BinaryWebSocketFrame new BinaryFrameHandler(), // ContinuationFrameHandler 處理 Continuation WebSocketFrame new ContinuationFrameHandler()); } public static final class TextFrameHandler extends SimpleChannelInboundHandler<TextWebSocketFrame> { @Override protected void messageReceived(ChannelHandlerContext ctx, TextWebSocketFrame msg) throws Exception { // do something } } public static final class BinaryFrameHandler extends SimpleChannelInboundHandler<BinaryWebSocketFrame> { @Override protected void messageReceived(ChannelHandlerContext ctx, BinaryWebSocketFrame msg) throws Exception { // do something } } public static final class ContinuationFrameHandler extends SimpleChannelInboundHandler<ContinuationWebSocketFrame> { @Override protected void messageReceived(ChannelHandlerContext ctx, ContinuationWebSocketFrame msg) throws Exception { // do something } } }
以上就是如何開發(fā)基于Netty的HTTP/HTTPS應(yīng)用程序的詳細(xì)內(nèi)容,更多關(guān)于Netty HTTP/HTTPS的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
SpringBoot使用PageHelper插件實(shí)現(xiàn)Mybatis分頁效果
這篇文章主要介紹了SpringBoot使用PageHelper插件實(shí)現(xiàn)Mybatis分頁效果,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的參考借鑒價(jià)值,需要的朋友可以參考下2024-02-02Python連接Java Socket服務(wù)端的實(shí)現(xiàn)方法
這篇文章主要介紹了Python連接Java Socket服務(wù)端的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01Maven項(xiàng)目分析剔除無用jar引用的方法步驟
這篇文章主要介紹了Maven項(xiàng)目分析剔除無用jar引用的方法步驟,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10Java劍指offer之刪除鏈表的節(jié)點(diǎn)
這篇文章主要介紹了Java劍指offer之刪除鏈表的節(jié)點(diǎn),給定單向鏈表的頭指針和一個(gè)要?jiǎng)h除的節(jié)點(diǎn)的值,定義一個(gè)函數(shù)刪除該節(jié)點(diǎn)。返回刪除后的鏈表的頭節(jié)點(diǎn),下文更多相關(guān)內(nèi)容介紹,需要的小伙伴可以參考一下2022-04-04java實(shí)現(xiàn)簡(jiǎn)單的推箱子小游戲
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)簡(jiǎn)單的推箱子小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-05-05劍指Offer之Java算法習(xí)題精講鏈表專項(xiàng)訓(xùn)練
跟著思路走,之后從簡(jiǎn)單題入手,反復(fù)去看,做過之后可能會(huì)忘記,之后再做一次,記不住就反復(fù)做,反復(fù)尋求思路和規(guī)律,慢慢積累就會(huì)發(fā)現(xiàn)質(zhì)的變化2022-03-03springboot+swagger2.10.5+mybatis-plus 入門詳解
這篇文章主要介紹了springboot+swagger2.10.5+mybatis-plus 入門,本文通過實(shí)例圖文相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-12-12