日日操夜夜添-日日操影院-日日草夜夜操-日日干干-精品一区二区三区波多野结衣-精品一区二区三区高清免费不卡

公告:魔扣目錄網(wǎng)為廣大站長(zhǎng)提供免費(fèi)收錄網(wǎng)站服務(wù),提交前請(qǐng)做好本站友鏈:【 網(wǎng)站目錄:http://www.ylptlb.cn 】, 免友鏈快審服務(wù)(50元/站),

點(diǎn)擊這里在線咨詢客服
新站提交
  • 網(wǎng)站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會(huì)員:747

NIO

(1)基本介紹

1)JAVA NIO全程 java non-blocking IO,是指JDK提供的新API。從JDK1.4開始,Java提供了一系列改進(jìn)的輸入/輸出的新特性,被統(tǒng)稱為NIO,是同步非阻塞的

2)NIO相關(guān)類都被放在java.nio包及子包下,并且對(duì)原java.io包中的很多類進(jìn)行改寫

3)NIO有三大核心部分:Channel(通道),Buffer(緩沖區(qū)),Selector(選擇器)

4)NIO是面向緩沖區(qū),或者面向塊編程的。數(shù)據(jù)讀取到一個(gè)它稍后處理的緩沖區(qū),需要時(shí)可在緩沖區(qū)中前后移動(dòng),這就增加了處理過程中的靈活性,使用它可以提供非阻塞式高伸縮性網(wǎng)絡(luò)

5)Java NIO的非阻塞模式,使一個(gè)線程從某通道發(fā)送請(qǐng)求或者讀取數(shù)據(jù),但是它僅能得到目前可用的數(shù)據(jù),如果目前沒有數(shù)據(jù)可用時(shí),就什么都不會(huì)獲取,而不是保持線程阻塞,所以直至數(shù)據(jù)變的可以讀取之前,該線程可以繼續(xù)做其它的事情。非阻塞也是如此,一個(gè)線程請(qǐng)求寫入一些數(shù)據(jù)到某通道,但不需要等待它完全寫入,這個(gè)線程同時(shí)可以去做別的事情

6)HTTP2.0使用了多路復(fù)用的技術(shù),做到同一個(gè)連接并發(fā)處理多個(gè)請(qǐng)求,并且并發(fā)請(qǐng)求的數(shù)量比HTTP1.1大了好幾個(gè)數(shù)量級(jí)

(2)NIO和BIO的比較

1)BIO以流的方式處理數(shù)據(jù),而NIO以塊的方式處理數(shù)據(jù),塊I/O的效率比流I/O高很多

2)BIO是阻塞的,NIO是非阻塞的

3)BIO基于字節(jié)流和字符流進(jìn)行操作,而NIO基于Channel(通道)和Buffer(緩沖區(qū))進(jìn)行操作,數(shù)據(jù)總是從通道讀取到緩沖區(qū)中,或者從緩沖區(qū)寫入到通道中。Selector(選擇器)用于監(jiān)聽多個(gè)通道的時(shí)間(比如:連接請(qǐng)求,數(shù)據(jù)到達(dá)等),因此使用單個(gè)線程就可以監(jiān)聽多個(gè)客戶端通道

(3)NIO三大核心原理

 

程序員:java NIO模型,三大核心原理,不防來看看

 

每個(gè)channel都會(huì)對(duì)應(yīng)一個(gè)Buffer

2)Selector 對(duì)應(yīng)一個(gè)線程, 一個(gè)線程對(duì)應(yīng)多個(gè)channel(連接)

3)該圖反應(yīng)了有三個(gè)channel注冊(cè)到該selector//程序

4)程序切換到哪個(gè)channel是由事件決定的,Event就是一個(gè)重要的概念

5)Selector 會(huì)根據(jù)不同的事件,在各個(gè)通道上切換

6)Buffer 就是一個(gè)內(nèi)存塊 , 底層是有一個(gè)數(shù)組

