掘金 后端 ( ) • 2024-04-23 11:07

废话不多说系列,直接上~

女22.webp


核心功能:

  1. 锁定单元格,不能被任意修改;
  2. 合并单元格;
  3. 单元格设置指定字体颜色;
  4. 设置列宽度(自适应);

(1)锁定单元格

核心代码 - 锁定excel工作表

// 创建Excel文件
HSSFWorkbook workbook = new HSSFWorkbook();
HSSFSheet sheet = workbook.createSheet(DateUtils.getDate("yyyyMMdd"));
//锁定sheet
sheet.protectSheet("pwd");

核心代码 - 指定单元格可以修改

 CellStyle unlockCell = workbook.createCellStyle();
 unlockCell.setLocked(false);

 // 设置dataRow这一行的第i个单元格不锁定
 dataRow.getCell(i).setCellStyle(unlockCell);

(2)设置列宽

方法1:自适应列宽度
sheet.autoSizeColumn(1); // 此方法在合并单元格的里面不好使,必须用下面的方法。
sheet.autoSizeColumn(1, true);

// 这两个方法对英文数字还好,对中文支持的并不好。
sheet.trackAllColumnsForAutoSizing();
sheet.autoSizeColumn(i);
方法2:用数组将大概的宽度设置好,手动 set 宽度
int[] width = {xxx,xxx};
for循环
sheet.setColumnWidth(i,width[i]);
方法3:自己根据一列数据中的最长的字符串长度设置宽度
// 自己还得手动判断这一列的最长字符串长度。
int length = str.getBytes().length;
sheet.setColumnWidth((short)列数,(short)(length*256));

关键判断列字符串长度技巧:

  1. 用一个Map<Integer, List>, key是指具体哪一列, List中放的是每行的这一列的内容的长度 , 每遍历一行的一列, 就map.put(i, list.add(length)), 然后用Collections.max(map.get(i))来获取第i列的最长的长度;
  2. 还是一样,用一个map: Map<Integer, Integer>,key是指具体哪一列,value是每行的这一列的内容的长度, map.put(i,Math.max(length,map.get(i))),来确保map中的key对应的value永远是目前的最大的长度.

这里使用第二种:

// 设置自动换行后,不要设置固定的行高,否则超出的部分也会被遮住不显示。

// 创建Excel文件
HSSFWorkbook workbook = new HSSFWorkbook();
HSSFSheet sheet = workbook.createSheet("sheet");
//设置样式
 CellStyle blackStyle = workbook.createCellStyle();
//自动换行*重要*
 blackStyle.setWrapText(true);

//存储最大列宽
Map<Integer,Integer> maxWidth = new HashMap<>();
// 标题行
HSSFRow titleRow = sheet.createRow(0);
titleRow.setHeightInPoints(20);//目的是想把行高设置成20px
titleRow.createCell(0).setCellValue("sku编号");
titleRow.createCell(1).setCellValue("商品标题");
titleRow.createCell(2).setCellValue("商品名");
// 初始化标题的列宽,字体
for (int i= 0; i<=3;i++){
    maxWidth.put(i,titleRow.getCell(i).getStringCellValue().getBytes().length  * 256 + 200);
    titleRow.getCell(i).setCellStyle(blackStyle);//设置自动换行
}

for (Map<String, Object> map : list) {
    int currentRowNum = sheet.getLastRowNum() + 1;
    //数据行
    HSSFRow dataRow = sheet.createRow(currentRowNum);
    // 记录这一行的每列的长度
    List<Object> valueList = new ArrayList<Object>();

    String val0 = map.get("skuId") == null ? "—" : ((Double) (map.get("skuId"))).intValue()+"";
    valueList.add(val0);
    dataRow.createCell(0).setCellValue(val0);
    String val1 = map.get("title") == null ? "" : map.get("title").toString();
    valueList.add(val1);
    dataRow.createCell(1).setCellValue(val1);
    String val2 = map.get("goodsName") == null ? "" : map.get("goodsName").toString();
    valueList.add(val2);
    dataRow.createCell(2).setCellValue(val2);
    String val3 = map.get("catName") == null ? "" : map.get("catName").toString();
    valueList.add(val3);
    dataRow.createCell(3).setCellValue(val3);
    String val4 = map.get("brandName") == null ? "" : map.get("brandName").toString();

     for(int i = 0;i<=3;i++){
         int length = valueList.get(i).toString().getBytes().length  * 256 + 200;
         //这里把宽度最大限制到15000
         if (length>15000){
             length = 15000;
         }
         maxWidth.put(i,Math.max(length,maxWidth.get(i)));
          dataRow.getCell(i).setCellStyle(blackStyle);//设置自动换行
    }
}


