• 文件一键上传、汉字转拼音、excel文件上传下载功能模块的实现


    ----------------------------------------------------------------------------------------------
    [版权申明:本文系作者原创,转载请注明出处] 
    文章出处:http://blog.csdn.net/sdksdk0/article/details/52557755
    作者:朱培      ID:sdksdk0     

    --------------------------------------------------------------------------------------------

    今天要分享的功能是一键上传excel文件,然后显示在页面中,以及将网页上的数据已excel文件的形式下载下来。使用的是Apache的POI,本文分享的实例是SSH框架完成的一个项目中的一个功能模块之一,使用了Maven,所以相关的jar包都贴出了maven的相关jar坐标。因为这个项目设计了很多ssh的配置,这里本文就不在一一列出,只写了关键的代码,文末提供源码下载,希望可以帮助到有需要的开发者。

    一、POI简介

    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)。

    网址是http://poi.apache.org/,我们可以通过添加maven倚赖的方式下载,poi的坐标是:

    		<dependency>
    			<groupId>org.apache.poi</groupId>
    			<artifactId>poi</artifactId>
    			<version>3.9</version>
    		</dependency>

    二、实例准备

    页面效果如下:

    (这里的数据我是随便写的,仅作为参考数据,不是真实数据)

    通过点击页面上的批量导入按钮,可以实现界面无刷新一次性导入写好的数据。我们先excel中准备好数据:注意的是在excel中的数字都要改为文本格式的,不能是“常规",否则导入会失败的。

    三、一键上传

    jsp页面代码:

    需要先导入: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{
    		
    		// 1、 工作薄对象
    		HSSFWorkbook hssfWorkbook = new HSSFWorkbook(new FileInputStream(upload));
    		// 解析工作薄
    		hssfWorkbook.setMissingCellPolicy(Row.CREATE_NULL_AS_BLANK); // 避免空指针异常
    			
    		// 2、 获得Sheet
    		HSSFSheet sheet = hssfWorkbook.getSheetAt(0); // 获得第一个sheet
    
    
    		// 3、遍历每一行
    		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>



    这样我们就可以把数据下载出来了:

    总结:文本使用的都是非常经典的技术来实现了文件一键上传、汉字转拼音、文件上传、文件下载的功能,具有非常好的实际意义,属于开发者必备技能之一。

    项目地址:https://github.com/sdksdk0/BOS  



     
     
  • 相关阅读:
    P1141零一迷宫
    P1219八皇后
    P1233木棍加工
    三 Struts2 添加返回数据
    二 Struts2 接收数据
    一 Struts2 开发流程
    12-tinyMCE文本编辑器+图片上传预览+页面倒计时自动跳转
    11-page分页原理
    10-ajax技术简介
    9-文件上传和下载
  • 原文地址:https://www.cnblogs.com/sdksdk0/p/6060025.html
Copyright © 2020-2023  润新知