Java Web導(dǎo)出等比例圖片到Excel的實現(xiàn)過程
做個Excel導(dǎo)入導(dǎo)出的Java開發(fā)戶都知道,可以利用Apache的POI來實現(xiàn),為Excel創(chuàng)建每一行,每一單元及對應(yīng)的內(nèi)容。當(dāng)然,圖片的話需要用到POI中的HSSFClientAnchor這個類型來實現(xiàn), HSSFClientAnchor的構(gòu)造函數(shù)中有八個參數(shù),分類為:int dx1,int dy1,int dx2,int dy2,short col1,int row1,short col2, int row2,
其中dx1、dy1定義了該圖片在開始cell的起始位置,dx2、dy2定義了在終cell的結(jié)束位置,col1、row1定義了開始cell、col2、row2定義了結(jié)束cell。
那么,我們就可以用這個方法來講圖片定位到每一個單元格中。我的做法是:
鋪滿整個單元格,這里的j和i是代碼中要到處多張圖做了循環(huán)定義的變量HSSFClientAnchor anchor = new HSSFClientAnchor(0, 0, 1023, 255, (short) j, i + 1, (short) j, i + 1);
很順利,成功導(dǎo)出!
問題來了,打開成功到處的Excel一看,商品對應(yīng)的圖片都很規(guī)矩的按照我的設(shè)置鋪滿了整個單元格。但是,商品圖片卻都變形了,也就是圖片的 寬高被強行設(shè)置了成對應(yīng)單元格的寬高。
這樣肯定是不行的,于是第一反應(yīng)就是將圖片等比例導(dǎo)出,但是怎么讓圖片等比例的導(dǎo)出在Excel中呢,苦惱了許久,終于想到了一個可行的方案。就是利用POI設(shè)置每一行對應(yīng)要放圖片的那個單元格的寬度固定,也就是所有導(dǎo)出的圖片的寬度固定,然后根據(jù)這個固定寬度與原圖寬高對比,算出該固定寬度的等比例高度,然后動態(tài)設(shè)置該單元格的高度,然后再利用HSSFClientAnchor來鋪滿整個單元格,這樣看起來就不會變形,而且等比例縮放到指定單元格中。
關(guān)鍵兩個實現(xiàn)步驟:
第一步 設(shè)置單元格固定寬度: 循環(huán)創(chuàng)建每一行的時候,如何設(shè)置要放圖片的那個單元的寬度呢?大家都知道,在Excel中,同一列的單元格的寬度都是一樣的,故可以在創(chuàng)建工作簿的時候,就設(shè)定好要放圖片的那個單元格的寬度,我是這樣做的:
生成一個表格 HSSFSheet sheet = workbook.createSheet(title);
設(shè)置B列的寬度為 30*256; sheet.setColumnWidth(1, 30 * 256);
這里我們看到,SetColumnWidth的第二個參數(shù)要乘以256,是因為這個參數(shù)的單位是1/256個字符寬度,也就是說這里我設(shè)置的是30個字符的寬度,至于一個字符寬度是多少,可以用FontDesignMetrics來獲得,一般一個12號字體的寬度大約是13像素。
第二步,設(shè)置單元等比率高度:在循環(huán)每一行中,當(dāng)創(chuàng)建對應(yīng)放圖片的單元的時候,設(shè)置對應(yīng)的高度。我的做法是,先獲取要導(dǎo)出的圖片,在獲取圖片的原始寬度和高度(主要,這里獲取的是像素寬高),然后根據(jù)前面設(shè)置的固定寬度算出等比例的高度:
ByteArrayOutputStream byteArrayOut = new ByteArrayOutputStream(); BufferedImage bufferImg = ImageIO.read(new File(savePath + System.getProperty("file.separator") + map.get("productImg"))); ImageIO.write(bufferImg, "jpg", byteArrayOut); int width = bufferImg.getWidth();//原始寬度 int height = bufferImg.getHeight();//原始高度 // 一個12號字體的寬度為13,前面已設(shè)置了列的寬度為30*256,故這里的等比例高度計算如下 height = (int) Math.round((height * (30 * 13) * 1.0 / width)); // excel單元格高度是以點單位,1點=2像素; POI中Height的單位是1/20個點,故設(shè)置單元的等比例高度如下 row.setHeight((short) (height / 2 * 20)); // 畫圖的頂級管理器,一個sheet只能獲取一個(一定要注意這點) HSSFPatriarch patriarch = sheet.createDrawingPatriarch(); // anchor主要用于設(shè)置圖片的屬性 HSSFClientAnchor anchor = new HSSFClientAnchor(0, 0, 1023, 255, (short) j, i + 1, (short) j, i + 1); anchor.setAnchorType(3); // 插入圖片 patriarch.createPicture(anchor,workbook.addPicture(byteArrayOut.toByteArray(), HSSFWorkbook.PICTURE_TYPE_JPEG));
至此,等比率圖片及相關(guān)信息以完美導(dǎo)出到Excel中。
到此這篇關(guān)于Java Web導(dǎo)出等比例圖片到Excel的實現(xiàn)過程的文章就介紹到這了,更多相關(guān)Java Web導(dǎo)出等比例圖片內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Mybatis自定義SQL的關(guān)系映射、分頁、排序功能的實現(xiàn)
這篇文章主要介紹了Mybatis自定義SQL的關(guān)系映射、分頁、排序功能的實現(xiàn),本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-01-01JAVA如何判斷上傳文件后綴名是否符合規(guī)范MultipartFile
這篇文章主要介紹了JAVA判斷上傳文件后綴名是否符合規(guī)范MultipartFile,文中通過實例代碼介紹了java實現(xiàn)對上傳文件做安全性檢查,需要的朋友可以參考下2023-11-11java基本教程之常用的實現(xiàn)多線程的兩種方式 java多線程教程
下面開始學(xué)習(xí)“常用的實現(xiàn)多線程的2種方式”:Thread 和 Runnable。之所以說是常用的,是因為通過還可以通過java.util.concurrent包中的線程池來實現(xiàn)多線程2014-01-01SpringBoot下獲取resources目錄下文件的常用方法
本文詳細(xì)介紹了SpringBoot獲取resources目錄下文件的常用方法,包括使用this.getClass()方法、ClassPathResource獲取以及hutool工具類ResourceUtil獲取,感興趣的可以了解一下2024-10-10JAVA 16位ID生成工具類含16位不重復(fù)的隨機(jī)數(shù)數(shù)字+大小寫
這篇文章主要介紹了JAVA 16位ID生成工具類含16位不重復(fù)的隨機(jī)數(shù)數(shù)字+大小寫,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-02-02Spring復(fù)雜對象創(chuàng)建的方式小結(jié)
這篇文章主要介紹了Spring復(fù)雜對象創(chuàng)建的三種方式,現(xiàn)在使用Spring如何創(chuàng)建這種類型的對象?Spring中提供了三種方法來創(chuàng)建復(fù)雜對象,需要的朋友可以參考下2022-01-01