??? 在任何一個(gè)綜合性網(wǎng)站,我們往往需要上傳一些圖片資料。但隨著高分辨率DC的普及,上傳的圖片容量會很大,比如300萬象素DC出來的文件基本不下600K。為了管理方便,大家可能不愿意每次都用ACDsee修改它,而直接上傳到服務(wù)器。但是這種做法在客戶端看來就沒有那么輕松了,對于撥號上網(wǎng)的用戶簡直是一場惡夢,雖然你可以在圖片區(qū)域設(shè)置wide和high!
???
??? 問題的解決之道來了!我們可以在類中處理一張大圖,并縮小它。
??? 前提是需要JDK1.4,這樣才能進(jìn)行處理。按以下方法做:
???
????? import java.io.File;
????? import java.io.FileOutputStream;
????? import java.awt.Graphics;
????? import java.awt.Image;
????? import java.awt.image.BufferedImage;
????? import com.sun.image.codec.jpeg.JPEGCodec;
????? import com.sun.image.codec.jpeg.JPEGImageEncoder;
?????
????? public class JpgTest {
?
?public void JpgTset() throws Exception{
???? File _file = new File("/Order005-0001.jpg");?????????????????????? //讀入文件
???? Image src = javax.imageio.ImageIO.read(_file);???????????????????? //構(gòu)造Image對象
???? int wideth=src.getWidth(null);???????????????????????????????????? //得到源圖寬
???? int height=src.getHeight(null);??????????????????????????????????? //得到源圖長
???? BufferedImage tag = new BufferedImage(wideth/2,height/2,BufferedImage.TYPE_INT_RGB);
???? tag.getGraphics().drawImage(src,0,0,wideth/2,height/2,null);?????? //繪制縮小后的圖
???? FileOutputStream out=new FileOutputStream("newfile.jpg");????????? //輸出到文件流
???? JPEGImageEncoder encoder = JPEGCodec.createJPEGEncoder(out);??????
???? encoder.encode(tag);?????????????????????????????????????????????? //近JPEG編碼
???? //System.out.print(width+"*"+height);?????????????????????????????
???? out.close();
?}
????? }
???
??? 過程很簡單,從本地磁盤讀取文件Order005-0001.jpg(2032*1524),變成Image對象src,接著構(gòu)造目標(biāo)文件tag,設(shè)置tag的長寬為源圖的一半,對tag進(jìn)行編碼,輸出到文件流out,最后關(guān)閉文件流。
???
??? 還有一些問題需要說明:
??? 第一,目前只能支持JPG(JPEG)、GIF、PNG三種格式。(這里有些問題,在最下面解釋)
??? 第二,對于源圖的容量有限制,最好不要超過1M,否則會拋內(nèi)存不足的錯(cuò)誤,不過我試驗(yàn)過1.8M的源圖,可以成功,但是也很容易拋內(nèi)存不足。
???
??? 引用一位前輩的話:圖象運(yùn)算本身是密集型運(yùn)算,需要大量的內(nèi)存存放象素值。我用VC試了一下,4M的圖象也有問題,而且越是壓縮比大的圖片在內(nèi)存中還原成BITMAP時(shí)需要的內(nèi)存越大。解決的方法,可以重寫編碼類,先開一定的內(nèi)存,然后一段一段編碼寫到臨時(shí)文件中,輸出的時(shí)候再一段一段讀出來。或利用nio的內(nèi)存映象來操作。JavaMail由于采用了Builder模式,先生成一個(gè)郵件的每一個(gè)部分,然后合并成一個(gè)完整的郵件對象,這樣每個(gè)構(gòu)件都要先生成到內(nèi)存中,你如果發(fā)送一個(gè)上百兆的附件,那么在構(gòu)造Part時(shí)肯定內(nèi)存溢出,所以我就改寫了BodyPart的構(gòu)造,讓他和一個(gè)臨時(shí)文件關(guān)聯(lián),然后用臨時(shí)文件保存Part而不是構(gòu)造在內(nèi)存中,這樣任義大小的附件(硬盤能放得下為限)都可以發(fā)送了。
???
??? 最后,如果大家對圖像處理有更高的要求,不妨關(guān)注一下開源項(xiàng)目。比如JMagick,可以使用JMagick來實(shí)現(xiàn)圖片的復(fù)制、信息獲取、斜角、特效、組合、改變大小、加邊框、旋轉(zhuǎn)、切片、改變格式、去色等等功能。
2007-06-20更新
其實(shí)按照上面的做法只能壓縮jpg格式,gif是不能壓縮的(由于算法版權(quán)問題,直到Java 6.0才能壓縮gif),前段時(shí)間要用gif壓縮,在網(wǎng)上找了個(gè)類,可以解決問題。請參考:http://mindprod.com/jgloss/gifencoder.html。
替換以前代碼的相應(yīng)部分:
????????????????????size[ 0 ],?size[ 1 ],?BufferedImage.TYPE_INT_RGB);
????????????tag.getGraphics().drawImage(src,? 0 ,? 0 ,?size[ 0 ],?size[ 1 ],? null );?
????????????out? = ? new ?FileOutputStream(path);
????????????GIFEncoder?gifEncoder? = ? new ?GIFEncoder(tag);
????????????gifEncoder.write(out);
當(dāng)代碼運(yùn)行完畢,用EditPlus之類的文本編輯器打開之后可以發(fā)現(xiàn)文件頭已經(jīng)是gif87格式了。
請注意!引用、轉(zhuǎn)貼本文應(yīng)注明原作者:Rosen Jiang 以及出處:http://www.aygfsteel.com/rosen