BIO悔捶,NIO,AIO 模型分析

輸入IO與輸出IO原理

內(nèi)核態(tài): CPU可以訪問內(nèi)存所有數(shù)據(jù), 包括外圍設(shè)備, 例如硬盤, 網(wǎng)卡单芜;

用戶態(tài): (獨(dú)立創(chuàng)建應(yīng)用程序) 只能受限的訪問內(nèi)存, 且不允許訪問外圍設(shè)備. 占用CPU的能力被剝奪, CPU資源可以被其他程序獲

圖片來于余老師課堂筆記

1蜕该、BIO(Blocking I O) 同步阻塞模型: 一個(gè)線程對(duì)應(yīng)一個(gè)客戶端連接。

應(yīng)用場(chǎng)景:BIO方式適用于連接數(shù)目比較小且固定的架構(gòu), 這種方式對(duì)服務(wù)器資源要求比較高, 但程序簡(jiǎn)單易理解柜蜈。

2客年、NIO(Non Blocking I O) 同步非阻塞:服務(wù)器實(shí)現(xiàn)模式為一個(gè)線程可以處理多個(gè)請(qǐng)求(連接),客戶端發(fā)送的連接請(qǐng)求都會(huì)注冊(cè)到 多路復(fù)用器selector上,多路復(fù)用器輪詢到連接有IO請(qǐng)求就進(jìn)行處理。

應(yīng)用場(chǎng)景:NIO方式適用于連接數(shù)目多且連接比較短(輕操作) 的架構(gòu), 比如聊天服務(wù)器更啄, 彈幕系統(tǒng), 服務(wù)器間通訊居灯,編程比較復(fù)雜祭务, JDK1.4 開始支持

3、AIO(NIO 2.0) 異步非阻塞:由操作系統(tǒng)完成后回調(diào)通知服務(wù)端程序啟動(dòng)線程去處理怪嫌,一般適用于連接數(shù)較多且連接時(shí)間較長(zhǎng)的應(yīng)用义锥。是在NIO的基礎(chǔ)上進(jìn)一步封裝的。

應(yīng)用場(chǎng)景:AIO方式適用于連接數(shù)目多且連接比較長(zhǎng)(重操作) 的架構(gòu)岩灭,JDK7 開始支持

同步和異步的區(qū)別 同步和異步的區(qū)別

簡(jiǎn)單理解同步與異步:
同步也就是程序從上往下實(shí)現(xiàn)執(zhí)行拌倍,順序執(zhí)行
異步從新開啟一個(gè)新分支,相互不會(huì)影響噪径,并行執(zhí)行柱恤;

站在Http協(xié)議上分析同步與異步區(qū)別:
我們的Http協(xié)議請(qǐng)求默認(rèn)情況下同步形式調(diào)用,如果調(diào)用過程非常耗時(shí)的情況下 客戶端等待時(shí)間就非常長(zhǎng)找爱, 這種形式我們可以理解阻塞式梗顺;
解決辦法:耗時(shí)的代碼我們可以使用多線程或者M(jìn)Q實(shí)現(xiàn)處理,但是不能立馬獲取結(jié)果车摄; 客戶端可以主動(dòng)查詢

圖片來源于余老師課堂筆記

圖片來源于余老師課堂筆記

阻塞與非阻塞的區(qū)別

阻塞: 如果我沒有獲取到結(jié)果的情況下寺谤,當(dāng)前線程從運(yùn)行狀態(tài)切換為阻塞狀態(tài) 內(nèi)核角度分析:用戶空間切換到內(nèi)核空間
非阻塞:如果我沒有獲取到結(jié)果的情況下仑鸥,當(dāng)前的線程不會(huì)阻塞。

阻塞.png

BIO(Blocking IO) 同步阻塞模型

一個(gè)線程處理一個(gè)客戶端請(qǐng)求变屁;
缺點(diǎn):
1眼俊、IO代碼里read操作是阻塞操作,如果獲取不到數(shù)據(jù)的情況下敞贡,則會(huì)阻塞泵琳;
2、如果線程使用過多的情況下誊役,非常消耗服務(wù)器端cpu的資源;
應(yīng)用場(chǎng)景:
BIO 方式適用于連接數(shù)目比較小且固定的架構(gòu)谷市, 這種方式對(duì)服務(wù)器資源要求比較高

同步阻塞模型.png

BIO客戶端代碼:

import java.io.IOException;
import java.net.Socket;

/***
 * SocketBioClient
 */
public class SocketBioClient {