for (int i= 0; i<=3;i++){
      //设置列宽
     sheet.setColumnWidth(i,maxWidth.get(i));
 }

(3)设置行高

// 设置行高,很简单
titleRow.setHeightInPoints(20); //目的是想把行高设置成20px

🦊注意:设置了固定行高,自动换行就不会自适应行高了。

(4)设置字体、颜色

创建 CellStyle,然后拆该年间 HSSFFont,再把 HSSFFont 注入给 Cell Style,再把 Cell Style 给 cell 设置。

// 设置字体
CellStyle redStyle = workbook.createCellStyle();
HSSFFont redFont = workbook.createFont();
//颜色
redFont.setColor(Font.COLOR_RED);
//设置字体大小
redFont.setFontHeightInPoints((short) 10);
//字体
//redFont.setFontName("宋体");
redStyle.setFont(redFont);

HSSFCell cell13 = titleRow.createCell(13);
cell13.setCellStyle(redStyle);
cell13.setCellValue("注意:只允许修改销售价,供应价,市场价和库存");

(5)合并单元格

合并单元格:建议先合并,再从 第一行第一列设值。

//这里代表在第0行开始,到0行结束,从0列开始,到10列结束,进行合并,也就是合并第0行的0-10个单元格
CellRangeAddress cellRange1 = new CellRangeAddress(0, 0, (short) 0, (short) 10);

sheet.addMergedRegion(cellRange1);
 CellRangeAddress

(6)水平居中-垂直居中

//设置水平对齐的样式为居中对齐;
cellStyle.setAlignment(HSSFCellStyle.ALIGN_CENTER);
//设置垂直对齐的样式为居中对齐;
cellStyle.setVerticalAlignment(HSSFCellStyle.VERTICAL_CENTER);

(7)excel 追加数据

实例方法函数,如下:

    /**
     * 给工作铺的第一个工作表追加合计数据
     *
     * @param wb         工作铺
     * @param sheetIndex Excel文件中工作表索引号(第一个工作表,sheetIndex=0)
     * @param sumData    合计行数据
     */
public static void addSumData(XSSFWorkbook wb, int sheetIndex, String[] sumData) {
    //获取到工作表,因为一个excel可能有多个工作表
    XSSFSheet sheet = wb.getSheetAt(sheetIndex);
    XSSFRow row = sheet.getRow(0);
    int hang = 0;
    if (row != null) {
        hang = sheet.getLastRowNum();// 获取下一行的行号
        hang = hang + 1;
    }
    // 核心:在现有行号后追加数据
    row = sheet.createRow((short) (hang));
    for (int i = 0; i < sumData.length; i++) {
        XSSFCell cell = row.createCell(i);
        cell.setCellValue(sumData[i]);
        // 注意:只有此处设置激活单元格,才能调整样式成功!
        cell.setAsActiveCell();
        cell.setCellStyle(xssfCellStyleOfColumnSum(wb));
    }
}

    /**
     * 合计样式
     */
    private static CellStyle xssfCellStyleOfColumnSum(XSSFWorkbook wb) {
        XSSFCellStyle cs = wb.createCellStyle();
        XSSFFont font = wb.createFont();
        font.setColor(Font.COLOR_RED);
        cs.setFont(font);
        //设置水平对齐的样式为居中对齐;
        cs.setAlignment(HSSFCellStyle.ALIGN_CENTER);
        //设置垂直对齐的样式为居中对齐;
        cs.setVerticalAlignment(HSSFCellStyle.VERTICAL_CENTER);
        return cs;
    }

/** 测试追加函数功能 */
private static void st() throws IOException {
    File excel = new File("D:\\文件名字.xlsx");
    FileInputStream fs;
    fs = new FileInputStream(excel);
    XSSFWorkbook wb = new XSSFWorkbook(fs);
    // 核心:追加
    addSumData(wb, 0, new String[]{"one", "two", "three"});
    FileOutputStream out = new FileOutputStream(excel);
    wb.write(out);
}

未添加数据前:

image.png

追加数据后:

image.png

附录

(1)常用包
  • HSSF - 提供读写Microsoft Excel XLS格式档案的功能。
  • XSSF - 提供读写Microsoft Excel OOXML XLSX格式档案的功能。
  • HWPF - 提供读写Microsoft Word DOC格式档案的功能。
  • HSLF - 提供读写Microsoft PowerPoint格式档案的功能。
  • HDGF - 提供读Microsoft Visio格式档案的功能。
  • HPBF - 提供读Microsoft Publisher格式档案的功能。
  • HSMF - 提供读Microsoft Outlook格式档案的功能。

🙏至此,感谢阅读🙏

女22.webp