7)數(shù)據(jù)的讀取寫入是通過Buffer,這個(gè)和BIO,BIO中或者是輸入流,或者是輸出流, 不能雙向,但是 NIO 的 Buffer 是可以讀也可以寫, 需要 flip 方法切換。channel 是雙向的, 可以返回底層操作系統(tǒng)的情況, 比如 linux ,底層的操作系統(tǒng)通道就是雙向的

緩沖區(qū)(Buffer)

(1)基本介紹

緩沖區(qū)(Buffer):緩沖區(qū)本質(zhì)上是一個(gè)可以讀寫數(shù)據(jù)的內(nèi)存塊,可以理解成是一個(gè)容器對(duì)象(含數(shù)組),該對(duì)象提供了一組方法,可以更輕松地使用內(nèi)存塊,緩沖區(qū)對(duì)象內(nèi)置了一些機(jī)制,能夠跟蹤和記錄緩沖區(qū)的狀態(tài)變化情況。Channel 提供從文件、網(wǎng)絡(luò)讀取數(shù)據(jù)的渠道,但是讀取或?qū)懭氲臄?shù)據(jù)都必須經(jīng)由 Buffer

程序員:java NIO模型,三大核心原理,不防來看看

 

2)Buffer及其子類

1)在 NIO 中,Buffer 是一個(gè)頂層父類,它是一個(gè)抽象類, 類的層級(jí)關(guān)系圖:

程序員:java NIO模型,三大核心原理,不防來看看

 

 

程序員:java NIO模型,三大核心原理,不防來看看

 

2)Buffer類定義了所有的緩沖區(qū)都具有的四個(gè)屬性來提供關(guān)于其所包含的數(shù)據(jù)元素的信息:

程序員:java NIO模型,三大核心原理,不防來看看

 

 

程序員:java NIO模型,三大核心原理,不防來看看

 

常用方法

程序員:java NIO模型,三大核心原理,不防來看看

 

 通道(Channel)

(1)基本介紹

NIO的通道類似于流,但有些區(qū)別如下:

通道可以同時(shí)進(jìn)行讀寫,而流只能讀或者寫

通道可以實(shí)現(xiàn)異步讀寫數(shù)據(jù)

通道可以從緩沖讀數(shù)據(jù),也可以寫數(shù)據(jù)到緩沖

BIO中的stream是單向的,例如FileInputStream對(duì)象只能進(jìn)行讀取數(shù)據(jù)的操作,而NIO中的通道(Channel)是雙向的,可以讀操作,也可以寫操作

常見的Channel類有:FileChannel、DatagramChannel、ServerSocketChannel和SocketChannel。

FileChannel用于文件的數(shù)據(jù)讀寫,DatagramChannel用于UDP的數(shù)據(jù)讀寫,ServerSocketChannel和SocketChannel用于TCP的數(shù)據(jù)讀寫

(2)FileChannel

 

程序員:java NIO模型,三大核心原理,不防來看看

 

(3)案例1-本地文件寫數(shù)據(jù)

public class NIOFileChannel01 {

 public static void main(String[] args) throws Exception{


 String str = "hello,NIO";

 //創(chuàng)建一個(gè)輸出流->channel

 FileOutputStream fileOutputStream = new FileOutputStream("/Users/Apple/學(xué)習(xí)/study/test01.txt");


 //通過 fileOutputStream 獲取 對(duì)應(yīng)的 FileChannel

 //這個(gè) fileChannel 真實(shí) 類型是 FileChannelImpl

 FileChannel fileChannel = fileOutputStream.getChannel();


 //創(chuàng)建一個(gè)緩沖區(qū) ByteBuffer

 ByteBuffer byteBuffer = ByteBuffer.allocate(1024);


 //將 str 放入 byteBuffer

 byteBuffer.put(str.getBytes());


 //對(duì)byteBuffer 進(jìn)行flip

 byteBuffer.flip();

 //將byteBuffer 數(shù)據(jù)寫入到 fileChannel

 fileChannel.write(byteBuffer);

 fileOutputStream.close();


 }

}

(4)案例2-本地文件讀數(shù)據(jù)

