欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java如何實現(xiàn)圖片的疊加與拼接操作

 更新時間:2019年11月02日 15:01:03   作者:loytime  
這篇文章主要介紹了Java如何實現(xiàn)圖片的疊加與拼接操作,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下

關(guān)于Java實現(xiàn)圖片的疊加與拼接的文章網(wǎng)絡(luò)上確實很多,碰巧小編開發(fā)工作中也遇到這些問題,就做了簡要的梳理,作為筆記以備不時之需。

Java對圖片的處理主要使用的是BufferedImage類。

BufferedImage 子類描述具有可訪問圖像數(shù)據(jù)緩沖區(qū)的 Image。BufferedImage 由圖像數(shù)據(jù)的 ColorModel 和 Raster 組成。Raster 的 SampleModel 中 band 的數(shù)量和類型必須與 ColorModel 所要求的數(shù)量和類型相匹配,以表示其顏色和 alpha 分量。所有 BufferedImage 對象的左上角坐標(biāo)都為 (0, 0)。因此,用來構(gòu)造 BufferedImage 的任何 Raster 都必須滿足:minX=0 且 minY=0。此類依靠 Raster 的數(shù)據(jù)獲取方法、數(shù)據(jù)設(shè)置方法,以及 ColorModel 的顏色特征化方法。

以上主要來源于官方文檔,我們來時直接寫實踐代碼吧。

首先將文件轉(zhuǎn)化為BufferedImage對象,這里給出兩種讀取文件并轉(zhuǎn)化為BufferedImage對象的方法。

/**
   * @param fileUrl 文件絕對路徑或相對路徑
   * @return 讀取到的緩存圖像
   * @throws IOException 路徑錯誤或者不存在該文件時拋出IO異常
   */
  public static BufferedImage getBufferedImage(String fileUrl)
      throws IOException {
    File f = new File(fileUrl);
    return ImageIO.read(f);
  }  
  
  /**
   * 遠(yuǎn)程圖片轉(zhuǎn)BufferedImage
   * @param destUrl  遠(yuǎn)程圖片地址
   * @return
   */
  public static BufferedImage getBufferedImageDestUrl(String destUrl) {
    HttpURLConnection conn = null;
    BufferedImage image = null;
    try {
      URL url = new URL(destUrl);
      conn = (HttpURLConnection) url.openConnection();
      if (conn.getResponseCode() == 200) {
        image = ImageIO.read(conn.getInputStream());
        return image;
      }
    } catch (Exception e) {
      e.printStackTrace();
    } finally {
      conn.disconnect();
    }
    return image;
  }

接下來是將BufferedImage對象保存到本地,具體方法如下:

/**
   * 輸出圖片
   * @param buffImg 圖像拼接疊加之后的BufferedImage對象
   * @param savePath 圖像拼接疊加之后的保存路徑
   */
  public static void generateSaveFile(BufferedImage buffImg, String savePath) {
    int temp = savePath.lastIndexOf(".") + 1;
    try {
      File outFile = new File(savePath);
      if(!outFile.exists()){
        outFile.createNewFile();
      }
      ImageIO.write(buffImg, savePath.substring(temp), outFile);
      System.out.println("ImageIO write...");
    } catch (IOException e) {
      e.printStackTrace();
    }
  }

以上作為準(zhǔn)備部分,現(xiàn)在開始圖片疊加的實現(xiàn)方法:

