文件上传与下载
文件上传
文件上传的几个步骤:
1 获得上传路径 如果上传的文件需要安全的存放 则应该放在WEB-INF 下面
String savePath=request.getServletContext().getRealPath("/WEB-INF/upload1");
2 创建解析工厂
DiskFileItemFactory diskFileItemFactory=new DiskFileItemFactory();
3 创建解析器
ServletFileUpload servletFileUpload=new ServletFileUpload(diskFileItemFactory);
4 判断是不是表单数据(如果是表单数据 继续进行后面操作)
ServletFileUpload.isMultipartContent(request)
5 使用ServletFileUpload解析器解析上传数据,解析结果返回的是一个List<FileItem>集合,每一个FileItem对应一个Form表单的输入项
List<FileItem> list=servletFileUpload.parseRequest(request);
6 遍历每一个FileItem
7 如果FileItem是普通表单项,则按普通表单项处理方式处理(取值),如果是文件则进行后面操作
fileItem.isFormField()
8 获取上传文件的名字(通过截取字段获得真实名字)
(1) String fileName=fileItem.getName();
此名字字符串包含包含一些系统路径,如下面这个字符串所示。
C:\Windows\Installer\{90140000-0011-0000-0000-0000000FF1CE}\graph.ico
(2) 截取字符串 获得真实名字
fileName=fileName.substring(fileName.lastIndexOf("\\")+1);
9 将fileItem写入流中(方便后面写到磁盘中)
InputStream in=fileItem.getInputStream();
10 创建输出流 并将输入流中的数据写入输出流中,并删除临时文件
FileOutputStream out = new FileOutputStream(sqlPath);
byte[] bytes=new byte[1024]; int l=0; while((l=in.read(bytes))!=-1){ out.write(bytes, 0, l); } //删除处理文件上传时生成的临时文件 fileItem.delete(); msg = "文件上传成功!";
文件下载的步骤:
1 获取传递过来的要下载的文件的路径和姓名参数
String filename=req.getParameter("filename");
String filepath=req.getParameter("filepath");
2 将要下载的文件放进输入流中
InputStream in = new FileInputStream(filepath);
3 设置响应头,对文件进行url编码
filename = URLEncoder.encode(filename, "UTF-8"); resp.setHeader("Content-Disposition", "attachment;filename="+filename);
4 创建输出流 对文件进行copy
拷贝 OutputStream out = resp.getOutputStream(); byte[] b = new byte[1024]; int len = 0; while((len = in.read(b))!=-1){ out.write(b, 0, len); } out.flush(); out.close(); in.close()
在本实例中 我设计了一个简单数据库file (filename 文件名,filepath 文件路径), 采用dao模式 将上传的文件的路径和文件名保存在数据库中 ,在再前台页面中奖所有的已上传的文件读取出来显示在页面上 ,在使用文件下载的方式将其下载。
具体代码如下:
《一》 文件上传
(1)前台页面
<form method="post" action="${pageContext.request.contextPath }/fileUpload1" enctype="multipart/form-data"> 上传者:<input type="text" name="username"></br> <input type="file" name="file"><br> <input type="file" name="file1"><br> <input type="submit" value="submit"> </form>
(2) web层 (上传成功后 跳转到file_uplodad/g.jsp 这个页面 注意要让文件全部操作完后再跳转,如果上传的时候有个以上文件,你在上传一个就跳转页面会程序报错)
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { FileUploadServices fileUploadServices=new FileUploadServices(); /*获得上传路径*/ String savePath=request.getServletContext().getRealPath("/WEB-INF/upload1"); File file=new File(savePath); if(!file.exists()){ System.out.println(savePath+"不存在,请创建"); file.mkdir(); } String msg=""; //1创建解析工厂 DiskFileItemFactory diskFileItemFactory=new DiskFileItemFactory(); //2创建解析器 ServletFileUpload servletFileUpload=new ServletFileUpload(diskFileItemFactory); //中文乱码问题 servletFileUpload.setHeaderEncoding("utf-8"); //3判断是不是表单数据 if(!ServletFileUpload.isMultipartContent(request)){ System.out.println("上传的不是表单数据"); } //4、使用ServletFileUpload解析器解析上传数据,解析结果返回的是一个List<FileItem>集合,每一个FileItem对应一个Form表单的输入项 try { List<FileItem> list=servletFileUpload.parseRequest(request); for(FileItem fileItem:list){ if(fileItem.isFormField()){ String name=fileItem.getName(); String value=fileItem.getString("utf-8");//解决乱码问题 System.out.println(name+"="+value); } else{ String fileName=fileItem.getName();//注意是getName 而不是getFileName()---file 表单中的 if(fileName!=""){ System.out.println(fileName+" fileName"); //C:\Windows\Installer\{90140000-0011-0000-0000-0000000FF1CE}\graph.ico //截取后面部分得到真实名字 \graph.ico fileName=fileName.substring(fileName.lastIndexOf("\\")+1); System.out.println(fileName+"截取后面部分得到真实名字"); //获取fileItem的输入流 InputStream in=fileItem.getInputStream(); //创建一个文件输出流 Date date=new Date(); SimpleDateFormat SimpleDateFormat=new SimpleDateFormat("yyyy_MM_dd_HH_mm_ss"); String da=SimpleDateFormat.format(date); String sqlPath=savePath + "\\" +da+"_"+ fileName; System.out.println(sqlPath+" sqlPath"); //sqlPath=sqlPath.replaceAll("\\", "/"); //判断是否上传了文件 System.out.println(sqlPath.substring(sqlPath.lastIndexOf("_")+1)+"dd"); if(sqlPath.substring(sqlPath.lastIndexOf("_")+1)!=" "){ FileOutputStream out = new FileOutputStream(sqlPath); byte[] bytes=new byte[1024]; int l=0; while((l=in.read(bytes))!=-1){ out.write(bytes, 0, l); } //删除处理文件上传时生成的临时文件 fileItem.delete(); msg = "文件上传成功!"; String sql="insert into file (filepath,filename) values(?,?)"; try { if(sqlPath.substring(sqlPath.lastIndexOf("_")+1).length()>1){ sqlPath=sqlPath.replace("\\", "/"); /*pst=(PreparedStatement) conn.prepareStatement(sql); pst.setString(1,sqlPath ); pst.setString(2,fileName ); pst.execute();*/ fileUploadServices.insert(sql,new Object[]{sqlPath,fileName}); } } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } } } } } } catch (FileUploadException e) { // TODO Auto-generated catch block e.printStackTrace(); } response.sendRedirect("/file_uplodad/g.jsp"); }
(3) services层
FileUploadDao fileUploadDao =new FileUploadDao(); public void insert(String sql,Object ...obj) { // TODO Auto-generated method stub fileUploadDao.insert(sql,obj); }
(4)dao层
Connection conn=null; ResultSet rs=null; PreparedStatement stmt=null; try { MyPool myPool=new MyPool(10); ///自己写的简单连接池 conn=myPool.getConnection(); // 设置参数 stmt=(PreparedStatement) conn.prepareStatement(sql); ParameterMetaData parameterMetaData; parameterMetaData = stmt.getParameterMetaData(); int count =parameterMetaData.getParameterCount(); for (int i = 1; i <= count; i++) { stmt.setObject(i, obj[i - 1]); } stmt.executeUpdate(); conn.close();
《二》 文件下载
文件下载dao模式和上传的差不多 ,只是从数据库中将文件名和文件路径显示出来,并作为下载时的传递参数给文件下载的servlet 这里 我就只展示文件下载的servlet
下载传递的参数:
<a href="${pageContext.request.contextPath }/download?filepath=<%=list.get(i).getFilepath()%>&filename=<%=list.get(i).getFilename() %>">下载</a>
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //第一步:设置响应类型 resp.setContentType("application/force-download");//应用程序强制下载 //第二读取文件 // String path = getServletContext().getRealPath("/up/"+name); String filename=req.getParameter("filename"); String filepath=req.getParameter("filepath"); System.out.println(filename+" filename"); System.out.println(filepath+ " filepath"); InputStream in = new FileInputStream(filepath); //设置响应头,对文件进行url编码 filename = URLEncoder.encode(filename, "UTF-8"); resp.setHeader("Content-Disposition", "attachment;filename="+filename); //resp.setContentLength(in.available()); //第三步:拷贝 OutputStream out = resp.getOutputStream(); byte[] b = new byte[1024]; int len = 0; while((len = in.read(b))!=-1){ out.write(b, 0, len); } out.flush(); out.close(); in.close();
在将上传的文件路径存入数据库之前 课改替换掉路径中的"\",故用到两个方法replace(),replaceAll(),
replace和replaceAll是JAVA中常用的替换字符的方法,它们的区别是:
1)replace的参数是char和CharSequence,即可以支持字符的替换,也支持字符串的替换(CharSequence即字符串序列的意思,说白了也就是字符串);
2)replaceAll的参数是regex,即基于规则表达式的替换,比如,可以通过replaceAll("\\d", "*")把一个字符串所有的数字字符都换成星号;
相同点:都是全部替换,即把源字符串中的某一字符或字符串全部换成指定的字符或字符串,如果只想替换第一次出现的,可以使用replaceFirst(),这个方法也是基于规则表达式的替换,但与replaceAll()不同的是,只替换第一次出现的字符串;
另外,如果replaceAll()和replaceFirst()所用的参数据不是基于规则表达式的,则与replace()替换字符串的效果是一样的,即这两者也支持字符串的操作;
还有一点注意::执行了替换操作后,源字符串的内容是没有发生改变的。
举例如下:
String src = new String("ab43a2c43d"); System.out.println(src.replace("3","f"));=>ab4f2c4fd. System.out.println(src.replace('3','f'));=>ab4f2c4fd. System.out.println(src.replaceAll("\\d","f"));=>abffafcffd. System.out.println(src.replaceAll("a","f"));=>fb43fc23d. System.out.println(src.replaceFirst("\\d,"f"));=>abf32c43d System.out.println(src.replaceFirst("4","h"));=>abh32c43d.
如何将字符串中的"\"替换成"\\":
String msgIn; String msgOut; msgOut=msgIn.replaceAll("\\\\","\\\\\\\\");
原因:
'\'在java中是一个转义字符,所以需要用两个代表一个。例如System.out.println( "\\" ) ;只打印出一个"\"。但是'\'也是正则表达式中的转义字符(replaceAll 的参数就是正则表达式),需要用两个代表一个。所以:\\\\被java转换成\\,\\又被正则表达式转换成\。
将"\"替换成"/"
String sqlPath="C:\Windows.old\Users";
sqlPath=sqlPath.replace("\\", "/");