public class NIOFileChannel02 {

public static void main(String[] args) throws Exception {

//創(chuàng)建文件的輸入流

File file = new File("/Users/apple/學(xué)習(xí)/study/test01.txt");

FileInputStream fileInputStream = new FileInputStream(file);

//通過fileInputStream 獲取對(duì)應(yīng)的FileChannel -> 實(shí)際類型 FileChannelImpl

FileChannel fileChannel = fileInputStream.getChannel();

//創(chuàng)建緩沖區(qū)

ByteBuffer byteBuffer = ByteBuffer.allocate((int) file.length());

//將通道的數(shù)據(jù)讀入到Buffer

fileChannel.read(byteBuffer);

?

//將byteBuffer 的 字節(jié)數(shù)據(jù) 轉(zhuǎn)成String

System.out.println(new String(byteBuffer.array()));

fileInputStream.close();

?

}

}

(5)案例3-使用Buffer完成文件的讀取、寫入

public class NIOFileChannel03 {

public static void main(String[] args) throws Exception {

?

FileInputStream fileInputStream = new FileInputStream("1.txt");

FileChannel fileChannel01 = fileInputStream.getChannel();

?

FileOutputStream fileOutputStream = new FileOutputStream("2.txt");

FileChannel fileChannel02 = fileOutputStream.getChannel();

?

ByteBuffer byteBuffer = ByteBuffer.allocate(512);

?

//循環(huán)讀取

while (true) {

?

//這里有一個(gè)重要的操作,一定不要忘了

/*

public final Buffer clear() {

position = 0;

limit = capacity;

mark = -1;

return this;

}

*/

byteBuffer.clear(); //清空buffer

int read = fileChannel01.read(byteBuffer);

System.out.println("read =" + read);

//表示讀完

if (read == -1) {

break;

}

//將buffer 中的數(shù)據(jù)寫入到 fileChannel02 -- 2.txt

byteBuffer.flip();

fileChannel02.write(byteBuffer);

}

?

//關(guān)閉相關(guān)的流

fileInputStream.close();

fileOutputStream.close();

}

}

(6)案例4-拷貝文件 transferFrom 方法

public class NIOFileChannel04 {

public static void main(String[] args) throws Exception {

?

//創(chuàng)建相關(guān)流

FileInputStream fileInputStream = new FileInputStream("a.jpg");

FileOutputStream fileOutputStream = new FileOutputStream("a1.jpg");

?

//獲取各個(gè)流對(duì)應(yīng)的fileChannel

FileChannel sourceCh = fileInputStream.getChannel();

FileChannel destCh = fileOutputStream.getChannel();

?

//使用transferForm完成拷貝

destCh.transferFrom(sourceCh,0,sourceCh.size());

//關(guān)閉相關(guān)通道和流

sourceCh.close();

destCh.close();

fileInputStream.close();

fileOutputStream.close();

}

}

(7)案例5-零拷貝文件-transferTo文件

零拷貝參考資料:

https://www.cnblogs.com/yibutian/p/9482640.html

http://www.360doc.com/content/19/0528/13/99071_838741319.shtml

public class NewIOClient {

public static void main(String[] args) throws Exception {

String filename = "/Users/apple/password.txt";

?

//得到一個(gè)文件channel

FileChannel fileChannel = new FileInputStream(filename).getChannel();

FileChannel fileChannel1 = new FileOutputStream("/Users/apple/password1.txt").getChannel();

?

//準(zhǔn)備發(fā)送

long startTime = System.currentTimeMillis();

?

//在linux下一個(gè)transferTo 方法就可以完成傳輸

//在windows 下 一次調(diào)用 transferTo 只能發(fā)送8m,就需要分段傳輸文件

//transferTo 底層使用到零拷貝

long transferCount = fileChannel.transferTo(0, fileChannel.size(), fileChannel1);

?

System.out.println("發(fā)送的總的字節(jié)數(shù) =" + transferCount + " 耗時(shí):" + (System.currentTimeMillis() - startTime));

?

//關(guān)閉

fileChannel.close();

?

}

}

?

(8)注意事項(xiàng)和細(xì)節(jié)

1)ByteBuffer 支持類型化的 put 和 get, put 放入的是什么數(shù)據(jù)類型,get 就應(yīng)該使用相應(yīng)的數(shù)據(jù)類型來取出,否則可能有 BufferUnderflowException 異常

