Java中獲取文件大小的正確方法

本文出處:http://blog.csdn.net/chaijunkun/article/details/22387305,轉載請註明。由於本人不定期會整理相關博文,會對相應內容作出完善。因此強烈建議在原始出處查看此文。


今天寫代碼時需要實現獲取文件大小的功能,目前有兩種實現方法,一種是使用File的length()方法;另外一種是使用FileInputStream的available()方法,當InputStream未進行read操作時,available()的大小應該是等於文件大小的。但是在處理大文件時,後者會發生問題。我們來看一下:


在例子中,我使用了CentOS 6.5 的安裝鏡像文件,主要是考慮到這個文件足夠大(大於2GB)。


1.使用File的length()方法


[java] view plaincopy在CODE上查看代碼片派生到我的代碼片

  1. public static void main(String[] args) {  

  2.     File f= new File("D:\\CentOS-6.5-x86_64-bin-DVD1.iso");  

  3.     if (f.exists() && f.isFile()){  

  4.         logger.info(f.length());  

  5.     }else{  

  6.         logger.info("file doesn't exist or is not a file");  

  7.     }  

  8. }  

我們看一下輸出結果:


[plain] view plaincopy在CODE上查看代碼片派生到我的代碼片

  1. 4467982336  



結果是4.16GB,與Windows上顯示的結果一致。


接下來我們看一下通過FileInputStream來獲取的文件大小:


[java] view plaincopy在CODE上查看代碼片派生到我的代碼片

  1. public static void main(String[] args) {  

  2.     FileInputStream fis= null;  

  3.     try{  

  4.         File f= new File("D:\\CentOS-6.5-x86_64-bin-DVD1.iso");  

  5.         fis= new FileInputStream(f);  

  6.         logger.info(fis.available());  

  7.     }catch(Exception e){  

  8.         logger.error(e);  

  9.     } finally{  

  10.         if (null!=fis){  

  11.             try {  

  12.                 fis.close();  

  13.             } catch (IOException e) {  

  14.                 logger.error(e);  

  15.             }  

  16.         }  

  17.     }  

  18. }  

下面是運行結果:



[plain] view plaincopy在CODE上查看代碼片派生到我的代碼片

  1. 2147483647  

這個結果是不是很眼熟?它是Integer.MAX_VALUE,也就是有符號整型能表示的最大數值。


那麼換算成熟悉的單位,這種方式獲取的文件大小是多大呢?

約等於2GB,這顯然不是正確的結果。


究其原因,File的length()方法返回的類型爲long,long型能表示的正數最大值爲:9223372036854775807,折算成最大能支持的文件大小爲:8954730132868714 EB字節,這個量級將在人類IT發展史上受用很多很多年,而FileInputStream的avaliable()方法返回值是int,在之前也介紹了最大的表示範圍,所能支持的最大文件大小爲:1.99GB,而這個量級我們現在很容易就達到了。


2014年3月31日補充:

針對流式方法讀取大文件大小也不是不可行,只是不能再使用傳統的java.io.*下的包了,這裏要用到java.nio.*下的新工具——FileChannel。下面我們來看下示例代碼:


[java] view plaincopy在CODE上查看代碼片派生到我的代碼片

  1. public static void main(String[] args) {  

  2.     FileChannel fc= null;  

  3.     try {  

  4.         File f= new File("D:\\CentOS-6.5-x86_64-bin-DVD1.iso");  

  5.         if (f.exists() && f.isFile()){  

  6.             FileInputStream fis= new FileInputStream(f);  

  7.             fc= fis.getChannel();  

  8.             logger.info(fc.size());  

  9.         }else{  

  10.             logger.info("file doesn't exist or is not a file");  

  11.         }  

  12.     } catch (FileNotFoundException e) {  

  13.         logger.error(e);  

  14.     } catch (IOException e) {  

  15.         logger.error(e);  

  16.     } finally {  

  17.         if (null!=fc)){  

  18.             try{  

  19.                 fc.close();  

  20.             }catch(IOException e){  

  21.                 logger.error(e);  

  22.             }  

  23.         }   

  24.     }  

  25. }  


使用FileChannel後得到的結果與第一種情況吻合,準確地描述了文件的準確大小。



這裏也同樣提醒各位技術同仁,涉及到大文件讀取的時候,對int類型的數據一定要留個心,以免出現隱藏的bug,定位起來很困難。


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章