註解是 Java 的一個高級特性,Spring 更是以註解爲基礎,發展出一套「註解驅動編程」。git
這聽起來高大上,但畢竟是框架的事,咱們也能用好註解嗎?編程
的確,咱們不多有機會本身寫註解,致使咱們搞不清楚註解是怎麼回事,更別提用好註解了。框架
既然這樣,咱們就從具體的工做出發,開發一個 Excel 導出功能。我相信,你在搞懂這個例子後,就能明白註解是怎麼個用法。函數
在後臺管理系統中,經常須要把數據導出 Excel 表。this
好比,在雙十一事後,銷售部要把商品訂單錄入到 Excel 表,財務部要把支付訂單錄入到 Excel 表,而後各部門彙總分析,最後找個時間討論怎麼改善公司的服務。spa
你想呀,雙十一的訂單成千上萬,靠人工錄入,少說也要花三四天,並且還特別容易出錯。因此,你必須開發 Excel 導出功能。設計
那麼,具體怎麼作呢?excel
上次咱們提到,註解想發揮做用,有三個要素:定義、使用、讀取。此次,咱們就利用註解的三個特性,來實現 Excel 導出功能,設計過程是這樣的。code
第一步,咱們要建立不一樣的 Excel 模型。雙十一事後,銷售部要訂單數據,財務部要支付數據,兩個部門要的 Excel 表確定也不同,這就得幫每一個部門建立不一樣的 Excel 模型,他們拿到想要的數據。對象
第二步,咱們要根據 Excel 模型,來導出 Excel 表。
看到這,你應該明白 Excel 導出的設計過程了。接下來,咱們就來一步步實現這個功能。
建立 Excel 模型,涉及到註解三要素中的定義、使用。
首先,定義 Excel 註解,咱們直接看關鍵代碼。
@Target({ElementType.FIELD}) @Retention(RetentionPolicy.RUNTIME) public @interface ExcelField { /** * 導出字段標題 */ String title(); /** * 導出字段排序(升序) */ int sort() default 0; /** * 對齊方式(0:自動;1:靠左;2:居中;3:靠右) */ int align() default 0; }
這裏用到了兩個元註解@Retention
和@target
。@target
表明這個註解只能放在成員變量上;@Retention
表明這個註解要加載到 JVM 內存,咱們能夠用反射來讀取註解。
此外,註解還有 3 個成員變量,分別對應:Excel 的字段標題、字段排序、對齊方式,方便你們微調錶格。到了這,定義 Excel 註解就完成了。
接下來,使用註解,咱們仍是直接看代碼。
public class OrderModel { @ExcelField(title = "訂單號", align = 2, sort = 20) private String orderNo; @ExcelField(title = "金額", align = 2, sort = 20) private String amount; // 建立時間 private Date createTime; // 省略 getter/setter 方法 }
訂單模型有 3 個字段:訂單號、金額、建立時間,但這裏註解只加到訂單號、金額上,表示這兩個字段會導出 Excel 表,而建立時間會忽略,你能夠看看這副圖片。
至此,咱們完成了定義註解、使用註解,獲得了一個 Excel 模型。但要想實現導出功能,還必須根據這個模型,生成出 Excel 表。
讀取 Excel 模型,涉及到註解三要素中的讀取。 咱們須要讀取註解,生成 Excel 表,這主要分紅 3 個步驟:初始化 Excel 表對象—>寫入數據到 Excel 表對象—>輸出文件。
第一步,初始化 Excel 表對象。在這一步中,咱們要根據 Excel 模型,生成一個 Excel 表對象,要建立這幾個東西:標題、表頭、樣式等等。咱們來看代碼。
public class ExcelExporter { // ...省略無數代碼 /***************************** 初始化 Excel 表對象 ****************************/ /** * 構造函數 * @param title 表格標題,傳「空值」,表示無標題 * @param cls excel模型對象 */ public ExcelExporter(String title, Class<?> cls) { // 獲取註解list Field[] fs = cls.getDeclaredFields(); for (Field f : fs) { ExcelField ef = f.getAnnotation(ExcelField.class); if (ef != null) { annotationList.add(new Object[]{ef, f}); } } annotationList.sort(comparing(o -> ((ExcelField) o[0]).sort())); // 經過註解獲取表頭 List<String> headerList = new ArrayList<>(); for (Object[] os : annotationList) { String t = ((ExcelField) os[0]).title(); headerList.add(t); } // 初始化excel表:建立excel表、添加表標題、建立表頭等等 initialize(title, headerList); } }
在初始化的時候,咱們先從 Excel 模型對象中讀取註解,得到一個註解列表;而後,再從註解列表中,讀取 title-字段標題;最後,再初始化 Excel 表對象,包括:建立 Excel 表對象、添加表標題、建立表頭、添加樣式。
第二步,寫入數據到 Excel 表對象。在這一步中,咱們要把 Java 的列表數據寫到 Excel 表對象裏,讓這些數據能變成 Excel 表的一行行信息。仍是來看代碼。
public class ExcelExporter { /***************************** 初始化 Excel 表對象 ****************************/ // ...省略無數代碼 /***************************** 寫入數據到 Excel 表對象 ****************************/ /** * 寫入數據 * @return list 數據列表 */ public <E> ExcelExporter setDataList(List<E> list) { for (E dataObj : list) { // 添加行 Row row = this.addRow(); // 獲取數據,並寫入單元格 int cellNo = 0; for (Object[] os : annotationList) { // 獲取成員變量的值 Object value = null; try { value = Reflections.invokeGetter(dataObj, ((Field) os[1]).getName()); } catch (Exception ex) { log.info(ex.toString()); value = ""; } if (value == null) { value = ""; } // 寫入單元格 ExcelField ef = (ExcelField) os[0]; this.addCell(row, cellNo++, value, ef.align()); } } return this; } }
咱們先傳入一個數據列表 dataList,而後用循環來遍歷 dataList,在這個循環中,咱們不斷把數據寫進 Excel 表對象裏,具體操做是:建立了一個空白行,利用註解獲取成員變量裏的值,最後寫進 Excel 表的單元格里。
第三步,輸出文件。在這一步中,就是 Excel 表對象變成一個文件,來看下最後的代碼吧。
public class ExcelExporter { /***************************** 初始化 Excel 表對象 ****************************/ // ...省略無數代碼 /***************************** 寫入數據到 Excel 表對象 ****************************/ // ...省略無數代碼 /***************************** 輸出相關 ****************************/ /** * 輸出到文件 * @param fileName 輸出文件名,加上絕對路徑 */ public ExcelExporter writeFile(String fileName) throws IOException { FileOutputStream os = new FileOutputStream(fileName); this.write(os); return this; } }
輸出文件就沒什麼好說的了,就是指定文件名,而後把文件輸出到指定的地方。
到了這,讀取 Excel 模型就完成了。
固然,讀取 Excel 模型涉及到註解的讀取,這是註解最難理解的地方,由於讀取註解要用到 Java 另外一個高級特性—反射。並且,註解通常是用來簡化業務,你對業務沒有深入的瞭解,是很難用好註解的。
限於篇幅,我只講了最核心的代碼,項目的完整代碼放在文末的連接上,你們能夠好好看看。
註解想發揮做用,有三個要素:定義、使用、讀取。這篇文章利用了註解的三要素,實現了 Excel 導出功能。
這分紅兩步。第一步,建立 Excel 模型,這涉及到註解三要素中的定義、使用;第二步,讀取 Excel 模型,這涉及到註解三要素中的讀取。
總之,註解通常用來簡化業務,你要想用好註解,不但得熟練掌握 Java 的高級用法,還得對業務有深入的理解。
文章演示代碼: 點擊跳轉