三分鐘構建高性能WebSocket服務 | 超優雅的Springboot整合Netty方案

前言

每當使用SpringBoot進行Weboscket開發時,最容易想到的就是spring-boot-starter-websocket(或spring-websocket)。它可讓咱們使用註解,很簡單的進行Websocket開發,讓咱們更多的關注業務邏輯。它底層使用的是Tomcat,且不說把整個Tomcat放進一個WebSocket服務中是否會過重,但在大數據量高併發的場景下,它的表現並非很是理想。html

Netty一款高性能的NIO網絡編程框架,在推送量激增時,表現依然出色。(關於性能與表現的討論,網上不少,這裏不過多說明。)不少流行開源項目都在使用Netty,如:Dubbo、Storm、Spark、Elasticsearch、Apache Cassandra等,這得益於Netty的併發高、傳輸快、封裝好等特色。java

可是,要在SpringBoot項目中整合Netty來開發WebSocket不是一件舒服的事,這會讓你過多的關注非業務邏輯的實現。那麼,是否有一款框架,能使得在SpringBoot項目中使用Netty開發WebSocket變得簡單,甚至優雅,而且能夠從使用spring-boot-starter-websocket開發的項目無縫的遷移過來呢?git

netty-websocket-spring-boot-starter

這是個開源的框架。經過它,咱們能夠像spring-boot-starter-websocket同樣使用註解進行開發,只需關注須要的事件(如OnMessage)。而且底層是使用Netty,當須要調參的時候只須要修改配置參數便可,無需過多的關心handler的設置。github

快速入門

  • 建立SpringBoot項目(v2.0.0以上)並添加依賴:
<dependency>
		<groupId>org.yeauty</groupId>
		<artifactId>netty-websocket-spring-boot-starter</artifactId>
		<version>0.6.3</version>
	</dependency>
  • new一個ServerEndpointExporter對象,交給Spring容器,表示要開啓WebSocket功能:
@Configuration
public class WebSocketConfig {
    @Bean
    public ServerEndpointExporter serverEndpointExporter() {
        return new ServerEndpointExporter();
    }
}
  • 在端點類上加上@ServerEndpoint@Component註解,並在相應的方法上加上@OnOpen@OnClose@OnError@OnMessage註解(不想關注某個事件可不添加對應的註解):
@ServerEndpoint
@Component
public class MyWebSocket {

    @OnOpen
    public void onOpen(Session session, HttpHeaders headers) throws IOException {
        System.out.println("new connection");
    }

    @OnClose
    public void onClose(Session session) throws IOException {
       System.out.println("one connection closed"); 
    }

    @OnError
    public void onError(Session session, Throwable throwable) {
        throwable.printStackTrace();
    }

    @OnMessage
    public void OnMessage(Session session, String message) {
        System.out.println(message);
        session.sendText("Hello Netty!");
    }
}
  • 一個高性能的WebSocket服務端就完成了,直接run起來就能夠了。

測試

  • 服務端是寫完了,接下來須要測試一下,看看效果
  • 首先,新建一個html文件,把頁面擼出來
<!DOCTYPE html>
<html lang="en">
<body>
<div id="msg"></div>
<input type="text" id="text">
<input type="submit" value="send" onclick="send()">
</body>
<script>
    var msg = document.getElementById("msg");
    var wsServer = 'ws://127.0.0.1:80';
    var websocket = new WebSocket(wsServer);
    //監聽鏈接打開
    websocket.onopen = function (evt) {
        msg.innerHTML = "The connection is open";
    };

    //監聽服務器數據推送
    websocket.onmessage = function (evt) {
        msg.innerHTML += "<br>" + evt.data;
    };

    //監聽鏈接關閉
    websocket.onclose = function (evt) {
        alert("鏈接關閉");
    };

    function send() {
        var text = document.getElementById("text").value
        websocket.send(text);
    }
</script>
</html>
  • 頁面擼完,直接用Chrome打開上面html文件便可連上你的WebSocket服務。

總結

這個框架是基於Netty的,因此直接使用Netty優化時的理念便可。如:堆外內存的0拷貝、接收及發送緩衝區的調整、高低寫水位的調整等。web

生產環境的項目在充分調優後,Netty甚至能比Tomcat高效20倍。(固然,這是特定的場景下)spring

框架詳細文檔:https://github.com/YeautyYE/netty-websocket-spring-boot-starter編程

相關文章
相關標籤/搜索