這個功能也是我以前項目中經常用到的,感覺很實用,必須拿來分享下:
excel進行批量導入數據,結合struts2+ajax
導入的視圖:batchAdd.ftl(視圖無關緊要的,可以換成其他任何視圖,就是普通的form表單的提交,就是這里是調用ajax)
<div class="controltitle">當前操作:excel批量導入學生信息</div> <div> <form action="excelUploadAction.action" method="post" enctype="multipart/form-data" id="uploadInfor" onSubmit="return validateuploadInforFile(this);"> <input type="file" name="uploadExcel"> <input type="submit" value="提交"/><a href = "download/stuExcel.xls">[點擊下載excel樣例]</a> </form> </div> <div> <div>下面是導入的excel的樣式<font color = "red">[上傳的excel中的內容必須按照這個順序就行排版,否則存儲內容的位置會錯亂的]</font></div> <table class="tablefirst" id="radioSubStyle"> <tr> <th>姓名</th><th>學號</th><th>性別</th><th>籍貫</th><th>專業</th><th>學制</th><th>入學年份</th> <th>畢業年份</th><th>工作省市</th><th>工作單位</th><th>工作崗位</th><th>職務職稱</th> <th>電話</th><th>手機</th><th>qq</th><th>郵箱</th><th>通信地址</th><th>家庭地址</th> </tr> <tr> <td>沈浪</td><td>1006010054</td><td>男</td><td>漢</td><td>眼7</td><td>7</td><td>2009-11-11</td> <td>2011-11-11</td><td>浙江</td><td>醫院</td><td>臨床</td><td>主任</td> <td>642</td><td>159</td><td>5449</td><td>544@qq.com</td><td>杭州</td><td>杭州</td> </tr> <tr> <td>沈浪</td><td>1006010024</td><td>男</td><td>漢</td><td>眼7</td><td>7</td><td>2009-11-11</td> <td>2011-11-11</td><td>浙江</td><td>醫院</td><td>臨床</td><td>主任</td> <td>642</td><td>159</td><td>5449</td><td>544@qq.com</td><td>杭州</td><td>杭州</td> </tr> <tr> <td>沈浪</td><td>1006010034</td><td>男</td><td>漢</td><td>眼7</td><td>7</td><td>2009-11-11</td> <td>2011-11-11</td><td>浙江</td><td>醫院</td><td>臨床</td><td>主任</td> <td>642</td><td>159</td><td>5449</td><td>544@qq.com</td><td>杭州</td><td>杭州</td> </tr> </table> <div><font color="red">注:①excel各列的順序必須按樣式的順序,而內容可依實際內容而定;這里不支持照片的導入,照片可在修改中進行上傳②性別只能為‘男’與‘女’;③專業只能填:眼視光七年制;眼視光本科;眼視光專科;否則會影響后面的查詢</font></div> </div> <div id="excelUploadMsg"></div>
傳入的js(ajax+文件的驗證):這里涉及到的ajax可以去閱讀:http://www.cnblogs.com/shenliang123/archive/2012/04/16/2452670.html
/** * 以下是上傳excel的一系列方法 * */ function validateuploadInforFile(form){ if(!validateExcelUpLoadFile(form)) return false; var options = { dataType: 'json', success: showResponse }; $("#uploadInfor").ajaxSubmit(options); return false; } function showResponse(responseText){ $("#excelUploadMsg").empty(); $("#excelUploadMsg").append(responseText.msg); } //導入考試時進行文件格式校驗 function validateExcelUpLoadFile(form) { var fileName = form.uploadExcel.value; if (fileName != "" ) { var fileType = (fileName.substring(fileName.lastIndexOf(".") + 1, fileName.length)).toLowerCase(); var suppotFile = ["xls", "XLS", "xlsx", "XLSX"]; for (var i = 0; i < suppotFile.length; i++) { if (suppotFile[i] == fileType) { return true; } else { continue; } } alert("文件格式不正確!"); return false; } else { alert("請選擇你需要的導入 的文件"); return false; } }
struts.xml:
<!-- excel導入 -->
<action name="excelUploadAction" class="xidian.sl.action.StuInforAction" method = "excelUpload">
<interceptor-ref name="fileUpload"/>
<interceptor-ref name="defaultStack" />
<result name="success">/WEB-INF/responseMsg.jsp</result>
</action>
action中的方法處理:再次進行文件的驗證和調用服務層進行excel的處理和數據持久化
//以下為單文件上傳,即excel private File uploadExcel; //文件 private String uploadExcelFileName; //文件名 private static String[] allowFileType = { "xls", "XLS", "xlsx", "XLSX" }; //控制文件類型 /** * excel批量導入 * */ public String excelUpload(){ try{ if ((uploadExcelFileName == null) || (uploadExcelFileName.equals(""))) { response = "{success:false,msg:'文件名不能為空!'}"; } if (!(FileUtil.validateFileType(uploadExcelFileName, allowFileType))) { //validateFileType方法在FileUtil中,返回的是boolean值 response = "{success:false,msg:'文件類型不正確!'}"; } excelBatchInput.uploadStu(uploadExcel); //只傳入一個excel文件 response = "{success:true,msg:'"+excelBatchInput.getFinalMsg()+"'}"; //得到結果的回復 }catch (Exception e) { e.printStackTrace(); response = "{success:true,msg:'sorry: 導入失敗'}"; } return SUCCESS; }
serviceInterface:
package xidian.sl.service.admin; import java.io.File; public interface ExcelBatchInput { /** * excel導入的主要邏輯 * */ public String uploadStu(File upload); /** * 導入后的消息 * */ public String getFinalMsg(); }
ServiceImpl:(這里對excel的內容沒有做過多的驗證,只是驗證了學號為不重復,大家可以自行對其進行添加)
package xidian.sl.service.impl.admin; import java.io.File; import java.io.FileInputStream; import java.io.InputStream; import java.util.List; import jxl.Cell; import jxl.CellType; import jxl.DateCell; import jxl.LabelCell; import jxl.Sheet; import jxl.Workbook; import xidian.sl.dao.admin.StuInforDAO; import xidian.sl.entity.StuInfor; import xidian.sl.service.admin.ExcelBatchInput; public class ExcelBatchInputImpl implements ExcelBatchInput { private Sheet sheet; private String[][] excelValue; private int successRow; private int failRow; private StringBuilder msg=new StringBuilder(); private String finalMsg; private StuInforDAO stuInforDAO; /** * excel導入的總方法 */ public String uploadStu(File upload) { initExcel(upload); //初始化 readExcel(); //讀取 insertIntoDB(); //插入 return msg.toString(); } /** * 讀取excel文件中數據,保存到sheet對象中 * @param upload */ private void initExcel(File upload) { Workbook rwb = null; try { InputStream is = new FileInputStream(upload); rwb = Workbook.getWorkbook(is); sheet = rwb.getSheet(0); } catch (Exception e) { e.printStackTrace(); } } /** * 讀取excel中數據進入excelValue數組中 */ private void readExcel(){ excelValue = new String[sheet.getRows()][sheet.getColumns()]; for (int i = 0; i < sheet.getRows(); i++) for (int j = 0; j < sheet.getColumns(); j++) { Cell cell = sheet.getCell(j, i); if ("".equals(cell.getContents().toString().trim())){ excelValue[i][j] = ""; } if (cell.getType() == CellType.LABEL) { LabelCell labelcell = (LabelCell)cell; excelValue[i][j] = labelcell.getString().trim(); } else if (cell.getType() == CellType.NUMBER){ excelValue[i][j] = cell.getContents(); } else if (cell.getType() == CellType.DATE) { DateCell datcell = (DateCell)cell; excelValue[i][j] = datcell.getDate().toString(); } else { excelValue[i][j] = cell.getContents().toString().trim(); } } } /** * 3.保存進入數據庫 * @param course */ private void insertIntoDB() { int excelRows = excelValue.length; //將消息清空 msg.delete(0, msg.length()); finalMsg = ""; successRow = 0; failRow = 0; if (excelValue.length > 1) { for (int i = 1; i < excelRows; i++) { //從第二排開始,第一排為文字說明 String[] DBValue = excelValue[i]; //取一行數據 if (validateInfor(i,DBValue)){ successRow += 1; finalInsert(DBValue); } else { failRow += 1; } } finalMsg = "錄入成功結束:"+"</br>"+"目標導入學生:"+(successRow+failRow)+"</br>" +"成功錄入數:"+(successRow)+"</br>" +"失敗錄入數:"+(failRow)+"</br>" +msg.toString(); System.out.println(finalMsg); } else { finalMsg = "excel中無任何數據!"; System.out.println("excel中沒有任何數據"); } } //最終返回檢驗結果 private boolean validateInfor(int i,String[] DBValue){ Boolean bol=true; if(!(validateNumRepeat(i,DBValue))){ bol=false; } return bol; } //驗證崗位是否重復 private boolean validateNumRepeat(int i,String[] DBValue){ boolean bolValidate=true; /** * 主要檢查學號是否有重復 * 這里需要改進,因為這里會每插入一條就會進行一次數據庫的搜索 * */ String hql = "from StuInfor"; List<StuInfor> stuInfors = stuInforDAO.getStuInforListByHQL(hql); for(StuInfor stuInfor: stuInfors){ if(stuInfor.getStuNum().equals(DBValue[1]) && DBValue[1]!=""){ bolValidate=false; msg.append("錯誤信息:第"+i+"學生的學號與數據庫已存儲的重復,該學號是["+DBValue[1]+"],導入的學生名字["+DBValue[0]+"],請檢查!</br>"); return bolValidate; } } return bolValidate; } //最終插入數據,這里需要修改 private void finalInsert(String[] DBValue){ StuInfor stuInfor = new StuInfor(); stuInfor.setStuName(DBValue[0]); stuInfor.setStuNum(DBValue[1]); stuInfor.setStuSex(DBValue[2]); stuInfor.setStuJg(DBValue[3]); stuInfor.setStuZy(DBValue[4]); stuInfor.setStuXz(DBValue[5]); stuInfor.setStuStartTime(DBValue[6]); stuInfor.setStuEndTime(DBValue[7]); stuInfor.setStuWorkAddress(DBValue[8]); stuInfor.setStuWorkPlace(DBValue[9]); stuInfor.setStuWorkPost(DBValue[10]); stuInfor.setStuWorkZc(DBValue[11]); stuInfor.setStuPhone(DBValue[12]); stuInfor.setStuTelephone(DBValue[13]); stuInfor.setStuQq(DBValue[14]); stuInfor.setStuEmail(DBValue[15]); stuInfor.setStuCommAddress(DBValue[16]); stuInfor.setStuAddress(DBValue[17]); stuInfor.setDeleteSign("1"); stuInforDAO.makePersistence(stuInfor); } public Sheet getSheet() { return sheet; } public void setSheet(Sheet sheet) { this.sheet = sheet; } public String[][] getExcelValue() { return excelValue; } public void setExcelValue(String[][] excelValue) { this.excelValue = excelValue; } public int getSuccessRow() { return successRow; } public void setSuccessRow(int successRow) { this.successRow = successRow; } public int getFailRow() { return failRow; } public void setFailRow(int failRow) { this.failRow = failRow; } public StringBuilder getMsg() { return msg; } public void setMsg(StringBuilder msg) { this.msg = msg; } public String getFinalMsg() { return finalMsg; } public void setFinalMsg(String finalMsg) { this.finalMsg = finalMsg; } public StuInforDAO getStuInforDAO() { return stuInforDAO; } public void setStuInforDAO(StuInforDAO stuInforDAO) { this.stuInforDAO = stuInforDAO; } }
里面涉及到的持久層的操作我這里就省略不寫了