亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

Java應用程序中加載位圖文件的逐步指南

2019-11-18 14:49:42
字體:
來源:轉載
供稿:網友


作者:Jeff West 和 John D. Mitchell

摘要
目前,標準的 getImage() 方法僅支持 GIF 和 JPEG 圖像。盡管存在用于讀取 PNG(可移植網絡圖形)格式的 java 例程,但我們還沒聽說過有用于讀取 Microsoft Windows 位圖圖像的閱讀程序。Jeff West 撰寫的這篇技巧提供了加載 Windows 位圖圖像的代碼。
Java 的當前發行版并不正式支持在 Java 應用程序中讀取 Microsoft Windows 位圖文件。但別擔心,我們有辦法解決這個問題!這篇技巧將說明如何完成這一任務 -- 我們首先說明讀取 Microsoft Windows 文件格式的基本步驟。
Windows DIB(設備獨立的位圖)文件格式比較簡單。與純位圖格式不同,DIB 格式保留著用于在內存中存儲圖像的明確信息。問題是圖像格式的變體如此之多(1 位、4 位、8 位和 16 位,以及其他格式)。本篇 Java 技巧中提供的解決方案只處理 8 位和 24 位兩種格式。這兩種格式代表了最常見的變體。

不管是哪種 Windows DIB 子類型,這種文件格式總是由 14 位文件頭和 40 位信息頭組成。這兩個標頭精確包含有關文件的存儲內容和存儲次序的信息。有關標頭中每一項的確切含義,請參考 Microsoft Software Development Kit (SDK)。文件其余部分的內容隨信息頭中數據的不同而不同。

我們看一下本文要處理的兩種子類型。24 位格式很簡單:RGB(紅-綠-藍)顏色值(3 個字節,并按 BGR 排序)緊接在信息頭之后。但是,每個掃描行都被補足到 4 個字節。按照說明文檔(請參閱 Microsoft SDK)的說法,這種“補足”是為了優化 Windows 位圖繪圖 API。同時,底部的掃描行是文件中的第一項內容 -- 因此相對普通的圖形坐標系統(其矢量方向的正向分別為向下和向右)而言,必須從后向前讀取圖像。

8 位子類型由于在信息頭和象素數據之間插入調色板信息而復雜化。因此,每個象素條目只是進入 24 位 RGB 顏色的調色板數組的一個 8 位索引。在象素信息中,每個掃描行同樣被補足到 4 個字節。

請注重,本文提供的位圖圖像加載方法不支持對壓縮位圖圖像進行解壓縮。實際上,這個例程甚至不尋求這種可能性!假如碰到壓縮 Windows DIB 文件,該例程肯定會產生異常。Windows SDK 中有對壓縮 Windows DIB 格式的說明。

至于性能,在運行 Microsoft Windows 95 的 486-DX2-66MHz 系統上,該例程讀取 24 位 640 x 480 的文件(大約 920 千字節)所需的時間不超過 10 秒。使用 BufferedInputStream 而不是 FileInputStream 可明顯提高性能。

