成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

探秘JDK 7之四:下一代I/O(NIO.2)

開發(fā) 后端
本文為51CTO.com探秘JDK 7系列的最后一篇。在前三篇當中,我們詳細介紹了JDK 7的語言特性等新特點,本文將介紹JDK 7中的下一代I/O(NIO.2)。

51CTO曾在Java 7 下一代Java開發(fā)技術詳解專題里對“JDK 7 I/O新功能”有過簡單地介紹,其實早在2000年的時候,Sun公司就啟動了JSR 51:為Java平臺開發(fā)新的I/O API,直接訪問操作系統(tǒng)底層輸入/輸出操作以提高應用程序的性能,首次引入這套API是在J2SE 1.4中,根據(jù)維基百科的新I/O詞條顯示,新I/O(NIO)由下列API組成:

◆ 原始類型數(shù)據(jù)緩沖

◆ 字符集編碼和解碼

◆ 通道,新的原始I/O抽象

◆ 支持上鎖和內存映射的文件接口,文件最大支持Integer.MAX_VALUE字節(jié)(2GB)

◆ 為可擴展服務器提供的多路復用,無阻塞I/O設施(基于選擇器和鍵)

JSR 203(NIO.2)除了解決JSR 51遺留下來的問題外,還為Java平臺提供了更多新的I/O API,NIO.2解決了java.awt.File文件系統(tǒng)接口存在的重大問題,引入了異步I/O,并完成了未包括在JSR 51中的功能,下面列出了包含在JSR 203中的主要組件:

◆ 新的文件系統(tǒng)接口,支持大塊訪問文件屬性,更改通知,繞開文件系統(tǒng)指定的API,也是可插拔文件系統(tǒng)實現(xiàn)的服務提供者接口。

◆ 對套接字和文件同時提供了異步I/O操作的API。

◆ JSR 51中定義的完整的套接字通道功能,此外還包括綁定,選項配置和多播數(shù)據(jù)報的支持。

新的文件系統(tǒng)接口

Java的File類存在重大問題,例如,操作出錯時,delete()和mkdir()方法返回一個狀態(tài)碼而不是一個異常,沒有辦法獲知失敗的原因,此外還包括以下問題:

◆ File沒有提供方法來檢測符號鏈接,要知道為什么檢測符號鏈接很重要,以及如何解決這個問題的辦法,請參考Patrick的文章“在Java中如何處理文件系統(tǒng)軟鏈接/符號鏈接”和“Java中的鏈接/別名/快捷方式”。

◆ File提供的方法只能訪問部分文件屬性,不能訪問文件權限和訪問控制列表。

◆ File沒有提供方法一次訪問文件的所有屬性(如文件的修改時間和它的類型),因為文件系統(tǒng)需要為每個屬性執(zhí)行查詢請求,可能存在性能問題。

◆ File的list()和listFiles()方法返回文件名和目錄名的數(shù)組,但不支持大目錄,通過網絡展示大目錄清單時,調用list()/listFiles()方法可能會使當前的線程阻塞相當長一段時間,而在服務器端,虛擬機可能會耗盡內存。

◆ File沒有提供復制和移動文件的方法,雖然File提供了一個renameTo()方法在某些時候可以用來移動文件,但它的行為與平臺關系緊密,即在不同平臺上的行為是不一致的,根據(jù)renameTo()的文檔說明,這個方法不能在文件系統(tǒng)之間移動文件,它可能不是原子的,如果目標路徑下已存在同名文件,這個操作可能不會成功。

◆ File也沒有提供改變通知方法,需要應用程序自己實現(xiàn),因此導致應用程序的性能下降,例如,服務器需要確定什么時候往目錄中添加了一個新的JAR文件,它需要實時監(jiān)視這個目錄,因為服務器后臺線程需要頻繁讀取文件系統(tǒng),因此性能會有所下降。

◆ File也不允許開發(fā)人員引入他們自己的文件系統(tǒng)訪問功能,例如,開發(fā)人員可能想將文件系統(tǒng)存儲到一個zip文件中,或創(chuàng)建一個內存文件系統(tǒng)。

NIO.2引入了新的文件系統(tǒng)接口,除了解決上述存在的問題外,還引入了更多的功能,這個接口由位于java.nio.file,java.nio.file.attribute和java.nio.file.spi包中的類和其它類型組成。

這些包提供了多個切入點,其中一個切入點就是java.nio.file.Paths類,它提供了兩個方法返回一個java.nio.file.Path實例:

◆ public static Path get(String path) – 它通過轉換給定路徑字符串返回給這個實例構造一個Path實例。

