通過SocketChannel 和 ServerSocketChannel 寫一個阻塞的網(wǎng)絡(luò)IO操作

服務(wù)器

import java.io.IOException;
import java.net.InetSocketAddress;
import java.net.ServerSocket;
import java.nio.ByteBuffer;
import java.nio.channels.ClosedChannelException;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.util.Date;
import java.util.Iterator;
import java.util.Set;

public class SelectorServiceDemo {

    public static void main(String[] args) throws Exception {
        int port = 8000;

        // 通過open()方法找到Selector
        Selector selector = Selector.open();

        // 打開服務(wù)器的通道
        ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();

        // 服務(wù)器配置為非阻塞
        serverSocketChannel.configureBlocking(false);
        ServerSocket serverSocket = serverSocketChannel.socket();
        InetSocketAddress address = new InetSocketAddress(port);

        // 進行服務(wù)的綁定
        serverSocket.bind(address);

        // 注冊到selector郊供,等待連接
        serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);
        System.out.println("服務(wù)器運行,端口:" + port);

        // 數(shù)據(jù)緩沖區(qū)
        ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
        while (true) {
            if ((selector.select()) > 0) {

                // 選擇一組鍵,并且相應(yīng)的通道已經(jīng)準(zhǔn)備就緒
                // 取出全部生成的key
                Set<SelectionKey> selectedKeys = selector.selectedKeys();
                Iterator<SelectionKey> iter = selectedKeys.iterator();

                while (iter.hasNext()) {

                    SelectionKey key = iter.next();
                    if (key.isAcceptable()) {
                        acceptable(key, byteBuffer, selector);

                    } else if (key.isReadable() && key.isValid()) {
                        readable(selector, byteBuffer, key);

                    } else if (key.isWritable() && key.isValid()) {
                        writable(selector, byteBuffer, key);
                    }
                }

                // 清除全部的key
                selectedKeys.clear();
            }
        }
    }

    public static void writable(Selector selector, ByteBuffer byteBuffer, SelectionKey key)
            throws IOException, ClosedChannelException {
        SocketChannel client = (SocketChannel) key.channel();
        byteBuffer.clear();

        // 向緩沖區(qū)中設(shè)置內(nèi)容
        byteBuffer.put(("歡迎學(xué)習(xí)NIO教程").getBytes());
        byteBuffer.flip();

        // 輸出內(nèi)容
        client.write(byteBuffer);
        client.register(selector, SelectionKey.OP_READ);
    }

    public static void readable(Selector selector, ByteBuffer byteBuffer, SelectionKey key)
            throws IOException, ClosedChannelException {
        SocketChannel client = (SocketChannel) key.channel();
        byteBuffer.clear();

        // 讀取內(nèi)容到緩沖區(qū)中
        int readSize = client.read(byteBuffer);
        if (readSize > 0) {
            System.out.println("服務(wù)器端接受客戶端數(shù)據(jù):" + new String(byteBuffer.array(), 0, readSize));
            client.register(selector, SelectionKey.OP_WRITE);
        }
    }

    public static void acceptable(SelectionKey key, ByteBuffer byteBuffer, Selector selector) throws IOException {
        ServerSocketChannel server = (ServerSocketChannel) key.channel();

        // 接收新連接 和BIO寫法類是都是accept
        SocketChannel client = server.accept();
        // 配置為非阻塞
        client.configureBlocking(false);
        byteBuffer.clear();

        // 向緩沖區(qū)中設(shè)置內(nèi)容
        byteBuffer.put(("當(dāng)前的時間為:" + new Date()).getBytes());
        byteBuffer.flip();
        // 輸出內(nèi)容
        client.write(byteBuffer);
        client.register(selector, SelectionKey.OP_READ);
    }
}

結(jié)果