以下例程讀取兩種文件格式中的任一種,并生成一個 Image 圖像。以下代碼并未包含全面的錯誤和異常處理,以避免使該例程更加復雜。您總可用 Windows Paint 程序對不支持的 Windows DIB 子類型進行轉換。

    /**
loadbitmap() 方法由 Windows C 代碼轉換而來。
只能讀取未壓縮的 24 位和 8 位圖像。已在
Windows 95 上用 Microsoft Paint 保存的圖像
對它進行了測試。假如圖像不是 24 位或 8 位圖像,
該程序拒絕進行任何嘗試。我猜測假如先用 1100,
然后用 0011 對字節執行掩碼操作,則也可將 4 位
圖像包括在內。我實際上對這些圖像不感愛好。
假如嘗試讀取壓縮圖像,該例程可能失敗,并產生
一個 IOException 異常。假如變量 ncomPRession
不為 0,則表示已經過壓縮。

參數:
     sdir 和 sfile 是 FileDialog 的
     getDirectory() 和 getFile() 方法的結果。

返回值:
     Image 對象,切記要檢查 (Image)null !!!!

    */
    public Image loadbitmap (String sdir, String sfile)
{
Image image;
System.out.println("loading:"+sdir+sfile);
try
     {
     FileInputStream fs=new FileInputStream(sdir+sfile);
     int bflen=14; // 14 字節 BITMAPFILEHEADER
     byte bf[]=new byte[bflen];
     fs.read(bf,0,bflen);
     int bilen=40; // 40 字節 BITMAPINFOHEADER
     byte bi[]=new byte[bilen];
     fs.read(bi,0,bilen);

     // 解釋數據。
     int nsize = (((int)bf[5]&0xff)<<24)
   (((int)bf[4]&0xff)<<16)
   (((int)bf[3]&0xff)<<8)
   (int)bf[2]&0xff;
     System.out.println("File type is :"+(char)bf[0]+(char)bf[1]);
     System.out.println("Size of file is :"+nsize);

     int nbisize = (((int)bi[3]&0xff)<<24)
   (((int)bi[2]&0xff)<<16)
   (((int)bi[1]&0xff)<<8)
   (int)bi[0]&0xff;
     System.out.println("Size of bitmapinfoheader is :"+nbisize);

     int nwidth = (((int)bi[7]&0xff)<<24)
   (((int)bi[6]&0xff)<<16)
   (((int)bi[5]&0xff)<<8)
   (int)bi[4]&0xff;
     System.out.println("Width is :"+nwidth);

     int nheight = (((int)bi[11]&0xff)<<24)
   (((int)bi[10]&0xff)<<16)
   (((int)bi[9]&0xff)<<8)
   (int)bi[8]&0xff;
     System.out.println("Height is :"+nheight);

     int nplanes = (((int)bi[13]&0xff)<<8) (int)bi[12]&0xff;
     System.out.println("Planes is :"+nplanes);

     int nbitcount = (((int)bi[15]&0xff)<<8) (int)bi[14]&0xff;
     System.out.println("BitCount is :"+nbitcount);

     // 查找表明壓縮的非零值
     int ncompression = (((int)bi[19])<<24)
   (((int)bi[18])<<16)
   (((int)bi[17])<<8)
   (int)bi[16];
     System.out.println("Compression is :"+ncompression);

     int nsizeimage = (((int)bi[23]&0xff)<<24)
   (((int)bi[22]&0xff)<<16)
   (((int)bi[21]&0xff)<<8)
   (int)bi[20]&0xff;
     System.out.println("SizeImage is :"+nsizeimage);

     int nXPm = (((int)bi[27]&0xff)<<24)
   (((int)bi[26]&0xff)<<16)
   (((int)bi[25]&0xff)<<8)
   (int)bi[24]&0xff;
     System.out.println("X-Pixels per meter is :"+nxpm);

     int nypm = (((int)bi[31]&0xff)<<24)
   (((int)bi[30]&0xff)<<16)
   (((int)bi[29]&0xff)<<8)
   (int)bi[28]&0xff;
     System.out.println("Y-Pixels per meter is :"+nypm);

     int nclrused = (((int)bi[35]&0xff)<<24)
   (((int)bi[34]&0xff)<<16)
   (((int)bi[33]&0xff)<<8)
   (int)bi[32]&0xff;
     System.out.println("Colors used are :"+nclrused);

     int nclrimp = (((int)bi[39]&0xff)<<24)
   (((int)bi[38]&0xff)<<16)
   (((int)bi[37]&0xff)<<8)
   (int)bi[36]&0xff;
     System.out.println("Colors important are :"+nclrimp);

     if (nbitcount==24)
  {
  // 24 位格式不包含調色板數據,但掃描行被補足到
  // 4 個字節。
  int npad = (nsizeimage / nheight) - nwidth * 3;
  int ndata[] = new int [nheight * nwidth];
  byte brgb[] = new byte [( nwidth + npad) * 3 * nheight];
  fs.read (brgb, 0, (nwidth + npad) * 3 * nheight);
  int nindex = 0;
  for (int j = 0; j < nheight; j++)
      {
      for (int i = 0; i < nwidth; i++)
   {
   ndata [nwidth * (nheight - j - 1) + i] =
       (255&0xff)<<24
        (((int)brgb[nindex+2]&0xff)<<16)
        (((int)brgb[nindex+1]&0xff)<<8)
        (int)brgb[nindex]&0xff;
   // System.out.println("Encoded Color at ("
       +i+","+j+")is:"+nrgb+" (R,G,B)= ("
       +((int)(brgb[2]) & 0xff)+","
       +((int)brgb[1]&0xff)+","
       +((int)brgb[0]&0xff)+")");
   nindex += 3;
   }
      nindex += npad;
      }

  image = createImage
      ( new MemoryImageSource (nwidth, nheight,
          ndata, 0, nwidth));
  }
     else if (nbitcount == 8)
  {
  // 必須確定顏色數。假如 clrsused 參數大于 0,
  // 則顏色數由它決定。假如它等于 0,則根據
  // bitsperpixel 計算顏色數。
  int nNumColors = 0;
  if (nclrused > 0)
      {
      nNumColors = nclrused;
      }
  else
      {
      nNumColors = (1&0xff)<<nbitcount;
      }
  System.out.println("The number of Colors is"+nNumColors);

  // 某些位圖不計算 sizeimage 域,請找出
  // 這些情況并對它們進行修正。
  if (nsizeimage == 0)
      {
      nsizeimage = ((((nwidth*nbitcount)+31) & ~31 ) >> 3);
      nsizeimage *= nheight;
      System.out.println("nsizeimage (backup) is"+nsizeimage);
      }

  // 讀取調色板顏色。
  int npalette[] = new int [nNumColors];
  byte bpalette[] = new byte [nNumColors*4];
  fs.read (bpalette, 0, nNumColors*4);
  int nindex8 = 0;
  for (int n = 0; n < nNumColors; n++)
      {
      npalette[n] = (255&0xff)<<24
    (((int)bpalette[nindex8+2]&0xff)<<16)
    (((int)bpalette[nindex8+1]&0xff)<<8)
    (int)bpalette[nindex8]&0xff;
      // System.out.println ("Palette Color "+n
   +" is:"+npalette[n]+" (res,R,G,B)= ("
   +((int)(bpalette[nindex8+3]) & 0xff)+","
   +((int)(bpalette[nindex8+2]) & 0xff)+","
   +((int)bpalette[nindex8+1]&0xff)+","
   +((int)bpalette[nindex8]&0xff)+")");
      nindex8 += 4;
      }

  // 讀取圖像數據(實際上是調色板的索引)
  // 掃描行仍被補足到 4 個字節。
  int npad8 = (nsizeimage / nheight) - nwidth;
  System.out.println("nPad is:"+npad8);

  int ndata8[] = new int [nwidth*nheight];
  byte bdata[] = new byte [(nwidth+npad8)*nheight];
  fs.read (bdata, 0, (nwidth+npad8)*nheight);
  nindex8 = 0;
  for (int j8 = 0; j8 < nheight; j8++)
      {
      for (int i8 = 0; i8 < nwidth; i8++)
   {
   ndata8 [nwidth*(nheight-j8-1)+i8] =
       npalette [((int)bdata[nindex8]&0xff)];
   nindex8++;
   }
      nindex8 += npad8;
      }

  image = createImage
      ( new MemoryImageSource (nwidth, nheight,
          ndata8, 0, nwidth));
  }
     else
  {
  System.out.println ("Not a 24-bit or 8-bit Windows Bitmap, aborting...");
  image = (Image)null;
  }

     fs.close();
     return image;
     }