/**
   * 
   * @Title: 構(gòu)造圖片
   * @Description: 生成水印并返回java.awt.image.BufferedImage
   * @param buffImg 源文件(BufferedImage)
   * @param waterFile 水印文件(BufferedImage)
   * @param x 距離右下角的X偏移量
   * @param y 距離右下角的Y偏移量
   * @param alpha 透明度, 選擇值從0.0~1.0: 完全透明~完全不透明
   * @return BufferedImage
   * @throws IOException
   */
  public static BufferedImage overlyingImage(BufferedImage buffImg, BufferedImage waterImg, int x, int y, float alpha) throws IOException {

    // 創(chuàng)建Graphics2D對象,用在底圖對象上繪圖
    Graphics2D g2d = buffImg.createGraphics();
    int waterImgWidth = waterImg.getWidth();// 獲取層圖的寬度
    int waterImgHeight = waterImg.getHeight();// 獲取層圖的高度
    // 在圖形和圖像中實現(xiàn)混合和透明效果
    g2d.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_ATOP, alpha));
    // 繪制
    g2d.drawImage(waterImg, x, y, waterImgWidth, waterImgHeight, null);
    g2d.dispose();// 釋放圖形上下文使用的系統(tǒng)資源
    return buffImg;
  }

圖片拼接的實現(xiàn)方法:

/**
   * 待合并的兩張圖必須滿足這樣的前提,如果水平方向合并,則高度必須相等;如果是垂直方向合并,寬度必須相等。
   * mergeImage方法不做判斷,自己判斷。 
   * @param img1 待合并的第一張圖
   * @param img2 帶合并的第二張圖
   * @param isHorizontal 為true時表示水平方向合并,為false時表示垂直方向合并
   * @return 返回合并后的BufferedImage對象
   * @throws IOException
   */
  public static BufferedImage mergeImage(BufferedImage img1,
      BufferedImage img2, boolean isHorizontal) throws IOException {
    int w1 = img1.getWidth();
    int h1 = img1.getHeight();
    int w2 = img2.getWidth();
    int h2 = img2.getHeight();

    // 從圖片中讀取RGB
    int[] ImageArrayOne = new int[w1 * h1];
    ImageArrayOne = img1.getRGB(0, 0, w1, h1, ImageArrayOne, 0, w1); // 逐行掃描圖像中各個像素的RGB到數(shù)組中
    int[] ImageArrayTwo = new int[w2 * h2];
    ImageArrayTwo = img2.getRGB(0, 0, w2, h2, ImageArrayTwo, 0, w2);

    // 生成新圖片
    BufferedImage DestImage = null;
    if (isHorizontal) { // 水平方向合并
      DestImage = new BufferedImage(w1+w2, h1, BufferedImage.TYPE_INT_RGB);
      DestImage.setRGB(0, 0, w1, h1, ImageArrayOne, 0, w1); // 設(shè)置上半部分或左半部分的RGB
      DestImage.setRGB(w1, 0, w2, h2, ImageArrayTwo, 0, w2);
    } else { // 垂直方向合并
      DestImage = new BufferedImage(w1, h1 + h2, BufferedImage.TYPE_INT_RGB);
      DestImage.setRGB(0, 0, w1, h1, ImageArrayOne, 0, w1); // 設(shè)置上半部分或左半部分的RGB
      DestImage.setRGB(0, h1, w2, h2, ImageArrayTwo, 0, w2); // 設(shè)置下半部分的RGB
    }

    return DestImage;
  }

測試方法如下:

/**
   * Java 測試圖片疊加方法
   */
  public static void overlyingImageTest() {

    String sourceFilePath = "D://test//test1.jpg";
    String waterFilePath = "D://test//test2.jpg";
    String saveFilePath = "D://test//overlyingImageNew.jpg";
    try {
      BufferedImage bufferImage1 = getBufferedImage(sourceFilePath);
      BufferedImage bufferImage2 = getBufferedImage(waterFilePath);

      // 構(gòu)建疊加層
      BufferedImage buffImg = overlyingImage(bufferImage1, bufferImage2, 0, 0, 1.0f);
      // 輸出水印圖片
      generateSaveFile(buffImg, saveFilePath);
    } catch (IOException e) {
      e.printStackTrace();
    }

  }
  
  
  /**
   * Java 測試圖片合并方法
   */
  public static void imageMargeTest() {
    // 讀取待合并的文件
    BufferedImage bi1 = null;
    BufferedImage bi2 = null;
    // 調(diào)用mergeImage方法獲得合并后的圖像
    BufferedImage destImg = null;
    System.out.println("下面是垂直合并的情況:");
    String saveFilePath = "D://test//new1.jpg";
    String divingPath = "D://test//new2.jpg";
    String margeImagePath = "D://test//margeNew.jpg";
    try {
      bi1 = getBufferedImage(saveFilePath);
      bi2 = getBufferedImage(divingPath);
      // 調(diào)用mergeImage方法獲得合并后的圖像
      destImg = mergeImage(bi1, bi2, false);
    } catch (IOException e) {
      e.printStackTrace();
    }
    // 保存圖像
    generateSaveFile(destImg, margeImagePath);
    System.out.println("垂直合并完畢!");
  }

  public static void main(String[] args) {
    // 測試圖片的疊加
    overlyingImageTest();
    // 測試圖片的垂直合并
    imageMargeTest();
  }