public class NIOByteBufferPutGet {

public static void main(String[] args) {

? //創(chuàng)建一個(gè)Buffer

ByteBuffer buffer = ByteBuffer.allocate(64);

?

//類型化方式放入數(shù)據(jù)

buffer.putInt(100);

buffer.putLong(9);

buffer.putChar('a');

buffer.putShort((short) 4);

?

//取出

buffer.flip();

?

System.out.println();

?

System.out.println(buffer.getInt());

System.out.println(buffer.getLong());

System.out.println(buffer.getChar());

System.out.println(buffer.getLong());

}

}

2)可以將一個(gè)普通Buffer轉(zhuǎn)成只讀Buffer

public class ReadOnlyBuffer {

public static void main(String[] args) {

?

//創(chuàng)建一個(gè)buffer

ByteBuffer buffer = ByteBuffer.allocate(64);

?

for(int i = 0; i < 64; i++) {

buffer.put((byte)i);

}

?

//讀取

buffer.flip();

?

//得到一個(gè)只讀的Buffer

ByteBuffer readOnlyBuffer = buffer.asReadOnlyBuffer();

System.out.println(readOnlyBuffer.getClass());

?

//讀取

while (readOnlyBuffer.hasRemaining()) {

System.out.println(readOnlyBuffer.get());

}

?

readOnlyBuffer.put((byte)100); //ReadOnlyBufferException

}

}

3)NIO 還提供了 MappedByteBuffer, 可以讓文件直接在內(nèi)存(堆外的內(nèi)存)中進(jìn)行修改

public class MappedByteBufferTest {

public static void main(String[] args) throws Exception {

?

RandomaccessFile randomAccessFile = new RandomAccessFile("/Users/apple/學(xué)習(xí)/study/test01.txt", "rw");

//獲取對(duì)應(yīng)的通道

FileChannel channel = randomAccessFile.getChannel();

?

/**

* 參數(shù)1: FileChannel.MapMode.READ_WRITE 使用的讀寫模式

* 參數(shù)2: 0 : 可以直接修改的起始位置

* 參數(shù)3: 5: 是映射到內(nèi)存的大小(不是索引位置) ,即將 1.txt 的多少個(gè)字節(jié)映射到內(nèi)存

* 可以直接修改的范圍就是 0-5

* 實(shí)際類型 DirectByteBuffer

*/

MappedByteBuffer mappedByteBuffer = channel.map(FileChannel.MapMode.READ_WRITE, 0, 5);

?

mappedByteBuffer.put(0, (byte) 'H');

mappedByteBuffer.put(3, (byte) '9');

// mappedByteBuffer.put(5, (byte) 'Y');//IndexOutOfBoundsException

?

randomAccessFile.close();

System.out.println("修改成功~~");

}

}

4)NIO 還支持 通過多個(gè) Buffer (即 Buffer 數(shù)組) 完成讀寫操作,即 Scattering 和 Gathering

/**

* Scattering:將數(shù)據(jù)寫入到buffer時(shí),可以采用buffer數(shù)組,依次寫入 [分散]

* Gathering: 從buffer讀取數(shù)據(jù)時(shí),可以采用buffer數(shù)組,依次讀

*/

