?????? 之前在面試中被問(wèn)到過(guò)兩次Java中文件讀入輸出怎么寫(xiě)史飞,當(dāng)時(shí)只記得一個(gè)大概,沒(méi)有辦法很清晰的說(shuō)出一個(gè)條理,今天特地看出總結(jié)了一下這方面的內(nèi)容哺徊,想要寫(xiě)出來(lái)給大家分享。
?????? 首先文件讀入輸出流常用有三種:FileInputStream/FileOutputStream乾闰,F(xiàn)ileReader/FileWriter落追,RandomAccessFile。下面具體列出一些簡(jiǎn)單的例子參考:
基礎(chǔ)篇:1.
FileRead? fr = new? FileReader(filename);
String s;
while( (s=fr.readLine())!=null){??
? ? ...? }?
fr.close();
//FileWriter同理涯肩,輸出時(shí)可用write()函數(shù)
//Java I/O中所有的Reader轿钠、Writer都是面向字符流的輸出輸出
2、
FileInputStream? fi =? new?? FileInputStream(filename);
int? in;
while( (in=fi.read())!=-1){??
...}
fi.close();
//FileOutputStream同理
//Java I/O中所有的Reader病苗、Writer都是面向字節(jié)流的輸出輸出
3疗垛、
RandomAccessFile ra =new? RandomAccessFile(filename,"rw");//后面的參數(shù)指定的是打開(kāi)文件流的方式,“rw”是指讀寫(xiě)硫朦,“r”是只讀贷腕,Java不提供只寫(xiě)
ra.seek(number);//將文件指針移動(dòng)到number處,這里文件指針可以理解為文件開(kāi)始讀的位置ra.skipByte(number);//跳過(guò)number個(gè)字節(jié)
ra.read();
ra.close();
//RandomAccessFile既可以讀也可以寫(xiě),而且可以利用seek()函數(shù)指定位置
下面是百度百科的一些介紹:
RandomAccessFile是不屬于InputStream和OutputStream類(lèi)系的泽裳。實(shí)際上芽世,除了實(shí)現(xiàn)DataInput和DataOutput接口之外(DataInputStream和DataOutputStream也實(shí)現(xiàn)了這兩個(gè)接口),它和這兩個(gè)類(lèi)系毫不相干诡壁,甚至都沒(méi)有用InputStream和OutputStream已經(jīng)準(zhǔn)備好的功能济瓢;它是一個(gè)完全獨(dú)立的類(lèi),所有方法(絕大多數(shù)都只屬于它自己)都是從零開(kāi)始寫(xiě)的妹卿。這可能是因?yàn)镽andomAccessFile能在文件里面前后移動(dòng)旺矾,所以它的行為與其它的I/O類(lèi)有些根本性的不同《峥耍總而言之箕宙,它是一個(gè)直接繼承Object的,獨(dú)立的類(lèi)铺纽。
進(jìn)階篇:
在nio中柬帕,Java重新實(shí)現(xiàn)了I/O流,并且引入了一些新的方法來(lái)提高速度狡门。我主要介紹通道陷寝、內(nèi)存映射文件
1.通道:
通道和緩沖器是一個(gè)成對(duì)的概念,Thinking in Java中的一個(gè)例子特別好理解:我們把想要讀入的文件看作一個(gè)煤礦其馏,數(shù)據(jù)就是我們想要的煤炭凤跑。通道好比是傳送煤礦的傳送帶,我們沒(méi)有辦法直接從傳送帶上拿走煤炭叛复,只好利用卡車(chē)來(lái)裝載這些煤炭仔引,卡車(chē)就是緩沖器,它主要負(fù)責(zé)從通道中取出數(shù)據(jù)褐奥,傳給我們寫(xiě)的程序咖耘。唯一能與通道交互的緩沖器是ByteBuffer∏寺耄可以看出和通道支持的解析流的方式是字節(jié)流儿倒。所以它配套使用的是FileInputStream/FileOutputStream,RandomAccessFile
例子:
a耍群、
FileChannel fc =new FileOutputStream(filename).getChannel();
fc.write(ByteBuffer.wrap("something test".getBytes() ));//這里使用ByteBuffer比較簡(jiǎn)單义桂,其實(shí)ByteBuffer可以利用個(gè)put()函數(shù)寫(xiě)入byte數(shù)組
fc.close();
b找筝、
fc= new FileOutputStream(filename).getChannel();
ByteBuffer buff = ByteBuffer.allocate(size);//沒(méi)錯(cuò)蹈垢,ByteBuffer是不提供顯示構(gòu)造函數(shù)的,想要新建一個(gè)對(duì)象必須利用allocate()函數(shù)來(lái)分配空間袖裕。
fc.read(buff);
fc.close();
為什么想到要用通道來(lái)做I/O呢曹抬?主要考慮的是性能問(wèn)題,通道加緩沖器能夠讓程序一些讀寫(xiě)一定量的字符急鳄,而只使用InputStream/OutputStream谤民,Reader/Writer只能一次讀寫(xiě)一個(gè)字節(jié)/字符堰酿。而程序在進(jìn)行I/O時(shí)要交給操作系統(tǒng)去解決這部分功能(調(diào)用系統(tǒng)調(diào)用),減少交給操作系統(tǒng)的次數(shù)可以有效的消減I/O花費(fèi)的時(shí)間
2.內(nèi)存映射文件:
內(nèi)存映射文件主要的意思其實(shí)假定將文件都放入內(nèi)存中张足,把它當(dāng)作非常大的數(shù)組來(lái)訪問(wèn)触创,效率特別好。為什么比較好呢为牍?這要從Java虛擬機(jī)和操作系統(tǒng)開(kāi)始說(shuō)起le(其實(shí)我也不太懂哼绑,剛才看了一篇文章講的比較清晰,鏈接是http://www.360doc.com/content...)這篇文章主要介紹了Java I/O的原理以及內(nèi)存映射文件的原理碉咆。我嘗試概括一下:Java I/O主要的實(shí)現(xiàn)手段肯定是利用系統(tǒng)調(diào)用抖韩,而系統(tǒng)調(diào)用先將想要使用的文件從硬盤(pán)調(diào)入到內(nèi)核的I/O緩沖區(qū)中,這次會(huì)導(dǎo)入比Java程序想要的文件更多的內(nèi)容(拷入更多的內(nèi)容是因?yàn)槌绦虻木植啃栽硪咄軌虻玫礁玫男剩┟。缓笤購(gòu)膬?nèi)核的I/O緩沖區(qū)導(dǎo)入到Java進(jìn)程自己的私有內(nèi)存空間中。而內(nèi)存映射文件放棄了兩次拷貝的方法壳咕,直接將Java進(jìn)程的虛擬空間與文件對(duì)象構(gòu)成一個(gè)映射席揽,當(dāng)私有內(nèi)存空間中找不打想要的內(nèi)容時(shí)發(fā)生缺頁(yè)異常,然后利用更底層的系統(tǒng)調(diào)用解決這個(gè)問(wèn)題(其實(shí)在I/O的系統(tǒng)調(diào)用中也涉及到了缺頁(yè)異常處理)谓厘,好處就是減少了一次從內(nèi)核I/O緩沖區(qū)到進(jìn)程私有地址的開(kāi)銷(xiāo)驹尼。
例子:
FileChannel fc = new RandomAccessFile(filename,"rw").getChannel();
MappedByteBuffer mb = fc.map(FileChannel.MapMode.READ_WRITE,start,length);
mb.put((byte)'x');
mb.get();
fc.close();
? 僅以此篇文章記錄一下。庞呕。新翎。