    public static void main(String[] args) throws IOException {
        Socket socket = new Socket("127.0.0.1", 9001);
        //向服務(wù)端發(fā)送數(shù)據(jù)
        socket.getOutputStream().write("來演示下同步阻塞Bio".getBytes());
        socket.getOutputStream().flush();
        System.out.println("向服務(wù)端發(fā)送數(shù)據(jù)結(jié)束");
        byte[] bytes = new byte[1024];
        //接收服務(wù)端回傳的數(shù)據(jù)
        socket.getInputStream().read(bytes);
        System.out.println("接收到服務(wù)端的數(shù)據(jù):" + new String(bytes));
        socket.close();
    }
} 

BIO服務(wù)器端代碼:


import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;

/**
 * @ClassName SocketBioServer
 */
public class SocketBioServer {
    public static void main(String[] args) throws IOException {

        ServerSocket serverSocket = new ServerSocket(9001);
        while (true) {
            System.out.println("服務(wù)器端正在等待連接中...");
            // 阻塞方法 如果沒有客戶端與服務(wù)器端建立連接時(shí)蛔垢,該方法會(huì)阻塞等待
            final Socket socket = serverSocket.accept();
            System.out.println("有客戶端和我連接啦");
            //如果不使用異步線程處理接受io操作的情況下,有可能會(huì)阻塞等待 無法接受新的連接請(qǐng)求迫悠。
            new Thread(() -> {
                try {
                    handler(socket);
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }).start();
           // handler(socket);
        }
    }

    private static void handler(Socket socket) throws IOException {
        System.out.println("線程id= " + Thread.currentThread().getId());
        byte[] bytes = new byte[1024];

        System.out.println("開始read..");
        //接收客戶端的數(shù)據(jù)鹏漆,如果沒有讀取到客戶端數(shù)據(jù)時(shí),該方法也會(huì)阻塞
        int read = socket.getInputStream().read(bytes);
        System.out.println("read結(jié)束");
        if (read != -1) {
            System.out.println("接收到客戶端的數(shù)據(jù):" + new String(bytes, 0, read));
            System.out.println("線程id= = " + Thread.currentThread().getId());

        }
        socket.getOutputStream().write("BIO阻塞效果演示完畢".getBytes());
        socket.getOutputStream().flush();
    }
} 

NIO(Non Blocking IO) 同步非阻塞

NIO同步非阻塞的原理:多個(gè)客戶端發(fā)送連接請(qǐng)求注冊(cè)到(多路復(fù)用器)selector中创泄,多路復(fù)用器使用輪訓(xùn)機(jī)制實(shí)現(xiàn)檢測(cè)每個(gè)io請(qǐng)求有數(shù)據(jù)就進(jìn)行處理艺玲。
底層實(shí)現(xiàn)原理:I/O多路復(fù)用底層一般用的Linux API(select,poll鞠抑,epoll)來實(shí)現(xiàn)

NIO 有三大核心組件: Channel(通道)饭聚, Buffer(緩沖區(qū)),Selector(選擇器)

1.Channel(通道) :稱之為通道搁拙,和IO相連秒梳,通信雙方進(jìn)行數(shù)據(jù)交流的通道,需要和buffer結(jié)合使用箕速。
2.Buffer(緩沖區(qū)) :對(duì)數(shù)據(jù)的讀取/寫入需要使用buffer酪碘,buffer本質(zhì)就是一個(gè)數(shù)組。
3.Selector(選擇器): IO多路復(fù)用盐茎,一個(gè)線程Thread使用選擇器Selector通過輪詢的方式去監(jiān)聽多個(gè)通道Channel上的事件兴垦,從而讓一個(gè)線程可以處理多個(gè)事件。

模擬NIO底層代碼實(shí)現(xiàn)

import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.List;

/**
 * 模擬偽裝nio底層原理
 * SocketNioTcpServer
 */
public class SimulationNioTcpServer {

    /**
     * 保存SocketChannel
     */
    private static List<SocketChannel> listSocketChannel = new ArrayList<>();
    /**
     * 緩沖區(qū)大小
     */
    private static ByteBuffer byteBuffer = ByteBuffer.allocate(512);