public class ScatteringAndGatheringTest {

public static void main(String[] args) throws Exception {

?

//使用 ServerSocketChannel 和 SocketChannel 網(wǎng)絡(luò)

?

ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();

InetSocketAddress inetSocketAddress = new InetSocketAddress(7000);

?

//綁定端口到socket ,并啟動(dòng)

serverSocketChannel.socket().bind(inetSocketAddress);

?

//創(chuàng)建buffer數(shù)組

ByteBuffer[] byteBuffers = new ByteBuffer[2];

byteBuffers[0] = ByteBuffer.allocate(5);

byteBuffers[1] = ByteBuffer.allocate(3);

?

//等客戶端連接(telnet)

SocketChannel socketChannel = serverSocketChannel.accept();

//假定從客戶端接收8個(gè)字節(jié)

int messageLength = 8;

//循環(huán)的讀取

while (true) {

?

int byteRead = 0;

?

while (byteRead < messageLength) {

long l = socketChannel.read(byteBuffers);

//累計(jì)讀取的字節(jié)數(shù)

byteRead += l;

System.out.println("byteRead=" + byteRead);

//使用流打印, 看看當(dāng)前的這個(gè)buffer的position 和 limit

Arrays.stream(byteBuffers).map(buffer -> "position=" + buffer.position() + ", limit=" + buffer.limit()).forEach(System.out::println);

}

?

//將所有的buffer進(jìn)行flip

Arrays.asList(byteBuffers).forEach(Buffer::flip);

?

//將數(shù)據(jù)讀出顯示到客戶端

long byteWirte = 0;

while (byteWirte < messageLength) {

long l = socketChannel.write(byteBuffers);

byteWirte += l;

}

?

//將所有的buffer 進(jìn)行clear

Arrays.asList(byteBuffers).forEach(Buffer::clear);

?

System.out.println("byteRead:=" + byteRead + " byteWrite=" + byteWirte + ", messageLength" + messageLength);

}

?

}

}

選擇器(Selector)

(1)基本介紹

1)Java的NIO,用非阻塞的IO方式。可以用一個(gè)線程,處理多個(gè)的客戶端連接,就會(huì)用到選擇器

2)Selector能夠檢測(cè)多個(gè)注冊(cè)的通道上是否有事件發(fā)生。如果有事件發(fā)生,便獲取事件然后針對(duì)每個(gè)事件進(jìn)行相應(yīng)的處理。這樣就可以只用一個(gè)單線程去管理多個(gè)通道,也就是管理多個(gè)連接和請(qǐng)求

3)只有在連接/通道真正有讀寫事件發(fā)生時(shí),才會(huì)進(jìn)行讀寫,就大大地減少了系統(tǒng)開銷,并且不必為每個(gè)連接都創(chuàng)建一個(gè)線程,不用去維護(hù)多個(gè)線程

4)避免了多線程之間的上下文切換導(dǎo)致的開銷

(2)常用方法

 

程序員:java NIO模型,三大核心原理,不防來看看

 

(3)代碼示例

NIOServer.java

public class NIOServer {

public static void main(String[] args) throws Exception {

?

//創(chuàng)建ServerSocketChannel -> ServerSocket

?

ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();

?

//得到一個(gè)Selector對(duì)象

Selector selector = Selector.open();

?

//綁定一個(gè)端口6666, 在服務(wù)器端監(jiān)聽

serverSocketChannel.socket().bind(new InetSocketAddress(6666));

//設(shè)置為非阻塞

serverSocketChannel.configureBlocking(false);

?

//把 serverSocketChannel 注冊(cè)到 selector 關(guān)心 事件為 OP_ACCEPT

serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);

?

System.out.println("注冊(cè)后的selectionkey 數(shù)量=" + selector.keys().size());

?

?

//循環(huán)等待客戶端連接

while (true) {

?

//這里我們等待1秒,如果沒有事件發(fā)生, 返回

if (selector.select(1000) == 0) {

System.out.println("服務(wù)器等待了1秒,無連接");

continue;

}

?

//如果返回的>0, 就獲取到相關(guān)的 selectionKey集合

//1.如果返回的>0, 表示已經(jīng)獲取到關(guān)注的事件

//2. selector.selectedKeys() 返回關(guān)注事件的集合

// 通過 selectionKeys 反向獲取通道

Set<SelectionKey> selectionKeys = selector.selectedKeys();

System.out.println("selectionKeys 數(shù)量 = " + selectionKeys.size());

?

//遍歷 Set<SelectionKey>, 使用迭代器遍歷

Iterator<SelectionKey> keyIterator = selectionKeys.iterator();

?

while (keyIterator.hasNext()) {

//獲取到SelectionKey

SelectionKey key = keyIterator.next();

//根據(jù)key 對(duì)應(yīng)的通道發(fā)生的事件做相應(yīng)處理

//如果是 OP_ACCEPT, 有新的客戶端連接

if (key.isAcceptable()) {

//該該客戶端生成一個(gè) SocketChannel

SocketChannel socketChannel = serverSocketChannel.accept();

System.out.println("客戶端連接成功 生成了一個(gè) socketChannel " + socketChannel.hashCode());

//將 SocketChannel 設(shè)置為非阻塞

socketChannel.configureBlocking(false);

//將socketChannel 注冊(cè)到selector, 關(guān)注事件為 OP_READ, 同時(shí)給socketChannel

//關(guān)聯(lián)一個(gè)Buffer

socketChannel.register(selector, SelectionKey.OP_READ, ByteBuffer.allocate(1024));

?

System.out.println("客戶端連接后 ,注冊(cè)的selectionkey 數(shù)量=" + selector.keys().size()); //2,3,4..

?

?

}

//發(fā)生 OP_READ

if (key.isReadable()) {

?

//通過key 反向獲取到對(duì)應(yīng)channel

SocketChannel channel = (SocketChannel) key.channel();

?

//獲取到該channel關(guān)聯(lián)的buffer

ByteBuffer buffer = (ByteBuffer) key.attachment();

channel.read(buffer);

System.out.println("form 客戶端 " + new String(buffer.array()));

?

}

?

//手動(dòng)從集合中移動(dòng)當(dāng)前的selectionKey, 防止重復(fù)操作

keyIterator.remove();

?

}

?

}

?

}

}

