• <noscript id="e0iig"><kbd id="e0iig"></kbd></noscript>
  • <td id="e0iig"></td>
  • <option id="e0iig"></option>
  • <noscript id="e0iig"><source id="e0iig"></source></noscript>
  • Netty學習(1):IO模型之BIO

    標簽: Netty  netty  java  socket

    概述

    Netty其實就是一個異步的、基于事件驅動的框架,其作用是用來開發高性能、高可靠的IO程序。

    因此下面就讓我們從Java的IO模型來逐步深入學習Netty。

    IO模型

    IO模型簡單來說,就是采用如何的方式來進行數據的接受和發送,因為存在著發送方和接收方兩個角色,因此IO模型一般分為以下3類:BIO(同步阻塞)、NIO(同步非阻塞)、AIO(異步非阻塞)。其3者的區別如下:

    • BIO:是最傳統的Java IO模型,采用的方式為服務器新接受到一個連接,就建立一個線程,但是如果這個連接不做任何事情,該線程也會被創建,閑置著,增加不必要的開銷。
    • NIO:服務器用一個線程處理多個請求,將所有請求注冊到一個線程管理的多路復用器(Selector)上。適用于連接數量多,但連接比較輕量的服務上,如聊天,彈幕等。
    • AIO:其特點是由OS完成后,才通知服務端程序啟動程序來處理,使用場景為相冊服務器等。因不是很清楚具體實現,因此在這不展開了。

    BIO代碼實現

    使用BIO模型來實現一個服務器端和客戶端,并采用線程池的概念,使其可以連接多個客戶端。

    服務端

    首先,讓我們來看建立一個服務端需要哪些步驟:

    // 創建線程池,如果有連接,就創建一個線程  ps:這里手動指定參數創建線程池會更好,但這里因為不是重點,因此就采用默認的線程池來實現。
            ExecutorService executorService = Executors.newCachedThreadPool();
    
            // create ServerSocket
            ServerSocket serverSocket = new ServerSocket(6666);
            System.out.println("服務端建立,監聽!!!");
    
            while (true) {
                System.out.println("thread info0 id : " + Thread.currentThread().getId() + " \tname: " + Thread.currentThread().getName());
    
                // waiting for client
                System.out.println("等待客戶端連接");
                // The method blocks until a connection is made.
                final Socket socket = serverSocket.accept();
                System.out.println("客戶端連接成功");
    
                // create a thread for communicated
                executorService.execute(new Runnable() {
                    @Override
                    public void run() {
                        handler(socket);
                    }
                });
            }
    

    上述代碼就是一個服務器端啟動的代碼實例,那么該如何處理客戶端發送的數據呢。

        public static void handler(Socket socket) {
    
            try (InputStream inputStream = socket.getInputStream();
                 OutputStream outputStream = socket.getOutputStream();
                 BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(inputStream));
                 PrintWriter printWriter = new PrintWriter(socket.getOutputStream(), true)) {
    
                System.out.println("thread info1 id : " + Thread.currentThread().getId() + " \tname: " + Thread.currentThread().getName());
    
                String expression;
                String result;
                while (true) {
                    System.out.println("thread info2 id : " + Thread.currentThread().getId() + " \tname: " + Thread.currentThread().getName());
                    //BufferedReader.readLine()會讀滿緩沖區或者在讀到文件末尾(遇到:"/r"、"/n"、"/r/n")才返回
                    System.out.println("等待讀取數據");
                    if ((expression = bufferedReader.readLine()) == null) {
                        break;
                    }
                    System.out.println("result: " + expression);
                }
            } catch (IOException e) {
                System.out.println(e);
            } finally {
                System.out.println("關閉和client連接!!!");
                try {
                    socket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    

    在這里我們采用循環讀取的形式來處理客戶端發送的數據,并將其進行輸出。

    客戶端

     // 發送
            try (Socket socket = new Socket(InetAddress.getByName("localhost"), 6666);
                 OutputStream outputStream = socket.getOutputStream()) {
                String data = "0A,3,863921030884418,23.178061,0,113.225998,0,2020/01/18 17:28:30,051,4,FF,";
                data = getData(data) + "\n";
                outputStream.write(data.getBytes());
            } catch (Exception e) {
                System.out.println(e);
            }
    

    客戶端的代碼就很簡單了,指定IP和端口,直接連接發送數據即可。

    實現效果

    首先我們啟動服務端,可以看到如下輸出,說明服務端已經被啟動起來了,等待客戶端連接。

    BIO服務端啟動

    然后,我們啟動客戶端1,服務端會啟動一個線程來接受并處理請求。

    BIO請求處理

    BIO實現問題

    上述就是一個BIO程序實現的Demo版本,這也就是BIO的優點所在,其實現簡單,代碼邏輯也十分的簡潔明了。但其有幾個很重要的問題沒有解決。

    1. 對應于每個請求都要創建一個來解決,即使有線程池的存在,也會造成很大的線程資源損耗;
    2. handler() 代碼中,我們可以看到其是在線程里處理的,但如果線程創建完成后,發現沒有數據需要讀寫,那么其則會阻塞在 read 操作中,影響性能。

    正因為BIO有著這許多問題,因此Java官方在1.4后引入了NIO的概念,來完善Java的IO模型,因此,在這里我們只需要了解這種模型即可,知道其會產生什么問題以及產生的原因,接下里,我們把重點放在NIO上,而這也是Netty實現的基礎。

    備注

    本文中代碼已上傳到GitHub上,地址為 https://github.com/wb1069003157/nettyPre-research ,歡迎大家來討論,探討。

    iceWang公眾號

    文章在公眾號「iceWang」第一手更新,有興趣的朋友可以關注公眾號,第一時間看到筆者分享的各項知識點,謝謝!筆芯!

    版權聲明:本文為wb1069003157原創文章,遵循 CC 4.0 BY-SA 版權協議,轉載請附上原文出處鏈接和本聲明。
    本文鏈接:https://blog.csdn.net/wb1069003157/article/details/104290651

    智能推薦

    freemarker + ItextRender 根據模板生成PDF文件

    1. 制作模板 2. 獲取模板,并將所獲取的數據加載生成html文件 2. 生成PDF文件 其中由兩個地方需要注意,都是關于獲取文件路徑的問題,由于項目部署的時候是打包成jar包形式,所以在開發過程中時直接安照傳統的獲取方法沒有一點文件,但是當打包后部署,總是出錯。于是參考網上文章,先將文件讀出來到項目的臨時目錄下,然后再按正常方式加載該臨時文件; 還有一個問題至今沒有解決,就是關于生成PDF文件...

    電腦空間不夠了?教你一個小秒招快速清理 Docker 占用的磁盤空間!

    Docker 很占用空間,每當我們運行容器、拉取鏡像、部署應用、構建自己的鏡像時,我們的磁盤空間會被大量占用。 如果你也被這個問題所困擾,咱們就一起看一下 Docker 是如何使用磁盤空間的,以及如何回收。 docker 占用的空間可以通過下面的命令查看: TYPE 列出了docker 使用磁盤的 4 種類型: Images:所有鏡像占用的空間,包括拉取下來的鏡像,和本地構建的。 Con...

    requests實現全自動PPT模板

    http://www.1ppt.com/moban/ 可以免費的下載PPT模板,當然如果要人工一個個下,還是挺麻煩的,我們可以利用requests輕松下載 訪問這個主頁,我們可以看到下面的樣式 點每一個PPT模板的圖片,我們可以進入到詳細的信息頁面,翻到下面,我們可以看到對應的下載地址 點擊這個下載的按鈕,我們便可以下載對應的PPT壓縮包 那我們就開始做吧 首先,查看網頁的源代碼,我們可以看到每一...

    Linux C系統編程-線程互斥鎖(四)

    互斥鎖 互斥鎖也是屬于線程之間處理同步互斥方式,有上鎖/解鎖兩種狀態。 互斥鎖函數接口 1)初始化互斥鎖 pthread_mutex_init() man 3 pthread_mutex_init (找不到的情況下首先 sudo apt-get install glibc-doc sudo apt-get install manpages-posix-dev) 動態初始化 int pthread_...

    猜你喜歡

    統計學習方法 - 樸素貝葉斯

    引入問題:一機器在良好狀態生產合格產品幾率是 90%,在故障狀態生產合格產品幾率是 30%,機器良好的概率是 75%。若一日第一件產品是合格品,那么此日機器良好的概率是多少。 貝葉斯模型 生成模型與判別模型 判別模型,即要判斷這個東西到底是哪一類,也就是要求y,那就用給定的x去預測。 生成模型,是要生成一個模型,那就是誰根據什么生成了模型,誰就是類別y,根據的內容就是x 以上述例子,判斷一個生產出...

    styled-components —— React 中的 CSS 最佳實踐

    https://zhuanlan.zhihu.com/p/29344146 Styled-components 是目前 React 樣式方案中最受關注的一種,它既具備了 css-in-js 的模塊化與參數化優點,又完全使用CSS的書寫習慣,不會引起額外的學習成本。本文是 styled-components 作者之一 Max Stoiber 所寫,首先總結了前端組件化樣式中的最佳實踐原則,然后在此基...

    基于TCP/IP的網絡聊天室用Java來實現

    基于TCP/IP的網絡聊天室實現 開發工具:eclipse 開發環境:jdk1.8 發送端 接收端 工具類 運行截圖...

    19.vue中封裝echarts組件

    19.vue中封裝echarts組件 1.效果圖 2.echarts組件 3.使用組件 按照組件格式整理好數據格式 傳入組件 home.vue 4.接口返回數據格式...

    劍指Offer39-調整數組順序使奇數位于偶數前面

    一開始想著用冒泡排序的方法來做,但是bug還是很多,后來看了評論區答案,發現直接空間換時間是最簡單的,而且和快排的寫法是類似的。...

    精品国产乱码久久久久久蜜桃不卡