之前遇到了这样一个需求,文件上传到数据库中以CLOB存储,并下载。还需支持JPG和PDF格式,百度了解之后就开工了。
说明,本次使用语言为JAVA语言,前台框架为EASYUI2.1,后台SpringMVC+Spring+mybatis前台,由表单发送。直接上代码
<form>
<input class=” easyui-filebox” id=”fj”/> 也可直接用原生file文本框
</form>
easyui-filebox可以很简单的同时上传多个文件,由后台接收
JS验证:(仅对文件类型做了验证,还可以根据fileText.length对文件大小验证)
v
ar fileText = $("#specialChangeFile").filebox('getText');
var flag=meterReadRateSettingAdd.checkFile(fileText);
if(!flag){
API.alert('请上传文件或按格式要求上传!');
return;
}
checkFile:function(fileText){
var flag = true;
if(fileText == "") {
flag=false;
}
var type = fileText.slice(fileText.lastIndexOf(".")+1,fileText.length);
var checkType = type.toUpperCase();
if(checkType !== "PDF"&&checkType !== "JPG"){
flag=false;
};
return flag;
}
表单提交:(文件需同步传输,所以不能使用Ajax,我用的是异步表单提交,网上有ajaxFileUpload 的方法可以试试,原理类似)
var options = {
dataType : "json",
success : function(data) {
},
error : function(result) {
}
};
if($("#formId").form('validate')==true){
$('# formId).ajaxSubmit(options);
}
-
后台
(1)springMVC配置
(2)控制层接收文件
@RequestMapping(value="/xxx")
@ResponseBody
public Map<String, Object> xxx(@RequestParam(value = “fjFile”, required = false)MultipartFile fjFile)//如果是多文件上传,要用MultipartFile[] 接收
普通的流写入:
InputStream fileIs= null;
fileIs= fjFile.getInputStream();
s=fjFile.getOriginalFilename();
注意:这里我做的需求是将文件以CLOB形式存储到本地数据库,所以需要将文件转为二进制,并进行二进制和String的特殊转换处理(主要针对图片和PDF等格式,直接转换会损坏文件),转换之后可以直接以CLOB形式入库
byte[] byteFile = FileCopyUtils.copyToByteArray(fileIs);
indicatorExcludeSettingDto.setFj(byte2hex(byteFile));
/**
* 二进制转字符串
* @param b byte数组
* @return 二进制字符串
*/
public static String byte2hex(byte[] b){
StringBuffer sb = new StringBuffer();
String stmp = “”;
for (int n = 0; n < b.length; n++) {
stmp = Integer.toHexString(b[n] & 0XFF);
if (stmp.length() == 1) {
sb.append(“0” + stmp);
} else {
sb.append(stmp);
}
}
return sb.toString();
}
读取:
OutputStream os = null;
ByteArrayInputStream bin=null;
try {
response.setContentType("multipart/form-data");
//处理下载弹出框名字的编码问题
response.setHeader("Content-Disposition", "attachment;fileName=\"" + new String(file.getFileName.getBytes ("gb2312"), "ISO8859-1" )+file.getFjlx()+ "\"");
os = response.getOutputStream();
bin = new ByteArrayInputStream(hex2byte(file.getFj()));
int len=0;
byte[] buffer=new byte[1024];
while((len=bin.read(buffer))>0){
os.write(buffer,0,len);
}
os.flush();
当然,读取时也要进行转换。
/**
* 字符串转二进制
* @param str 字符串
* @return byte数组
*/
public static byte[] hex2byte(String str) {
if (str == null)
return null;
str = str.trim();
int len = str.length();
if (len == 0 || len % 2 == 1)
return null;
byte[] b = new byte[len / 2];
try {
for (int i = 0; i < str.length(); i += 2) {
b[i / 2] = (byte) Integer.decode("0X" + str.substring(i, i + 2)).intValue();
}
return b;
} catch (Exception e) {
return null;
}
}
至此,文件上传、下载并以CLOB形式存储到数据库便完成了,写出来还是很简单的。当时做的时候,没少被坑,尤其是字符串和二进制的转换,很关键。CLOB最大能存储4G,一般是能够用了,可以在SpringMVC配置里限制大小。系统图片操作多,有条件的公司,建议还是搭个图片服务器,既方便操作也可减轻数据库的压力。
接下来补充一个FTP上传下载的操作,由于历史原因(就是我也不清楚的原因),系统的所有文件导出功能都需要两次请求,而我们部署了20个节点,使用服务器本地上传下载又与后来配置的NGNIX的IPHASH算法冲突,鉴于此,我们决定导出均使用FTP服务器。导出格式为EXCLE,技术为POI的HSSF。
- 前台:Easyui导出官方文档:http://www.jeasyui.net/extension/204.html
- 算了,时隔二个月才写,忘完了,直接使用FTPCLIENT工具类吧,POI的使用可以参考官方文档