NIOClient.java

public class NIOClient {

public static void main(String[] args) throws Exception{

?

//得到一個(gè)網(wǎng)絡(luò)通道

SocketChannel socketChannel = SocketChannel.open();

//設(shè)置非阻塞

socketChannel.configureBlocking(false);

//提供服務(wù)器端的ip 和 端口

InetSocketAddress inetSocketAddress = new InetSocketAddress("127.0.0.1", 6666);

//連接服務(wù)器

if (!socketChannel.connect(inetSocketAddress)) {

?

while (!socketChannel.finishConnect()) {

System.out.println("因?yàn)檫B接需要時(shí)間,客戶端不會(huì)阻塞,可以做其它工作..");

}

}

?

//...如果連接成功,就發(fā)送數(shù)據(jù)

String str = "hello, Selector~";

//Wraps a byte array into a buffer

ByteBuffer buffer = ByteBuffer.wrap(str.getBytes());

//發(fā)送數(shù)據(jù),將 buffer 數(shù)據(jù)寫入 channel

socketChannel.write(buffer);

System.in.read();

?

}

}

(4)SelectionKey

SelectionKey,表示 Selector 和網(wǎng)絡(luò)通道的注冊(cè)關(guān)系, 共四種:

int OP_ACCEPT:有新的網(wǎng)絡(luò)連接可以 accept,值為 16

int OP_CONNECT:代表連接已經(jīng)建立,值為 8

int OP_READ:代表讀操作,值為 1

int OP_WRITE:代表寫操作,值為 4

相關(guān)方法:

 

程序員:java NIO模型,三大核心原理,不防來看看

 

分享到:
標(biāo)簽:模型 NIO
用戶無頭像

網(wǎng)友整理

注冊(cè)時(shí)間:

網(wǎng)站:5 個(gè)   小程序:0 個(gè)  文章:12 篇

  • 51998

    網(wǎng)站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會(huì)員

趕快注冊(cè)賬號(hào),推廣您的網(wǎng)站吧!
最新入駐小程序

數(shù)獨(dú)大挑戰(zhàn)2018-06-03

數(shù)獨(dú)一種數(shù)學(xué)游戲,玩家需要根據(jù)9

答題星2018-06-03

您可以通過答題星輕松地創(chuàng)建試卷

全階人生考試2018-06-03

各種考試題,題庫,初中,高中,大學(xué)四六

運(yùn)動(dòng)步數(shù)有氧達(dá)人2018-06-03

記錄運(yùn)動(dòng)步數(shù),積累氧氣值。還可偷

每日養(yǎng)生app2018-06-03

每日養(yǎng)生,天天健康

體育訓(xùn)練成績(jī)?cè)u(píng)定2018-06-03

通用課目體育訓(xùn)練成績(jī)?cè)u(píng)定