◆ public static Path get(URI uri) -它通過轉換給定路徑的URI(統(tǒng)一資源定位符)返回給這個實例構造一個Path實例。

與傳統(tǒng)的基于File的代碼互操作:

File類提供了一個public Path toPath()方法,它可以將一個File實例轉換成一個Path實例。

當你創(chuàng)建了一個Path實例后,你就可以使用這個實例執(zhí)行許多路徑操作(如返回路徑的一部分,連接兩個路徑)和許多文件操作(如刪除,移動和復制文件)。

為了不將問題復雜化,我就不深入講解Path了,這里我用一段代碼簡單地演示一下以前的get()方法和Path的delete()方法。

清單1. InformedDelete.java

  1. // InformedDelete.java  
  2. import java.io.IOException;  
  3. import java.nio.file.DirectoryNotEmptyException;  
  4. import java.nio.file.NoSuchFileException;  
  5. import java.nio.file.Path;  
  6. import java.nio.file.Paths;  
  7. public class InformedDelete  
  8. {  
  9.    public static void main (String [] args)  
  10.    {  
  11.       if (args.length != 1)  
  12.       {  
  13.           System.err.println ("usage: java InformedDelete path");  
  14.           return;  
  15.       }  
  16.       // Attempt to construct a Path instance by converting the path argument  
  17.       // string. If unsuccessful (you passed an empty string as the  
  18.       // command-line argument), the get() method throws an instance of the  
  19.       // unchecked java.nio.file.InvalidPathException class.  
  20.       Path path = Paths.get (args [0]);  
  21.       try  
  22.       {  
  23.           path.delete (); // Attempt to delete the path.  
  24.       }  
  25.       catch (NoSuchFileException e)  
  26.       {  
  27.           System.err.format ("%s: no such file or directory%n", path);  
  28.       }  
  29.       catch (DirectoryNotEmptyException e)  
  30.       {  
  31.           System.err.format ("%s: directory not empty%n", path);  
  32.       }  
  33.       catch (IOException e)  
  34.       {  
  35.           System.err.format ("%s: %s%n", path, e);  
  36.       }  
  37.    }  
  38. }  
  39.  

InformedDelete調用Path的delete()方法解決了File的delete()方法不能確定失敗原因的問題,當Path的delete()當?shù)臋z測到操作失敗時,它會根據(jù)情況拋出適當?shù)漠惓#纾?/p>

◆ 如果文件不存在,拋出java.nio.file.NoSuchFileException異常。

◆ 如果文件是一個目錄不能刪除,拋出java.nio.file.DirectoryNotEmptyException異常,因為這個目錄下可能還包括一個空目錄。

◆ 如果遇到其他I/O問題,則拋出java.io.IOException的子類異常,例如,如果文件是只讀的,拋出java.nio.file.AccessDeniedException異常。

#p#

異步I/O

JSR 51引入了多路復用I/O(無阻塞I/O和選擇就緒的結合)使創(chuàng)建高可擴展服務器變得更加容易,本質上是這樣的,客戶端代碼用一個選擇器注冊一個套接字通道,當通道準備好可以開始I/O操作時發(fā)出通知。

如果要深入研究多路復用I/O,請閱讀Ron Hitchens的《Java NIO》一書。

JSR 203還引入了異步I/O,它也被用來建立高可擴展服務器,和多路復用I/O不同,異步I/O是讓客戶端啟動一個I/O操作,當操作完成后向客戶端發(fā)送一個通知。

異步I/O是通過以下位于java.nio.channels包中的接口和類實現(xiàn)的,它們的名稱前面都加了Asynchronous前綴:

◆ AsynchronousChannel – 標識一個支持異步I/O的通道。

◆ AsynchronousByteChannel – 標識一個支持讀寫字節(jié)的異步通道,這個接口擴展了AsynchronousChannel。

◆ AsynchronousDatagramChannel – 標識一個面向數(shù)據(jù)報套接字異步通道,這個類實現(xiàn)了AsynchronousByteChannel。

◆ AsynchronousFileChannel – 標識一個可讀,寫和操作文件的異步通道,這個類實現(xiàn)了AsynchronousChannel。

◆ AsynchronousServerSocketChannel – 標識一個面向流監(jiān)聽套接字的異步通道,這個類實現(xiàn)了AsynchronousChannel。

◆ AsynchronousSocketChannel – 標識一個面向流連接套接字的異步通道,這個類實現(xiàn)了AsynchronousByteChannel。

◆ AsynchronousChannelGroup – 標識一個用于資源共享的異步通道組。