整體代碼如下:

package ImagePackage;

import java.awt.AlphaComposite;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.net.HttpURLConnection;
import java.net.URL;

import javax.imageio.ImageIO;

/**

 * 該類實現(xiàn)了圖片的合并功能,可以選擇水平合并或者垂直合并。
 * 當(dāng)然此例只是針對兩個圖片的合并,如果想要實現(xiàn)多個圖片的合并,只需要自己實現(xiàn)方法 BufferedImage
 * mergeImage(BufferedImage[] imgs, boolean isHorizontal)即可;
 * 而且這個方法更加具有通用性,但是時間原因不實現(xiàn)了,方法和兩張圖片實現(xiàn)是一樣的
 */

public class ImageMerge {

  /**
   * @param fileUrl
   *      文件絕對路徑或相對路徑
   * @return 讀取到的緩存圖像
   * @throws IOException
   *       路徑錯誤或者不存在該文件時拋出IO異常
   */
  public static BufferedImage getBufferedImage(String fileUrl)
      throws IOException {
    File f = new File(fileUrl);
    return ImageIO.read(f);
  }
  
  
  /**
   * 遠(yuǎn)程圖片轉(zhuǎn)BufferedImage
   * @param destUrl  遠(yuǎn)程圖片地址
   * @return
   */
  public static BufferedImage getBufferedImageDestUrl(String destUrl) {
    HttpURLConnection conn = null;
    BufferedImage image = null;
    try {
      URL url = new URL(destUrl);
      conn = (HttpURLConnection) url.openConnection();
      if (conn.getResponseCode() == 200) {
        image = ImageIO.read(conn.getInputStream());
        return image;
      }
    } catch (Exception e) {
      e.printStackTrace();
    } finally {
      conn.disconnect();
    }
    return image;
  }
  
  /**
   * 輸出圖片
   * 
   * @param buffImg
   *      圖像拼接疊加之后的BufferedImage對象
   * @param savePath
   *      圖像拼接疊加之后的保存路徑
   */
  public static void generateSaveFile(BufferedImage buffImg, String savePath) {
    int temp = savePath.lastIndexOf(".") + 1;
    try {
      File outFile = new File(savePath);
      if(!outFile.exists()){
        outFile.createNewFile();
      }
      ImageIO.write(buffImg, savePath.substring(temp), outFile);
      System.out.println("ImageIO write...");
    } catch (IOException e) {
      e.printStackTrace();
    }
  }
  
