EasyExcel 公式計算大全

EasyExcel 是基于 Apache POI 的封裝,主要專注于簡化 Excel 的讀寫操作,對于公式計算的支持相對有限。以下是 EasyExcel 中處理公式計算的全面指南:


1. 基本公式寫入

1.1 寫入簡單公式

@Data
public class FormulaData {@ExcelProperty("數值1")private Double value1;@ExcelProperty("數值2")private Double value2;@ExcelProperty("求和")private String sumFormula;@ExcelProperty("平均值")private String avgFormula;
}// 寫入公式
List<FormulaData> list = new ArrayList<>();
FormulaData data = new FormulaData();
data.setValue1(10.0);
data.setValue2(20.0);
data.setSumFormula("SUM(A2,B2)");  // 引用當前行的單元格
data.setAvgFormula("AVERAGE(A2,B2)");
list.add(data);EasyExcel.write("formula_example.xlsx", FormulaData.class).sheet("公式示例").doWrite(list);

1.2 使用絕對引用

data.setSumFormula("SUM($A$2:$B$2)");  // 絕對引用

2. 讀取包含公式的Excel

2.1 基本讀取

// 監聽器實現
public class FormulaDataListener extends AnalysisEventListener<FormulaData> {@Overridepublic void invoke(FormulaData data, AnalysisContext context) {// 這里獲取的是公式計算后的值System.out.println("求和結果: " + data.getSumFormula());}@Overridepublic void doAfterAllAnalysed(ActionContext context) {}
}// 讀取文件
EasyExcel.read("formula_example.xlsx", FormulaData.class, new FormulaDataListener()).sheet().doRead();

2.2 獲取公式本身(而非計算結果)

public class FormulaReadListener extends AnalysisEventListener<Map<Integer, String>> {@Overridepublic void invoke(Map<Integer, String> data, AnalysisContext context) {// 獲取底層CellReadCellData<?> cellData = (ReadCellData<?>) data.get(2); // 第三列if (cellData.getFormulaData() != null) {System.out.println("公式內容: " + cellData.getFormulaData());}}
}EasyExcel.read("formula_example.xlsx", new FormulaReadListener()).sheet().doRead();

3. 動態公式設置

3.1 使用攔截器動態設置公式

public class FormulaWriteHandler implements CellWriteHandler {@Overridepublic void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {if (!isHead && head.getFieldName().equals("sumFormula")) {// 設置求和公式,如SUM(A2,B2)int rowNum = cell.getRowIndex() + 1; // Excel行號從1開始String formula = String.format("SUM(A%d,B%d)", rowNum, rowNum);cell.setCellFormula(formula);}}
}// 使用
EasyExcel.write("dynamic_formula.xlsx", FormulaData.class).registerWriteHandler(new FormulaWriteHandler()).sheet().doWrite(dataList);

3.2 復雜公式示例

// IF函數示例
String ifFormula = "IF(A2>B2, \"A大于B\", IF(A2<B2, \"A小于B\", \"A等于B\"))";// VLOOKUP函數示例
String vlookupFormula = "VLOOKUP(D2,$A$2:$B$10,2,FALSE)";// 日期函數
String dateFormula = "TEXT(TODAY(),\"yyyy-mm-dd\")";

4. 公式計算控制

4.1 強制重新計算公式

// 寫入后強制計算
ExcelWriter excelWriter = EasyExcel.write("output.xlsx", FormulaData.class).build();
WriteSheet writeSheet = EasyExcel.writerSheet("Sheet1").build();
excelWriter.write(dataList, writeSheet);// 獲取底層Workbook并強制重新計算
SXSSFWorkbook workbook = (SXSSFWorkbook) excelWriter.writeContext().writeWorkbookHolder().getWorkbook();
workbook.getCreationHelper().createFormulaEvaluator().evaluateAll();excelWriter.finish();

4.2 設置不計算公式(僅保留公式)

// 在讀取時不計算公式
EasyExcel.read("input.xlsx", new FormulaReadListener()).ignoreEmptyRow(false).autoTrim(true).formulaResult(false)  // 不計算公式結果.sheet().doRead();

5. 常見函數示例

5.1 數學函數

// 基本數學運算
String addFormula = "A2+B2";
String multiplyFormula = "PRODUCT(A2,B2)";
String roundFormula = "ROUND(A2,2)";// 條件求和
String sumifFormula = "SUMIF(A2:A10,\">10\",B2:B10)";

5.2 文本函數

// 連接文本
String concatFormula = "CONCATENATE(A2,\"-\",B2)";// 提取子串
String leftFormula = "LEFT(A2,3)";
String midFormula = "MID(A2,2,3)";

5.3 日期函數

// 計算日期差
String dateDiffFormula = "DATEDIF(A2,B2,\"d\")";// 添加天數
String addDaysFormula = "A2+7";  // A2是日期單元格

6. 高級技巧

6.1 跨工作表公式

// 引用其他工作表的單元格
String crossSheetFormula = "SUM(Sheet2!A1:A10)";// 引用其他工作簿(需要文件在同一目錄)
String externalRefFormula = "'[OtherWorkbook.xlsx]Sheet1'!A1";

6.2 數組公式

// 數組公式需要用大括號包圍
String arrayFormula = "{SUM(A2:A10*B2:B10)}";// 寫入時需要特殊處理
cell.setCellFormula("SUM(A2:A10*B2:B10)");
// 然后需要將單元格標記為數組公式區域
CellRangeAddress range = new CellRangeAddress(cell.getRowIndex(), cell.getRowIndex(),cell.getColumnIndex(), cell.getColumnIndex());
sheet.addMergedRegion(range);
sheet.setArrayFormula(cell.getCellFormula(), range);

6.3 命名范圍公式

// 創建命名范圍
Name namedRange = workbook.createName();
namedRange.setNameName("SalesData");
namedRange.setRefersToFormula("Sheet1!$A$2:$A$10");// 使用命名范圍
String namedRangeFormula = "SUM(SalesData)";
// 1. 定義數據類
@Data
public class FinancialData {@ExcelProperty("月份")private String month;@ExcelProperty("收入")private Double income;@ExcelProperty("支出")private Double expense;@ExcelProperty("利潤")private String profitFormula;@ExcelProperty("利潤率")private String marginFormula;
}// 2. 公式寫入處理器
public class FinancialFormulaHandler implements CellWriteHandler {@Overridepublic void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {if (!isHead) {int rowNum = cell.getRowIndex() + 1;String column = head.getFieldName();if ("profitFormula".equals(column)) {cell.setCellFormula(String.format("B%d-C%d", rowNum, rowNum));} else if ("marginFormula".equals(column)) {cell.setCellFormula(String.format("(B%d-C%d)/B%d", rowNum, rowNum, rowNum));// 設置百分比格式CellStyle style = writeSheetHolder.getSheet().getWorkbook().createCellStyle();style.setDataFormat((short)0xa);  // 百分比格式cell.setCellStyle(style);}}}
}// 3. 使用示例
public void exportFinancialReport() {List<FinancialData> dataList = new ArrayList<>();// 添加數據...ExcelWriter excelWriter = EasyExcel.write("financial_report.xlsx", FinancialData.class).registerWriteHandler(new FinancialFormulaHandler()).build();WriteSheet writeSheet = EasyExcel.writerSheet("財務報告").build();excelWriter.write(dataList, writeSheet);// 強制計算公式SXSSFWorkbook workbook = (SXSSFWorkbook) excelWriter.writeContext().writeWorkbookHolder().getWorkbook();workbook.getCreationHelper().createFormulaEvaluator().evaluateAll();excelWriter.finish();
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/917227.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/917227.shtml
英文地址,請注明出處:http://en.pswp.cn/news/917227.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

2025年AI+數模競賽培訓意見征集-最后一輪

在過去幾天的“AI時代下2025年數模競賽培訓課程需求調研緊急征集”我們收到了大量老師、學生的反饋。我們通過大家的實際需求&#xff0c;編寫了下述2025年AI時代下最新的數學建模競賽教學課程課程表&#xff0c;具體授課內容以及相關課件、支撐材料都將會免費發布&#xff0c;…

Qwen2 RotaryEmbedding 位置編碼僅僅是第一層有嗎

Qwen2 RotaryEmbedding 位置編碼僅僅是第一層有嗎,還是全部層都有 Qwen2 模型中的 Rotary Embedding(旋轉位置編碼)是應用于所有 Transformer 層 的,而非僅第一層。 1. Transformer 架構的核心邏輯 Qwen2 基于 Decoder-only Transformer 架構,而位置編碼(如 Rotary Emb…

CNN卷積神經網絡之LeNet和AlexNet經典網絡模型(三)

CNN卷積神經網絡之LeNet和AlexNet經典網絡模型&#xff08;三&#xff09; 文章目錄CNN卷積神經網絡之LeNet和AlexNet經典網絡模型&#xff08;三&#xff09;深度學習兩大經典 CNN 模型速覽1. LeNet-5&#xff1a;CNN 的開山之作&#xff08;1998&#xff09;2. AlexNet&#…

江協科技STM32 12-2 BKP備份寄存器RTC實時時鐘

這一節我們要講的主要內容是RTC實時時鐘&#xff0c;實時時鐘本質上是一個定時器&#xff0c;但是這個定時器是專門用來產生年月日時分秒&#xff0c;這種日期和時間信息的。所以學會了STM32的RTC就可以在STM32內部擁有一個獨立運行的鐘表。想要記錄或讀取日期和時間&#xff0…

【10】大恒相機SDK C++開發 ——對相機采集的原圖像數據IFrameData裁剪ROI 實時顯示在pictureBox中,3種方法實現(效率不同)

文章目錄1 在回調函數中實現2 獨立封裝調用2.1 獲取圖像寬、高、pBuffer、channel2.2 內存圖像數據截取ROI并顯示2.3 回調函數調用3 for循環嵌套 方法24 for循環嵌套 方法35 按行復制數據提高效率&#xff0c;但很耗內存6 unsafe代碼 解釋及注意事項 看我另一篇文章7 ConvertTo…

ubuntu22.04系統入門 linux入門(二) 簡單命令 多實踐以及相關文件管理命令

以下有免費的4090云主機提供ubuntu22.04系統的其他入門實踐操作 地址&#xff1a;星宇科技 | GPU服務器 高性能云主機 云服務器-登錄 相關兌換碼星宇社區---4090算力卡免費體驗、共享開發社區-CSDN博客 之所以推薦給大家使用&#xff0c;是因為上面的云主機目前是免費使用的…

分布式ID方案(標記)

一、參考文章-標記 分布式ID方案有哪些&#xff1f;雪花算法如何搞定時鐘回撥和動態機器ID&#xff1f; 二、應用 1.百度 uid-generator github項目地址 原理參考 2.百度 uid-generator 擴展應用 燈官網 燈 項目代碼 lamp-util 單元模塊 lamp-util 單元模塊子模塊 lamp-…

std::map 加鎖

在并發環境下使用std::map&#xff0c;必須采取同步措施。 在并發環境下對 std::map 進行不加鎖的讀寫操作會導致嚴重的線程安全問題&#xff0c;主要會產生以下幾種問題&#xff1a; ?? 主要風險與后果數據競爭&#xff08;Data Race&#xff09; 當多個線程同時修改同一個鍵…

學習筆記090——Ubuntu 中 UFW 防火墻的使用

文章目錄1、允許特定的端口訪問2、允許特定 IP 訪問某個端口3、允許某個范圍的端口4、查看 UFW 狀態5、重新加載 UFW6、啟用 UFW7、關閉 UFW1、允許特定的端口訪問 # 允許 TCP 端口&#xff08;例如 80&#xff09;&#xff1a; sudo ufw allow 80/tcp# 允許 UDP 端口&#xf…

移動端 WebView 內存泄漏與性能退化問題如何排查 實戰調試方法匯總

在混合 App 應用中&#xff0c;WebView 頁面常承載復雜業務邏輯與交互。隨著用戶使用時間增長&#xff0c;特別在切換多個頁面或反復打開界面后&#xff0c;常常會出現性能下降、頁面卡頓、甚至白屏崩潰等現象。這通常是因為頁面存在內存泄漏、事件監聽未解綁或垃圾回收阻塞導致…

JSON 對象在瀏覽器中順序與后端接口返回不一致的問題

一、問題描述 后端接口返回一個字典表的JSON對象&#xff0c;頁面展示排序與預期排序不一致。 在瀏覽器調試面板Response中看到接口原始響應字符串&#xff0c;是期望順序&#xff1a;在Preview中看到&#xff0c; key “22” 被提到最前&#xff0c;順序發生變化&#xff1a;頁…

Spring MVC數據傳遞全攻略

Spring MVC數據傳遞一、前端到后端的數據傳遞1. 使用 RequestParam 傳遞簡單參數2. 使用 PathVariable傳遞路徑參數3. 使用RequestBody傳遞 JSON 數據二、后端到前端的數據傳遞1. 使用Model或 ModelAndView傳遞數據到前端2. 使用HttpServletResponse直接寫回數據3.使用Response…

倉庫管理系統-12-前端之頭部區域Header基于嵌套路由訪問個人中心

文章目錄 1 個人中心 1.1 DateUtils.vue(子組件) 1.2 Home.vue(父組件) 1.3 router/index.js(嵌套路由) 1.4 index.vue(路由占位符) 2 Header.vue 2.1 頁面布局 2.2 toUser方法 2.3 初始加載 2.4 Header.vue 頭部區域Header中有一個個人中心下拉菜單,點擊個人中心選項,通過嵌…

【智能協同云圖庫】第七期:基于AI調用阿里云百煉大模型,實現AI圖片編輯功能

摘要&#xff1a;AI 高速發展賦能傳統業務&#xff0c;圖庫網站亦有諸多 AI 應用空間。以 AI 擴圖功?能為例&#xff0c;讓我們來學習如何在項目?中快速接入 AI 繪圖大模型。?用戶可以選擇一張已上傳的圖片&#xff0c;?通過 AI 擴圖得到新的圖片&#xff0c;希望可以幫到大…

Notepad++插件安裝

方式一&#xff1a;自動安裝&#xff08;有些notepad并不好用&#xff0c;推薦方式二&#xff09;工具欄-》插件-》插件管理如下點擊安裝后會提示&#xff0c;后端安裝&#xff0c;安裝成功后自動啟動&#xff0c;本人使用的v8.6.4的版本&#xff0c;插件基本都無法自動安裝&am…

git pull和git fetch的區別

git pull和git fetch是git版本控制系統中的兩個基本命令&#xff0c;它們都用于從遠程倉庫更新本地倉庫的信息&#xff0c;但執行的具體操作不同。git fetch:git fetch下載遠程倉庫最新的內容到你的本地倉庫&#xff0c;但它并不自動合并或修改你當前的工作。它取回了遠程倉庫的…

Item35:考慮virtual函數以外的其他選擇

在C++中,虛函數是實現多態的傳統方式,但并非唯一選擇。過度依賴虛函數可能導致派生類與基類的強耦合,或難以在運行時靈活切換行為。《Effective C++》Item35指出:應根據場景選擇更合適的替代方案,包括NVI模式、函數指針、策略模式等。本文解析這些方案的原理、適用場景及實…

Vue3 狀態管理新選擇:Pinia 從入門到實戰

一、什么是pinia? 在 Vue3 生態中&#xff0c;狀態管理一直是開發者關注的核心話題。隨著 Vuex 的逐步淡出&#xff0c;Pinia 作為官方推薦的狀態管理庫&#xff0c;憑借其簡潔的 API、強大的功能和對 Vue3 特性的完美適配&#xff0c;成為了新時代的不二之選。今天我們就來深…

Unity相機控制

相機的控制無非移動和旋轉&#xff0c;每種操作各3個軸6個方向&#xff0c;一共12種方式。在某些需要快速驗證的項目或Demo里常常需要絲滑的控制相機調試效果。相機控制雖然不是什么高深的技術&#xff0c;但是要寫的好用還是很磨人的。 鎖定Z軸的旋轉 一個自由的相機可以繞 …

vue2 使用liveplayer加載視頻

vue2 使用liveplayer加載視頻 官網: https://www.liveqing.com/docs/manuals/LivePlayer.html支持WebRTC/MP4播放;支持m3u8/HLS播放;支持HTTP-FLV/WS-FLV/RTMP播放;支持直播和點播播放;支持播放器快照截圖;支持點播多清晰度播放;支持全屏或比例顯示;自動檢測IE瀏覽器兼容播放;支…