AsynchronousChannel文檔指定了兩種形式的異步I/O操作:

◆ Future operation(...)

◆ void operation(... A attachment, CompletionHandler handler)

operation列舉I/O操作(如讀,寫),V是操作的結果類型,A是附加給操作的對象類型。

第一種形式需要你調用java.util.concurrent.Future方法檢查操作是否完成,等待完成和檢索結果,清單2的代碼演示了這樣一個示例。

清單2. AFCDemo1.java

  1. // AFCDemo1.java  
  2. import java.io.IOException;  
  3. import java.nio.ByteBuffer;  
  4. import java.nio.channels.AsynchronousFileChannel;  
  5. import java.nio.file.Path;  
  6. import java.nio.file.Paths;  
  7. import java.util.concurrent.Future;  
  8. public class AFCDemo1  
  9. {  
  10.    public static void main (String [] args) throws Exception  
  11.    {  
  12.       if (args.length != 1)  
  13.       {  
  14.           System.err.println ("usage: java AFCDemo1 path");  
  15.           return;  
  16.       }  
  17.       Path path = Paths.get (args [0]);  
  18.       AsynchronousFileChannel ch = AsynchronousFileChannel.open (path);  
  19.       ByteBuffer buf = ByteBuffer.allocate (1024);  
  20.       Future<Integer> result = ch.read (buf, 0);  
  21.       while (!result.isDone ())  
  22.       {  
  23.          System.out.println ("Sleeping...");  
  24.          Thread.sleep (500);  
  25.       }  
  26.       System.out.println ("Finished = "+result.isDone ());  
  27.       System.out.println ("Bytes read = "+result.get ());  
  28.       ch.close ();  
  29.    }  

調用AsynchronousFileChannel's public static AsynchronousFileChannel open(Path file, OpenOption... options)方法打開file參數(shù)進行讀取,然后創(chuàng)建了一個字節(jié)緩沖區(qū)存儲讀取操作的結果。

接下來調用public abstract Future read(ByteBuffer dst, long position)方法異步讀取文件的前1024個字節(jié),這個方法返回一個Future實例代表這個操作的結果。

調用read()方法后,進入一個表決循環(huán),重復調用Future的isDone()方法檢查操作是否完成,一直等到讀操作結束,最后調用Future的get()方法返回讀取到的字節(jié)大小。

第二種形式需要你指定java.nio.channels.CompletionHandler,并實現(xiàn)下面的方法使用前面操作返回的結果,或是了解操作為什么失敗,并采取適當?shù)男袆樱?/p>

◆ 當操作完成時調用void completed(V result, A attachment),這個操作的結果是由result標識的,附加給操作的對象是由attachment標識的。

◆ 當操作失敗時調用void failed(Throwable exc, A attachment),操作失敗的原因是由exc標識的,附加給操作的對象是由attachment標識的。

#p#

我創(chuàng)建了一個程序演示創(chuàng)建和接收讀操作狀態(tài)的通知,其代碼如清單3所示。

