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

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

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

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

◆ 原始類型數據緩沖

◆ 字符集編碼和解碼

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

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

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

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

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

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

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

新的文件系統接口

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

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

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

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

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

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

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

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

NIO.2引入了新的文件系統接口,除了解決上述存在的問題外,還引入了更多的功能,這個接口由位于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(統一資源定位符)返回給這個實例構造一個Path實例。

與傳統的基于File的代碼互操作:

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

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

為了不將問題復雜化,我就不深入講解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()當的檢測到操作失敗時,它會根據情況拋出適當的異常,如:

◆ 如果文件不存在,拋出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和選擇就緒的結合)使創建高可擴展服務器變得更加容易,本質上是這樣的,客戶端代碼用一個選擇器注冊一個套接字通道,當通道準備好可以開始I/O操作時發出通知。

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

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

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

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

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

◆ AsynchronousDatagramChannel – 標識一個面向數據報套接字異步通道,這個類實現了AsynchronousByteChannel。

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

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

◆ AsynchronousSocketChannel – 標識一個面向流連接套接字的異步通道,這個類實現了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參數進行讀取,然后創建了一個字節緩沖區存儲讀取操作的結果。

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

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

第二種形式需要你指定java.nio.channels.CompletionHandler,并實現下面的方法使用前面操作返回的結果,或是了解操作為什么失敗,并采取適當的行動:

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

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

#p#

我創建了一個程序演示創建和接收讀操作狀態的通知,其代碼如清單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字節。

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

完整的套接字通道功能

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

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

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

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

總結

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

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

【JDK 7相關內容推薦】

  1. 探秘JDK 7之三:JLayer裝飾Swing組件
  2. 探秘JDK 7之二:半透明和任意形狀的窗口
  3. 探秘JDK 7:將會出現新的語言特性
  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

2013-07-25 21:08:37

2020-09-27 17:27:58

邊緣計算云計算技術

2025-05-26 00:00:00

GoogleAIVeo 3

2020-09-16 10:28:54

邊緣計算云計算數據中心

2025-01-03 09:24:10

模型架構論文

2022-05-12 13:15:11

谷歌AI模型

2016-01-26 11:58:12

2013-09-09 16:28:36

2015-09-28 16:24:34

YARNHadoop計算

2018-09-25 07:00:50

2018-09-27 18:47:45

AIOpsDevOps

2013-07-27 21:41:14

APT攻擊下一代威脅

2009-04-06 08:42:18

Firefox瀏覽器

2022-07-06 11:38:40

人工智能AI

2011-06-30 11:02:22

2012-10-29 12:23:44

BYODIT

2009-01-11 10:13:39

Stripes開發框架JSP
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美区日韩区 | 久久久久久久久淑女av国产精品 | 免费中文字幕日韩欧美 | 国产精品久久久久无码av | 亚洲综合视频 | 伊大人久久 | av黄色免费在线观看 | 国产精品亚洲视频 | 免费观看的av | 中文字幕av在线一二三区 | 成人福利在线视频 | 国产精品一区2区 | 亚洲精品乱码久久久久久按摩观 | 国产黄色av网站 | 成人免费在线电影 | 日韩欧美一区二区三区免费看 | 亚洲精品免费视频 | 91精品久久久久久久久久小网站 | 亚洲精品久久久久中文字幕二区 | 97精品超碰一区二区三区 | 国产日韩精品视频 | 91中文在线观看 | 国产精品久久久久久久久久久久久 | 久久精品亚洲 | 日本精品久久久久 | 九九亚洲精品 | 亚洲色图第一页 | 国产一级在线 | 黄色一级片视频 | 蜜桃免费av | 黄视频免费观看 | av中文天堂| 中文日韩字幕 | 激情小说综合网 | 国产精品亚洲视频 | 亚洲成网站| 91精品国产综合久久婷婷香蕉 | 日韩高清成人 | 欧美寡妇偷汉性猛交 | 国产精品69av | 日韩在线一区二区 |