![圖一.png](https://upload-images.jianshu.io/upload_images/25693962-5ca5a931b4852ff5.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

客戶端

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.ClosedChannelException;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.SocketChannel;
import java.util.Date;
import java.util.Iterator;
import java.util.Set;

public class SelectorClientDemo {

    public static void main(String[] args) throws Exception {
        // 打開socket通道
        SocketChannel socketChannel = SocketChannel.open();

        // 設(shè)置為非阻塞方式
        socketChannel.configureBlocking(false);
        // 通過open()方法找到
        Selector selector = Selector.open();

        // 注冊連接服務(wù)端socket動作
        socketChannel.register(selector, SelectionKey.OP_CONNECT);

        // 連接
        socketChannel.connect(new InetSocketAddress("127.0.0.1", 8000));

        /* 數(shù)據(jù)緩沖區(qū) */
        ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
        while (true) {
            if ((selector.select()) > 0) {
                // 選擇一組鍵在辆,并且相應(yīng)的通道已經(jīng)準(zhǔn)備就緒
                Set<SelectionKey> selectedKeys = selector.selectedKeys();// 取出全部生成的key
                Iterator<SelectionKey> iter = selectedKeys.iterator();
                while (iter.hasNext()) {
                    SelectionKey key = iter.next(); // 取出每一個key
                    if (key.isConnectable()) {
                        connection(selector, byteBuffer, key);
                    } else if (key.isReadable() && key.isValid()) {
                        readable(selector, byteBuffer, key);
                    } else if (key.isWritable() && key.isValid()) {
                        writable(selector, byteBuffer, key);
                    }
                }
                selectedKeys.clear(); // 清楚全部的key
            }
        }
    }

    public static void writable(Selector selector, ByteBuffer byteBuffer, SelectionKey key)
            throws IOException, ClosedChannelException {
        SocketChannel client = (SocketChannel) key.channel();
        byteBuffer.clear();

        // 向緩沖區(qū)中設(shè)置內(nèi)容
        byteBuffer.put(("學(xué)習(xí)NIO!").getBytes());
        byteBuffer.flip();
        // 輸出內(nèi)容
        client.write(byteBuffer);
        client.register(selector, SelectionKey.OP_READ);
    }

    public static void readable(Selector selector, ByteBuffer byteBuffer, SelectionKey key)
            throws IOException, ClosedChannelException {
        SocketChannel client = (SocketChannel) key.channel();
        byteBuffer.clear();

        // 讀取內(nèi)容到緩沖區(qū)中
        int readSize = client.read(byteBuffer);
        if (readSize > 0) {
            System.out.println("客戶端接受服務(wù)器端數(shù)據(jù):" + new String(byteBuffer.array(), 0, readSize));
            client.register(selector, SelectionKey.OP_WRITE);
        }
    }

    public static void connection(Selector selector, ByteBuffer byteBuffer, SelectionKey key)
            throws IOException, ClosedChannelException {
        SocketChannel client = (SocketChannel) key.channel();
        if (client.isConnectionPending()) {
            client.finishConnect();
            byteBuffer.clear();

            // 向緩沖區(qū)中設(shè)置內(nèi)容
            byteBuffer.put(("isConnect,當(dāng)前的時間為:" + new Date()).getBytes());
            byteBuffer.flip();
            // 輸出內(nèi)容
            client.write(byteBuffer);
        }
        client.register(selector, SelectionKey.OP_READ);
    }
}  
結(jié)果
圖二.png
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子别渔,更是在濱河造成了極大的恐慌,老刑警劉巖惧互,帶你破解...
    沈念sama閱讀 211,290評論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件哎媚,死亡現(xiàn)場離奇詭異,居然都是意外死亡壹哺,警方通過查閱死者的電腦和手機抄伍,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,107評論 2 385
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來管宵,“玉大人,你說我怎么就攤上這事攀甚÷崞樱” “怎么了?”我有些...
    開封第一講書人閱讀 156,872評論 0 347
  • 文/不壞的土叔 我叫張陵秋度,是天一觀的道長炸庞。 經(jīng)常有香客問我,道長荚斯,這世上最難降的妖魔是什么埠居? 我笑而不...
    開封第一講書人閱讀 56,415評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮事期,結(jié)果婚禮上滥壕,老公的妹妹穿的比我還像新娘。我一直安慰自己兽泣,他們只是感情好绎橘,可當(dāng)我...
    茶點故事閱讀 65,453評論 6 385
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著唠倦,像睡著了一般称鳞。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上稠鼻,一...
    開封第一講書人閱讀 49,784評論 1 290
  • 那天冈止,我揣著相機與錄音,去河邊找鬼候齿。 笑死熙暴,一個胖子當(dāng)著我的面吹牛闺属,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播怨咪,決...
    沈念sama閱讀 38,927評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼屋剑,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了诗眨?” 一聲冷哼從身側(cè)響起唉匾,我...
    開封第一講書人閱讀 37,691評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎匠楚,沒想到半個月后巍膘,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,137評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡芋簿,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,472評論 2 326
  • 正文 我和宋清朗相戀三年峡懈,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片与斤。...
    茶點故事閱讀 38,622評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡肪康,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出撩穿,到底是詐尸還是另有隱情磷支,我是刑警寧澤,帶...
    沈念sama閱讀 34,289評論 4 329
  • 正文 年R本政府宣布食寡,位于F島的核電站雾狈,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏抵皱。R本人自食惡果不足惜善榛,卻給世界環(huán)境...
    茶點故事閱讀 39,887評論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望呻畸。 院中可真熱鬧移盆,春花似錦、人聲如沸擂错。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,741評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽钮呀。三九已至剑鞍,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間爽醋,已是汗流浹背蚁署。 一陣腳步聲響...
    開封第一講書人閱讀 31,977評論 1 265
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留蚂四,地道東北人光戈。 一個月前我還...
    沈念sama閱讀 46,316評論 2 360
  • 正文 我出身青樓哪痰,卻偏偏與公主長得像,于是被迫代替她去往敵國和親久妆。 傳聞我的和親對象是個殘疾皇子晌杰,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,490評論 2 348

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