catch (Exception e)
     {
     System.out.println("Caught exception in loadbitmap!");
     }
return (Image) null;
}


您已把握了讀取位圖文件的技巧。很輕易對此方法進行擴展,使它能夠讀取單色和 16 色(4 位)格式。

作者簡介
Jeff West 是加州圣地亞哥市的一名工程學研究生。在研究燃燒和火焰擴張的閑暇之余,他沉迷于 Java。



發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品高潮在线| 情事1991在线| 亚洲福利视频在线| 国产精品久久久久久一区二区| 亚洲精品美女久久| 日本精品免费观看| 亚洲美女视频网站| 亚洲成人av中文字幕| 在线观看国产欧美| 国产精品一区二区av影院萌芽| 国产999精品久久久影片官网| 亚洲国模精品私拍| 日本一区二区在线播放| 精品久久久免费| 亚洲自拍偷拍色图| 亚洲美女福利视频网站| 精品一区精品二区| 欧美色欧美亚洲高清在线视频| 国产不卡在线观看| 国产香蕉一区二区三区在线视频| 国产成人综合亚洲| 欧美日韩福利视频| 国产精品久久久久久久久久新婚| 亚洲精品电影在线观看| 日韩麻豆第一页| 精品久久久久久国产| 国产精品69久久| 九九热这里只有在线精品视| 国产在线精品成人一区二区三区| 日韩一区二区三区国产| 97激碰免费视频| 欧美成人免费在线观看| 伊人久久五月天| 92福利视频午夜1000合集在线观看| 欧美一级电影免费在线观看| 九九视频这里只有精品| 国产精品网站视频| 91精品国产乱码久久久久久蜜臀| 亚洲无限乱码一二三四麻| 欧美在线国产精品| 中文字幕v亚洲ⅴv天堂| 亚洲午夜精品久久久久久久久久久久| 69av成年福利视频| 久久综合伊人77777尤物| 国产亚洲美女久久| 久久99视频免费| 久久99精品久久久久久噜噜| 国产精品久久久久久av福利| 欧美日韩福利电影| 亚洲国产美女精品久久久久∴| 色播久久人人爽人人爽人人片视av| 国产玖玖精品视频| 777国产偷窥盗摄精品视频| 欧美高清在线视频观看不卡| 国产亚洲精品美女| 亚洲性xxxx| 一道本无吗dⅴd在线播放一区| 日韩免费中文字幕| 精品动漫一区二区三区| 欧美激情综合色综合啪啪五月| 欧美性猛交xxxxx免费看| 国内精品一区二区三区四区| 91网站免费观看| 国产亚洲精品久久久久久牛牛| 欧美日韩综合视频| 国产精品久久久久久久久久ktv| 亚洲欧洲中文天堂| 欧美精品情趣视频| 亚洲福利视频免费观看| 亚洲国产精品热久久| 欧美成人午夜剧场免费观看| 国产精品1区2区在线观看| 日韩欧美aⅴ综合网站发布| 中文字幕视频在线免费欧美日韩综合在线看| 91精品国产综合久久香蕉的用户体验| 国产精品午夜一区二区欲梦| 九九综合九九综合| 欧美亚洲午夜视频在线观看| 欧洲s码亚洲m码精品一区| 国产91久久婷婷一区二区| 91久久精品在线| 成人日韩在线电影| 色综合天天狠天天透天天伊人| 色偷偷88888欧美精品久久久| 成人免费午夜电影| 亚洲国产欧美一区| 国产精品入口福利| 欧美一级淫片丝袜脚交| 日韩av免费看| 欧美在线观看www| 欧美日韩一区二区免费在线观看| 欧美性猛交xxx| 成人免费在线视频网站| 日韩av手机在线观看| 国产成人综合久久| 韩国19禁主播vip福利视频| 日韩中文字幕网| 亚洲欧美国产日韩天堂区| 久久久视频在线| 久久久久成人精品| 亚洲欧美日韩一区二区三区在线| 国产手机视频精品| 国产精品一区二区三区免费视频| 一本色道久久88综合亚洲精品ⅰ| 亚洲欧洲国产伦综合| 国产美女精品视频免费观看| 国产亚洲精品一区二区| 日韩av综合网| 久久亚洲成人精品| 国产美女扒开尿口久久久| 国自产精品手机在线观看视频| 国产一区玩具在线观看| 在线观看国产成人av片| 亚洲精品v天堂中文字幕| 日韩在线不卡视频| 亚洲一区二区中文字幕| 国内精品中文字幕| 97在线精品国自产拍中文| 亚洲综合在线中文字幕| 北条麻妃久久精品| 69av在线视频| 亚洲福利在线视频| 亚洲综合国产精品| xxx一区二区| 精品久久久久久电影| 国产精品999| 久久久久久18| 91精品国产综合久久香蕉最新版| 色99之美女主播在线视频| 欧美日韩国产中文精品字幕自在自线| 欧美日本精品在线| 国产一区二区三区四区福利| 一区二区三区久久精品| 久久中国妇女中文字幕| 午夜精品久久久久久99热软件| 成人性生交大片免费看小说| 国内精品国产三级国产在线专| 亚洲一级免费视频| 97视频在线观看免费高清完整版在线观看| 欧美视频国产精品| 国产亚洲精品美女久久久| 日韩精品在线观| 亚洲最新av网址| 日本精品视频网站| 久热精品视频在线| 欧美丝袜一区二区| 日韩成人小视频| 国产91露脸中文字幕在线| 亚州成人av在线| 成人国产精品免费视频| 中文字幕精品一区二区精品| www.美女亚洲精品| 91久久久久久久一区二区| 91爱视频在线| 精品国产福利在线| 久久久免费电影| 在线观看国产成人av片| 欧美裸体xxxx| 中文字幕亚洲欧美在线| 亚洲欧美制服第一页| 亚洲精品国产精品久久清纯直播| 日韩成人高清在线| 91在线视频导航| 色综合伊人色综合网站|