  /**
   * 
   * @Title: 構(gòu)造圖片
   * @Description: 生成水印并返回java.awt.image.BufferedImage
   * @param buffImg
   *      源文件(BufferedImage)
   * @param waterFile
   *      水印文件(BufferedImage)
   * @param x
   *      距離右下角的X偏移量
   * @param y
   *      距離右下角的Y偏移量
   * @param alpha
   *      透明度, 選擇值從0.0~1.0: 完全透明~完全不透明
   * @return BufferedImage
   * @throws IOException
   */
  public static BufferedImage overlyingImage(BufferedImage buffImg, BufferedImage waterImg, int x, int y, float alpha) throws IOException {

    // 創(chuàng)建Graphics2D對象,用在底圖對象上繪圖
    Graphics2D g2d = buffImg.createGraphics();
    int waterImgWidth = waterImg.getWidth();// 獲取層圖的寬度
    int waterImgHeight = waterImg.getHeight();// 獲取層圖的高度
    // 在圖形和圖像中實現(xiàn)混合和透明效果
    g2d.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_ATOP, alpha));
    // 繪制
    g2d.drawImage(waterImg, x, y, waterImgWidth, waterImgHeight, null);
    g2d.dispose();// 釋放圖形上下文使用的系統(tǒng)資源
    return buffImg;
  }
  
  
  /**
   * 待合并的兩張圖必須滿足這樣的前提,如果水平方向合并,則高度必須相等;如果是垂直方向合并,寬度必須相等。
   * mergeImage方法不做判斷,自己判斷。
   *
   * @param img1
   *      待合并的第一張圖
   * @param img2
   *      帶合并的第二張圖
   * @param isHorizontal
   *      為true時表示水平方向合并,為false時表示垂直方向合并
   * @return 返回合并后的BufferedImage對象
   * @throws IOException
   */
  public static BufferedImage mergeImage(BufferedImage img1,
      BufferedImage img2, boolean isHorizontal) throws IOException {
    int w1 = img1.getWidth();
    int h1 = img1.getHeight();
    int w2 = img2.getWidth();
    int h2 = img2.getHeight();

    // 從圖片中讀取RGB
    int[] ImageArrayOne = new int[w1 * h1];
    ImageArrayOne = img1.getRGB(0, 0, w1, h1, ImageArrayOne, 0, w1); // 逐行掃描圖像中各個像素的RGB到數(shù)組中
    int[] ImageArrayTwo = new int[w2 * h2];
    ImageArrayTwo = img2.getRGB(0, 0, w2, h2, ImageArrayTwo, 0, w2);

    // 生成新圖片
    BufferedImage DestImage = null;
    if (isHorizontal) { // 水平方向合并
      DestImage = new BufferedImage(w1+w2, h1, BufferedImage.TYPE_INT_RGB);
      DestImage.setRGB(0, 0, w1, h1, ImageArrayOne, 0, w1); // 設(shè)置上半部分或左半部分的RGB
      DestImage.setRGB(w1, 0, w2, h2, ImageArrayTwo, 0, w2);
    } else { // 垂直方向合并
      DestImage = new BufferedImage(w1, h1 + h2, BufferedImage.TYPE_INT_RGB);
      DestImage.setRGB(0, 0, w1, h1, ImageArrayOne, 0, w1); // 設(shè)置上半部分或左半部分的RGB
      DestImage.setRGB(0, h1, w2, h2, ImageArrayTwo, 0, w2); // 設(shè)置下半部分的RGB
    }

    return DestImage;
  }

  /**
   * Java 測試圖片疊加方法
   */
  public static void overlyingImageTest() {

    String sourceFilePath = "D://test//test1.jpg";
    String waterFilePath = "D://test//test2.jpg";
    String saveFilePath = "D://test//overlyingImageNew.jpg";
    try {
      BufferedImage bufferImage1 = getBufferedImage(sourceFilePath);
      BufferedImage bufferImage2 = getBufferedImage(waterFilePath);

      // 構(gòu)建疊加層
      BufferedImage buffImg = overlyingImage(bufferImage1, bufferImage2, 0, 0, 1.0f);
      // 輸出水印圖片
      generateSaveFile(buffImg, saveFilePath);
    } catch (IOException e) {
      e.printStackTrace();
    }

  }
  
  
  /**
   * Java 測試圖片合并方法
   */
  public static void imageMargeTest() {
    // 讀取待合并的文件
    BufferedImage bi1 = null;
    BufferedImage bi2 = null;
    // 調(diào)用mergeImage方法獲得合并后的圖像
    BufferedImage destImg = null;
    System.out.println("下面是垂直合并的情況:");
    String saveFilePath = "D://test//new1.jpg";
    String divingPath = "D://test//new2.jpg";
    String margeImagePath = "D://test//margeNew.jpg";
    try {
      bi1 = getBufferedImage(saveFilePath);
      bi2 = getBufferedImage(divingPath);
      // 調(diào)用mergeImage方法獲得合并后的圖像
      destImg = mergeImage(bi1, bi2, false);
    } catch (IOException e) {
      e.printStackTrace();
    }
    // 保存圖像
    generateSaveFile(destImg, margeImagePath);
    System.out.println("垂直合并完畢!");
  }

  public static void main(String[] args) {
    // 測試圖片的疊加
    overlyingImageTest();
    // 測試圖片的垂直合并
    imageMargeTest();
  }

}

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • java括號匹配問題介紹

    java括號匹配問題介紹

    大家好,本篇文章主要講的是java括號匹配問題介紹,感興趣的同學(xué)趕快來看一看吧,對你有幫助的話記得收藏一下,方便下次瀏覽
    2021-12-12
  • Java通過Callable實現(xiàn)多線程

    Java通過Callable實現(xiàn)多線程

    這篇文章主要介紹了Java通過Callable實現(xiàn)多線程,Callable的任務(wù)執(zhí)行后可返回值,運行Callable任務(wù)可以拿到一個Future對象,Future表示異步計算的結(jié)果,它提供了檢查計算是否完成的方法,以等待計算的完成,并檢查計算的結(jié)果,需要的朋友可以參考下
    2023-10-10
  • Mybatis-plus依賴及配置文件方式

    Mybatis-plus依賴及配置文件方式

    這篇文章主要介紹了Mybatis-plus依賴及配置文件方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-07-07
  • Maven包沖突導(dǎo)致NoSuchMethodError錯誤的解決辦法

    Maven包沖突導(dǎo)致NoSuchMethodError錯誤的解決辦法

    web 項目 能正常編譯,運行時也正常啟動,但執(zhí)行到需要調(diào)用 org.codehaus.jackson 包中的某個方法時,產(chǎn)生運行異常,這篇文章主要介紹了Maven包沖突導(dǎo)致NoSuchMethodError錯誤的解決辦法,需要的朋友可以參考下
    2024-05-05
  • Spring Boot右鍵maven build成功但是直接運行main方法出錯的解決方案

    Spring Boot右鍵maven build成功但是直接運行main方法出錯的解決方案

    這篇文章主要介紹了Spring Boot-右鍵maven build成功但是直接運行main方法出錯的解決方案,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-08-08
  • Java中的List接口實現(xiàn)類解析

    Java中的List接口實現(xiàn)類解析

    這篇文章主要介紹了Java中的List接口實現(xiàn)類解析,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-02-02
  • Java interrupt()方法使用注意_動力節(jié)點Java學(xué)院整理

    Java interrupt()方法使用注意_動力節(jié)點Java學(xué)院整理

    這篇文章主要介紹了Java interrupt()方法使用注意_動力節(jié)點Java學(xué)院整理,需要的朋友可以參考下
    2017-05-05
  • java實現(xiàn)數(shù)據(jù)結(jié)構(gòu)單鏈表示例(java單鏈表)

    java實現(xiàn)數(shù)據(jù)結(jié)構(gòu)單鏈表示例(java單鏈表)

    這篇文章主要介紹了java數(shù)據(jù)結(jié)構(gòu)實現(xiàn)單鏈表示例,需要的朋友可以參考下
    2014-03-03
  • spring,mybatis事務(wù)管理配置與@Transactional注解使用詳解

    spring,mybatis事務(wù)管理配置與@Transactional注解使用詳解

    這篇文章主要介紹了spring,mybatis事務(wù)管理配置與@Transactional注解使用,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-07-07
  • Java springboot項目jar發(fā)布過程解析

    Java springboot項目jar發(fā)布過程解析

    這篇文章主要介紹了Java springboot項目jar發(fā)布過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-09-09

最新評論