----------------------------------------------------------------------------------------------
[版權申明:本文系做者原創,轉載請註明出處]
文章出處:http://blog.csdn.net/sdksdk0/article/details/52557755
做者:朱培 ID:sdksdk0 html
--------------------------------------------------------------------------------------------java
今天要分享的功能是一鍵上傳excel文件,而後顯示在頁面中,以及將網頁上的數據已excel文件的形式下載下來。使用的是Apache的POI,本文分享的實例是SSH框架完成的一個項目中的一個功能模塊之一,使用了Maven,因此相關的jar包都貼出了maven的相關jar座標。由於這個項目設計了不少ssh的配置,這裏本文就不在一一列出,只寫了關鍵的代碼,文末提供源碼下載,但願能夠幫助到有須要的開發者。jquery
Apache POI 是用Java編寫的免費開源的跨平臺的 Java API,Apache POI提供API給Java程式對Microsoft Office格式檔案讀和寫的功能。POI爲「Poor Obfuscation Implementation」的首字母縮寫,意爲「可憐的模糊實現」。
Apache POI 是建立和維護操做各類符合Office Open XML(OOXML)標準和微軟的OLE 2複合文檔格式(OLE2)的Java API。用它可使用Java讀取和建立,修改MS Excel文件.並且,還可使用Java讀取和建立MS Word和MSPowerPoint文件。Apache POI 提供Java操做Excel解決方案(適用於Excel97-2008)。git
網址是http://poi.apache.org/,咱們能夠經過添加maven倚賴的方式下載,poi的座標是:github
<dependency> <groupId>org.apache.poi</groupId> <artifactId>poi</artifactId> <version>3.9</version> </dependency>
頁面效果以下:web
(這裏的數據我是隨便寫的,僅做爲參考數據,不是真實數據)apache
經過點擊頁面上的批量導入按鈕,能夠實現界面無刷新一次性導入寫好的數據。咱們先excel中準備好數據:注意的是在excel中的數字都要改成文本格式的,不能是「常規",不然導入會失敗的。json
jsp頁面代碼:api
須要先導入:jquery.ocupload-1.1.2.js數組
而後再頁面中給你的按鈕的id添加upload的方法。 (如下片斷在文末項目中的位置是:/BOS/src/main/webapp/WEB-INF/pages/base/region.jsp)
// 對批量導入添加一鍵上傳效果 $('#button-import').upload({ name : 'upload', action : '${pageContext.request.contextPath}/region_OCimport', // 表單提交路徑 onComplete : function(response){ var data = eval("("+response+")"); $.messager.alert('信息',data.msg,'info'); $('#grid').datagrid('reload'); } });
這個我轉到的是Action中:
這裏的話我就是加了pinyin4j來將漢字轉爲拼音,這樣的話作的就是簡碼了;
在pom.xml中添加maven依賴:(如下片斷在文末項目中的位置是:/BOS/pom.xml)
<dependency> <groupId>com.belerweb</groupId> <artifactId>pinyin4j</artifactId> <version>2.5.0</version> </dependency>
在Action中處理:(如下片斷在文末項目中的位置是:/BOS/src/main/java/cn/tf/bos/web/action/bc/RegionAction.java)
//接收上傳的數據 public String OCimport() throws IOException{ // 一、 工做薄對象 HSSFWorkbook hssfWorkbook = new HSSFWorkbook(new FileInputStream(upload)); // 解析工做薄 hssfWorkbook.setMissingCellPolicy(Row.CREATE_NULL_AS_BLANK); // 避免空指針異常 // 二、 得到Sheet HSSFSheet sheet = hssfWorkbook.getSheetAt(0); // 得到第一個sheet // 三、遍歷每一行 for (Row row : sheet) { if (row.getRowNum() == 0) { continue; } // 從第二行 開始解析 Region region = new Region(); String id = row.getCell(0).getStringCellValue(); // 得到第一個單元格信息 if (id.trim().equals("")) { // id 無值,跳過 continue; } region.setId(id); region.setProvince(row.getCell(1).getStringCellValue()); region.setCity(row.getCell(2).getStringCellValue()); region.setDistrict(row.getCell(3).getStringCellValue()); region.setPostcode(row.getCell(4).getStringCellValue()); //使用pinyin4j生成編碼和簡碼 String str=region.getProvince()+region.getCity()+region.getDistrict(); str=str.replaceAll("省", "").replaceAll("市", "").replaceAll("區", "").replaceAll("縣", ""); String[] arr=PinYin4jUtils.getHeadByString(str); StringBuffer sb = new StringBuffer(); for (String headChar : arr) { sb.append(headChar); } region.setShortcode(sb.toString()); // 簡碼 // 生成城市編碼 region.setCitycode(PinYin4jUtils.hanziToPinyin(region.getCity(), "")); //保存數據時出錯 try { regionService.saveRegion(region); } catch (Exception e) { // 導入region失敗,記錄日誌 LOG.error("區域導入失敗,編號:" + region.getId(), e); } } //返回json Map<String,Object> map=new HashMap<String,Object>(); map.put("result", "success"); map.put("msg", "區域導入完成"); ActionContext.getContext().put("map", map); return "OCimport"; } private File upload; public void setUpload(File upload) { this.upload = upload; }
漢字轉換爲拼音的工具類爲:(如下片斷在文末項目中的位置是:/BOS/src/main/java/cn/tf/bos/utils/PinYin4jUtils.java)
public class PinYin4jUtils { /** * 將字符串轉換成拼音數組 * * @param src * @return */ public static String[] stringToPinyin(String src) { return stringToPinyin(src, false, null); } /** * 將字符串轉換成拼音數組 * * @param src * @return */ public static String[] stringToPinyin(String src, String separator) { return stringToPinyin(src, true, separator); } /** * 將字符串轉換成拼音數組 * * @param src * @param isPolyphone * 是否查出多音字的全部拼音 * @param separator * 多音字拼音之間的分隔符 * @return */ public static String[] stringToPinyin(String src, boolean isPolyphone, String separator) { // 判斷字符串是否爲空 if ("".equals(src) || null == src) { return null; } char[] srcChar = src.toCharArray(); int srcCount = srcChar.length; String[] srcStr = new String[srcCount]; for (int i = 0; i < srcCount; i++) { srcStr[i] = charToPinyin(srcChar[i], isPolyphone, separator); } return srcStr; } /** * 將單個字符轉換成拼音 * * @param src * @return */ public static String charToPinyin(char src, boolean isPolyphone, String separator) { // 建立漢語拼音處理類 HanyuPinyinOutputFormat defaultFormat = new HanyuPinyinOutputFormat(); // 輸出設置,大小寫,音標方式 defaultFormat.setCaseType(HanyuPinyinCaseType.LOWERCASE); defaultFormat.setToneType(HanyuPinyinToneType.WITHOUT_TONE); StringBuffer tempPinying = new StringBuffer(); // 若是是中文 if (src > 128) { try { // 轉換得出結果 String[] strs = PinyinHelper.toHanyuPinyinStringArray(src, defaultFormat); // 是否查出多音字,默認是查出多音字的第一個字符 if (isPolyphone && null != separator) { for (int i = 0; i < strs.length; i++) { tempPinying.append(strs[i]); if (strs.length != (i + 1)) { // 多音字之間用特殊符號間隔起來 tempPinying.append(separator); } } } else { tempPinying.append(strs[0]); } } catch (BadHanyuPinyinOutputFormatCombination e) { e.printStackTrace(); } } else { tempPinying.append(src); } return tempPinying.toString(); } public static String hanziToPinyin(String hanzi) { return hanziToPinyin(hanzi, " "); } /** * 將漢字轉換成拼音 * * @param hanzi * @param separator * @return */ public static String hanziToPinyin(String hanzi, String separator) { // 建立漢語拼音處理類 HanyuPinyinOutputFormat defaultFormat = new HanyuPinyinOutputFormat(); // 輸出設置,大小寫,音標方式 defaultFormat.setCaseType(HanyuPinyinCaseType.LOWERCASE); defaultFormat.setToneType(HanyuPinyinToneType.WITHOUT_TONE); String pinyingStr = ""; try { pinyingStr = PinyinHelper.toHanyuPinyinString(hanzi, defaultFormat, separator); } catch (BadHanyuPinyinOutputFormatCombination e) { // TODO Auto-generated catch block e.printStackTrace(); } return pinyingStr; } /** * 將字符串數組轉換成字符串 * * @param str * @param separator * 各個字符串之間的分隔符 * @return */ public static String stringArrayToString(String[] str, String separator) { StringBuffer sb = new StringBuffer(); for (int i = 0; i < str.length; i++) { sb.append(str[i]); if (str.length != (i + 1)) { sb.append(separator); } } return sb.toString(); } /** * 簡單的將各個字符數組之間鏈接起來 * * @param str * @return */ public static String stringArrayToString(String[] str) { return stringArrayToString(str, ""); } /** * 將字符數組轉換成字符串 * * @param str * @param separator * 各個字符串之間的分隔符 * @return */ public static String charArrayToString(char[] ch, String separator) { StringBuffer sb = new StringBuffer(); for (int i = 0; i < ch.length; i++) { sb.append(ch[i]); if (ch.length != (i + 1)) { sb.append(separator); } } return sb.toString(); } /** * 將字符數組轉換成字符串 * * @param str * @return */ public static String charArrayToString(char[] ch) { return charArrayToString(ch, " "); } /** * 取漢字的首字母 * * @param src * @param isCapital * 是不是大寫 * @return */ public static char[] getHeadByChar(char src, boolean isCapital) { // 若是不是漢字直接返回 if (src <= 128) { return new char[] { src }; } // 獲取全部的拼音 String[] pinyingStr = PinyinHelper.toHanyuPinyinStringArray(src); // 建立返回對象 int polyphoneSize = pinyingStr.length; char[] headChars = new char[polyphoneSize]; int i = 0; // 截取首字符 for (String s : pinyingStr) { char headChar = s.charAt(0); // 首字母是否大寫,默認是小寫 if (isCapital) { headChars[i] = Character.toUpperCase(headChar); } else { headChars[i] = headChar; } i++; } return headChars; } /** * 取漢字的首字母(默認是大寫) * * @param src * @return */ public static char[] getHeadByChar(char src) { return getHeadByChar(src, true); } /** * 查找字符串首字母 * * @param src * @return */ public static String[] getHeadByString(String src) { return getHeadByString(src, true); } /** * 查找字符串首字母 * * @param src * @param isCapital * 是否大寫 * @return */ public static String[] getHeadByString(String src, boolean isCapital) { return getHeadByString(src, isCapital, null); } /** * 查找字符串首字母 * * @param src * @param isCapital * 是否大寫 * @param separator * 分隔符 * @return */ public static String[] getHeadByString(String src, boolean isCapital, String separator) { char[] chars = src.toCharArray(); String[] headString = new String[chars.length]; int i = 0; for (char ch : chars) { char[] chs = getHeadByChar(ch, isCapital); StringBuffer sb = new StringBuffer(); if (null != separator) { int j = 1; for (char ch1 : chs) { sb.append(ch1); if (j != chs.length) { sb.append(separator); } j++; } } else { sb.append(chs[0]); } headString[i] = sb.toString(); i++; } return headString; } }
在struts.xml中配置返回:(如下片斷在文末項目中的位置是:/BOS/src/main/resources/struts.xml)
<action name="region_*" class="regionAction" method="{1}"> <result name="OCimport" type="json"> <param name="root">map</param> </result> </action>
這樣的話就所有實現了上傳的功能了。
接下來講一下下載的功能:
例如咱們要將這個頁面中的內容變爲excel文件下載下來。
在下載的jsp頁面中:
點擊按鈕觸發這個下載的函數,咱們有一個函數:點擊後跳轉到action中。(如下片斷在文末項目中的位置是:/BOS/src/main/webapp/WEB-INF/pages/base/subarea.jsp)
function doExport(){ location.href="${pageContext.request.contextPath}/subarea_exportFile"; }
在action中:(如下片斷在文末項目中的位置是:/BOS/src/main/java/cn/tf/bos/web/action/bc/SubareaAction.java)
public String exportFile() throws IOException{ // 對文件名進行編碼 String downloadFileName = "分區數據.xls"; // 得到用戶使用瀏覽器類型 String agent = ServletActionContext.getRequest().getHeader("user-agent"); // 對下載文件名編碼 downloadFileName = FileUtils.encodeDownloadFilename(downloadFileName, agent); // 將結果放入值棧 ActionContext.getContext().put("downloadFileName", downloadFileName); return "exportFile"; } //文件下載流 public InputStream getInputStream() throws IOException{ PageResponseBean pageResponseBean=(PageResponseBean) ServletActionContext.getRequest().getSession().getAttribute("pageResponseBean"); List<Subarea> subareas=pageResponseBean.getRows(); HSSFWorkbook hssfWorkbook=new HSSFWorkbook(); HSSFSheet sheet=hssfWorkbook.createSheet("分區數據"); HSSFRow headRow=sheet.createRow(0); headRow.createCell(0).setCellValue("分區編號"); headRow.createCell(1).setCellValue("關鍵字"); headRow.createCell(2).setCellValue("起始號"); headRow.createCell(3).setCellValue("結束號"); headRow.createCell(4).setCellValue("是否區分單雙號號"); headRow.createCell(5).setCellValue("位置信息"); // 向excel寫數據 for (Subarea subarea : subareas) { // 每一個分區一行 HSSFRow dataRow = sheet.createRow(sheet.getLastRowNum() + 1); dataRow.createCell(0).setCellValue(subarea.getId()); dataRow.createCell(1).setCellValue(subarea.getAddresskey()); dataRow.createCell(2).setCellValue(subarea.getStartnum()); dataRow.createCell(3).setCellValue(subarea.getEndnum()); dataRow.createCell(4).setCellValue(subarea.getSingle()); dataRow.createCell(5).setCellValue(subarea.getPosition()); } // 將數據緩存到字節數組 ByteArrayOutputStream arrayOutputStream = new ByteArrayOutputStream(); hssfWorkbook.write(arrayOutputStream); arrayOutputStream.close(); byte[] data = arrayOutputStream.toByteArray(); // 再經過字節數組輸入流讀取數據 return new ByteArrayInputStream(data); }
在struts.xml中配置:
<result name="exportFile" type="stream"> <param name="contentType">application/vnd.ms-excel</param> <param name="contentDisposition">attachment;filename=${downloadFileName}</param> </result>
這樣咱們就能夠把數據下載出來了:
總結:文本使用的都是很是經典的技術來實現了文件一鍵上傳、漢字轉拼音、文件上傳、文件下載的功能,具備很是好的實際意義,屬於開發者必備技能之一。