    public static void main(String[] args) {
        try {
            // 1.創(chuàng)建一個(gè)ServerSocketChannel
            ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
            // 2. 綁定地址
            ServerSocketChannel bind = serverSocketChannel.bind(new InetSocketAddress(8080));
            // 3.設(shè)置非阻塞模式
            serverSocketChannel.configureBlocking(false);
            while (true) {
                // 4.等待建立連接  如果設(shè)置非阻塞的情況下字柠,如果沒有獲取連接的情況下直接返回null探越,如果建立連接之后返回socketChannel
                // 建立三次握手
                SocketChannel socketChannel = serverSocketChannel.accept();
                // 5. 如果socketChannel 不為空的情況下,則將該連接保存起來募谎。
                if (socketChannel != null) {
                    // 設(shè)置該socketChannel通道為fasle
                    socketChannel.configureBlocking(false);
                    listSocketChannel.add(socketChannel);
                }
                // 循環(huán)SocketChannel,檢查每個(gè)SocketChannel中數(shù)據(jù)有傳輸數(shù)據(jù)
                for (SocketChannel scl : listSocketChannel) {
                    try {
                        // 6.以緩沖區(qū)方式讀取
                        int read = scl.read(byteBuffer);
                        if (read > 0) {
                            byteBuffer.flip();
                            // 轉(zhuǎn)換格式為中文的格式
                            Charset charset = Charset.forName("UTF-8");
                            String receiveText = charset.newDecoder().decode
                                    (byteBuffer.asReadOnlyBuffer()).toString();
                            System.out.println("receiveText:" + receiveText);
                        }
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
} 

選擇器IO多路復(fù)用底層原理
IO多路復(fù)用 一個(gè)線程Thread使用選擇器Selector通過輪詢的方式去監(jiān)聽多個(gè)通道Channel上的事件扶关,從而讓一個(gè)線程可以處理多個(gè)事。I/O多路復(fù)用底層一般用的Linux API(select数冬,poll节槐,epoll)來實(shí)現(xiàn)

選擇器.png

1.select:底層采用該數(shù)組方式存放 每次調(diào)用遍歷的時(shí)間復(fù)雜度就是為O(n),有可能會(huì)產(chǎn)生空輪訓(xùn)搀庶,比如 保存1萬個(gè)連接,最終只有1個(gè)連接有傳輸數(shù)據(jù)铜异。
2.poll:底層采用鏈表結(jié)構(gòu)存放哥倔,每次調(diào)用遍歷的時(shí)間復(fù)雜度就是為O(n),poll與select之間區(qū)別不是很大揍庄;select監(jiān)視器單個(gè)進(jìn)程可監(jiān)視的fd數(shù)量被限制(可以通過cat /proc/sys/fs/file-max咆蒿, poll 是沒有監(jiān)視的fd數(shù)量限制,Linux服務(wù)器中創(chuàng)建Socket服務(wù)器端 單個(gè)select進(jìn)程可監(jiān)事的fd(連接數(shù)據(jù))限制)
3.epoll:采用事件通知回調(diào)方式,避免空輪休時(shí)間復(fù)雜度為o(1);

注意:windows操作系統(tǒng)是沒有epoll蚂子,只有l(wèi)inux系統(tǒng)才有epoll

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末沃测,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子食茎,更是在濱河造成了極大的恐慌蒂破,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,378評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件别渔,死亡現(xiàn)場(chǎng)離奇詭異附迷,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)哎媚,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門喇伯,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人拨与,你說我怎么就攤上這事稻据。” “怎么了截珍?”我有些...
    開封第一講書人閱讀 152,702評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵攀甚,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我岗喉,道長(zhǎng)秋度,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,259評(píng)論 1 279
  • 正文 為了忘掉前任钱床,我火速辦了婚禮荚斯,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘查牌。我一直安慰自己事期,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,263評(píng)論 5 371
  • 文/花漫 我一把揭開白布纸颜。 她就那樣靜靜地躺著兽泣,像睡著了一般。 火紅的嫁衣襯著肌膚如雪胁孙。 梳的紋絲不亂的頭發(fā)上唠倦,一...
    開封第一講書人閱讀 49,036評(píng)論 1 285
  • 那天称鳞,我揣著相機(jī)與錄音,去河邊找鬼稠鼻。 笑死冈止,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的候齿。 我是一名探鬼主播熙暴,決...
    沈念sama閱讀 38,349評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼慌盯!你這毒婦竟也來了周霉?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,979評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤润匙,失蹤者是張志新(化名)和其女友劉穎诗眨,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體孕讳,經(jīng)...
    沈念sama閱讀 43,469評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,938評(píng)論 2 323
  • 正文 我和宋清朗相戀三年巍膘,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了厂财。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,059評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡峡懈,死狀恐怖璃饱,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情肪康,我是刑警寧澤荚恶,帶...
    沈念sama閱讀 33,703評(píng)論 4 323
  • 正文 年R本政府宣布,位于F島的核電站磷支,受9級(jí)特大地震影響谒撼,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜雾狈,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,257評(píng)論 3 307
  • 文/蒙蒙 一廓潜、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧善榛,春花似錦辩蛋、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至咒循,卻和暖如春据途,著一層夾襖步出監(jiān)牢的瞬間绞愚,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工昨凡, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留爽醋,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,501評(píng)論 2 354
  • 正文 我出身青樓便脊,卻偏偏與公主長(zhǎng)得像蚂四,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子哪痰,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,792評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容