清單3. AFCDemo2.java

  1. // AFCDemo2.java  
  2. import java.io.IOException;  
  3. import java.nio.ByteBuffer;  
  4. import java.nio.channels.AsynchronousFileChannel;  
  5. import java.nio.channels.CompletionHandler;  
  6. import java.nio.file.Path;  
  7. import java.nio.file.Paths;  
  8. public class AFCDemo2  
  9. {  
  10.    static Thread current;  
  11.    public static void main (String [] args) throws Exception  
  12.    {  
  13.       if (args.length != 1)  
  14.       {  
  15.           System.err.println ("usage: java AFCDemo1 path");  
  16.           return;  
  17.       }  
  18.       Path path = Paths.get (args [0]);  
  19.       AsynchronousFileChannel ch = AsynchronousFileChannel.open (path);  
  20.       ByteBuffer buf = ByteBuffer.allocate (1024);  
  21.       current = Thread.currentThread ();  
  22.       ch.read (buf, 0, null,  
  23.                new CompletionHandler<Integer, Void> ()  
  24.                {  
  25.                    public void completed (Integer result, Void v)  
  26.                    {  
  27.                       System.out.println ("Bytes read = "+result);  
  28.                       current.interrupt ();  
  29.                    }  
  30.                    public void failed (Throwable exc, Void v)  
  31.                    {  
  32.                       System.out.println ("Failure: "+exc.toString ());  
  33.                       current.interrupt ();  
  34.                    }  
  35.                });  
  36.       System.out.println ("Waiting for completion");  
  37.       try  
  38.       {  
  39.           current.join ();  
  40.       }  
  41.       catch (InterruptedException e)  
  42.       {  
  43.       }  
  44.       System.out.println ("Terminating");  
  45.       ch.close ();  
  46.    }  

上面的代碼調用AsynchronousFileChannel's public abstract void read(ByteBuffer dst, long position, A attachment, CompletionHandler handler)方法異步讀取前1024字節(jié)。

雖然我們只演示了單一的讀操作,但attachment部分也很重要,上面的代碼演示了傳遞一個null給read()方法,并指定附加類型為Void。

完整的套接字通道功能

JSR 51的DatagramChannel,ServerSocketChannel和SocketChannel類沒有完整抽象一個網絡套接字,為了綁定通道的套接字,或為了獲得/設置套接字選項,你必須先調用每個類的socket()方法檢索對等套接字。

JSR 51生效時沒有時間定義完整的套接字通道API,因此形成了套接字通道和套接字API混合的局面,JSR203引入新的java.nio.channels.NetworkChannel接口解決了這個問題。

NetworkChannel提供了將套接字綁定到本地地址,返回綁定地址,以及獲得/設置套接字選項的方法,這個接口是通過同步和異步套接字類實現(xiàn)的,不再需要調用socket()方法。

JSR 203也引入了新的java.nio.channels.MulticastChannel接口,它為DatagramChannel提供了IP多播的支持,以及對應的異步支持。

總結

本系列文章介紹了即將發(fā)布的JDK 7包含的一些新特性,新的里程碑版本可能很快就會發(fā)布,你現(xiàn)在就可以嘗試一下這些新特性,也許Oracle/Sun將會增加更多的新特性,如JWebPane瀏覽器組件,因為之前Sun就曾用閉包讓我們驚訝過一次了。

關于Java 7的更多內容,歡迎訪問51CTO推薦專題:Java 7 下一代Java開發(fā)技術詳解

【JDK 7相關內容推薦】

  1. 探秘JDK 7之三:JLayer裝飾Swing組件
  2. 探秘JDK 7之二:半透明和任意形狀的窗口
  3. 探秘JDK 7:將會出現(xiàn)新的語言特性
  4. Google技術演講介紹Java 7 NIO.2概覽
責任編輯:佚名 來源: IT168
相關推薦

2013-07-27 21:28:44

2013-06-27 11:21:17

2012-06-15 09:21:03

Windows 7Windows XP

2020-09-16 10:28:54

邊緣計算云計算數(shù)據(jù)中心

2025-01-03 09:24:10

模型架構論文

2013-07-25 21:08:37

2020-09-27 17:27:58

邊緣計算云計算技術

2025-05-26 00:00:00

GoogleAIVeo 3

2013-09-09 16:28:36

2016-01-26 11:58:12

2022-05-12 13:15:11

谷歌AI模型

2012-07-16 09:27:19

BYOD下一代IT

2012-07-16 10:08:31

下一代ITBYOD

2023-04-28 10:02:50

2015-10-15 10:30:32

2018-09-11 08:00:00

DevOpsAIOps機器學習

2013-05-10 09:36:04

下一代網絡銳捷網絡簡網絡

2021-05-22 23:01:21

人工智能網絡安全

2024-02-26 14:46:53

移動計算人工智能5G

2012-12-10 16:15:43

下一代防火墻NGWF
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 九九热热九九 | 国产精品色| av网站免费观看 | 在线观看免费av网站 | 国产乱码久久久久久一区二区 | 日本福利在线观看 | 国产农村一级国产农村 | 婷婷色成人 | 全免一级毛片 | 欧美精品一区二区三区在线播放 | 男女国产网站 | 秋霞电影院午夜伦 | 久久久久久亚洲精品不卡 | 精品伊人久久 | 日韩免费高清视频 | 国产精品一区二区免费 | 精品欧美一区二区三区精品久久 | 精品日本中文字幕 | 久久久一区二区三区四区 | 日韩一级黄色片 | 91中文字幕在线 | 99热精品在线 | 日韩一区在线播放 | 久久久久国产视频 | 一区二区三区高清 | 亚洲视频免费观看 | 亚洲综合色视频在线观看 | 国产精品视频久久久 | 一区二区福利视频 | 亚洲国产欧美精品 | 日韩一级精品视频在线观看 | 99久久精品一区二区毛片吞精 | 精品视频一区二区三区 | 亚洲国产精品久久久久秋霞不卡 | 在线观看国产视频 | 精品无码三级在线观看视频 | 日韩精品一区二区三区老鸭窝 | 国产91中文 | 91资源在线 | 亚洲婷婷六月天 | 国产欧美一区二区三区免费 |