java 文件上傳和下載

轉(zhuǎn)自 :https://www.cnblogs.com/xdp-gacl/p/4200090.html
在Web應(yīng)用系統(tǒng)開發(fā)中,文件上傳和下載功能是非常常用的功能咒吐,今天來講一下JavaWeb中的文件上傳和下載功能的實(shí)現(xiàn)。

??????對(duì)于文件上傳酪耕,瀏覽器在上傳的過程中是將文件以流的形式提交到服務(wù)器端的呜象,如果直接使用Servlet獲取上傳文件的輸入流然后再解析里面的請(qǐng)求參數(shù)是比較麻煩,所以一般選擇采用apache的開源工具common-fileupload這個(gè)文件上傳組件晚岭。這個(gè)common-fileupload上傳組件的jar包可以去apache官網(wǎng)上面下載鸥印,也可以在struts的lib文件夾下面找到,struts上傳的功能就是基于這個(gè)實(shí)現(xiàn)的。common-fileupload是依賴于common-io這個(gè)包的库说,所以還需要下載這個(gè)包狂鞋。

一、開發(fā)環(huán)境搭建

創(chuàng)建一個(gè)FileUploadAndDownLoad項(xiàng)目潜的,加入Apache的commons-fileupload文件上傳組件的相關(guān)Jar包骚揍,如下圖所示:

image

二、實(shí)現(xiàn)文件上傳

2.1啰挪、文件上傳頁面和消息提示頁面

upload.jsp頁面的代碼如下:

<%@ page language="java" pageEncoding="UTF-8"%>
<!DOCTYPE HTML>
<html>
  <head>
    <title>文件上傳</title>
  </head>
  
  <body>
    <form action="${pageContext.request.contextPath}/servlet/UploadHandleServlet" enctype="multipart/form-data" method="post">
        上傳用戶:<input type="text" name="username"><br/>
        上傳文件1:<input type="file" name="file1"><br/>
        上傳文件2:<input type="file" name="file2"><br/>
        <input type="submit" value="提交">
    </form>
  </body>
</html>

message.jsp的代碼如下:

<%@ page language="java" pageEncoding="UTF-8"%>
<!DOCTYPE HTML>
<html>
  <head>
    <title>消息提示</title>
  </head>
  
  <body>
        ${message}
  </body>
</html>

2.2信不、處理文件上傳的Servlet

UploadHandleServlet的代碼如下:

package me.gacl.web.controller;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.List;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;

public class UploadHandleServlet extends HttpServlet {

    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
                //得到上傳文件的保存目錄,將上傳的文件存放于WEB-INF目錄下亡呵,不允許外界直接訪問抽活,保證上傳文件的安全
                String savePath = this.getServletContext().getRealPath("/WEB-INF/upload");
                File file = new File(savePath);
                //判斷上傳文件的保存目錄是否存在
                if (!file.exists() && !file.isDirectory()) {
                    System.out.println(savePath+"目錄不存在,需要?jiǎng)?chuàng)建");
                    //創(chuàng)建目錄
                    file.mkdir();
                }
                //消息提示
                String message = "";
                try{
                    //使用Apache文件上傳組件處理文件上傳步驟:
                    //1锰什、創(chuàng)建一個(gè)DiskFileItemFactory工廠
                    DiskFileItemFactory factory = new DiskFileItemFactory();
                    //2酌壕、創(chuàng)建一個(gè)文件上傳解析器
                    ServletFileUpload upload = new ServletFileUpload(factory);
                     //解決上傳文件名的中文亂碼
                    upload.setHeaderEncoding("UTF-8"); 
                    //3、判斷提交上來的數(shù)據(jù)是否是上傳表單的數(shù)據(jù)
                    if(!ServletFileUpload.isMultipartContent(request)){
                        //按照傳統(tǒng)方式獲取數(shù)據(jù)
                        return;
                    }
                    //4歇由、使用ServletFileUpload解析器解析上傳數(shù)據(jù)卵牍,解析結(jié)果返回的是一個(gè)List<FileItem>集合,每一個(gè)FileItem對(duì)應(yīng)一個(gè)Form表單的輸入項(xiàng)
                    List<FileItem> list = upload.parseRequest(request);
                    for(FileItem item : list){
                        //如果fileitem中封裝的是普通輸入項(xiàng)的數(shù)據(jù)
                        if(item.isFormField()){
                            String name = item.getFieldName();
                            //解決普通輸入項(xiàng)的數(shù)據(jù)的中文亂碼問題
                            String value = item.getString("UTF-8");
                            //value = new String(value.getBytes("iso8859-1"),"UTF-8");
                            System.out.println(name + "=" + value);
                        }else{//如果fileitem中封裝的是上傳文件
                            //得到上傳的文件名稱沦泌,
                            String filename = item.getName();
                            System.out.println(filename);
                            if(filename==null || filename.trim().equals("")){
                                continue;
                            }
                            //注意:不同的瀏覽器提交的文件名是不一樣的糊昙,有些瀏覽器提交上來的文件名是帶有路徑的,如:  c:\a\b\1.txt谢谦,而有些只是單純的文件名释牺,如:1.txt
                            //處理獲取到的上傳文件的文件名的路徑部分,只保留文件名部分
                            filename = filename.substring(filename.lastIndexOf("\\")+1);
                            //獲取item中的上傳文件的輸入流
                            InputStream in = item.getInputStream();
                            //創(chuàng)建一個(gè)文件輸出流
                            FileOutputStream out = new FileOutputStream(savePath + "\\" + filename);
                            //創(chuàng)建一個(gè)緩沖區(qū)
                            byte buffer[] = new byte[1024];
                            //判斷輸入流中的數(shù)據(jù)是否已經(jīng)讀完的標(biāo)識(shí)
                            int len = 0;
                            //循環(huán)將輸入流讀入到緩沖區(qū)當(dāng)中回挽,(len=in.read(buffer))>0就表示in里面還有數(shù)據(jù)
                            while((len=in.read(buffer))>0){
                                //使用FileOutputStream輸出流將緩沖區(qū)的數(shù)據(jù)寫入到指定的目錄(savePath + "\\" + filename)當(dāng)中
                                out.write(buffer, 0, len);
                            }
                            //關(guān)閉輸入流
                            in.close();
                            //關(guān)閉輸出流
                            out.close();
                            //刪除處理文件上傳時(shí)生成的臨時(shí)文件
                            item.delete();
                            message = "文件上傳成功没咙!";
                        }
                    }
                }catch (Exception e) {
                    message= "文件上傳失敗千劈!";
                    e.printStackTrace();
                    
                }
                request.setAttribute("message",message);
                request.getRequestDispatcher("/message.jsp").forward(request, response);
    }

    public void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        doGet(request, response);
    }
}

在Web.xml文件中注冊(cè)UploadHandleServlet

<servlet>
    <servlet-name>UploadHandleServlet</servlet-name>
    <servlet-class>me.gacl.web.controller.UploadHandleServlet</servlet-class>
</servlet>

<servlet-mapping>
    <servlet-name>UploadHandleServlet</servlet-name>
    <url-pattern>/servlet/UploadHandleServlet</url-pattern>
</servlet-mapping>

運(yùn)行效果如下:

image

文件上傳成功之后祭刚,上傳的文件保存在了WEB-INF目錄下的upload目錄,如下圖所示:  
image

2.3墙牌、文件上傳的細(xì)節(jié)

上述的代碼雖然可以成功將文件上傳到服務(wù)器上面的指定目錄當(dāng)中涡驮,但是文件上傳功能有許多需要注意的小細(xì)節(jié)問題,以下列出的幾點(diǎn)需要特別注意的:
  1喜滨、為保證服務(wù)器安全捉捅,上傳文件應(yīng)該放在外界無法直接訪問的目錄下,比如放于WEB-INF目錄下虽风。
  2棒口、為防止文件覆蓋的現(xiàn)象發(fā)生寄月,要為上傳文件產(chǎn)生一個(gè)唯一的文件名。
  3无牵、為防止一個(gè)目錄下面出現(xiàn)太多文件剥懒,要使用hash算法打散存儲(chǔ)。
  4合敦、要限制上傳文件的最大值初橘。
  5、要限制上傳文件的類型充岛,在收到上傳文件名時(shí)保檐,判斷后綴名是否合法。
  針對(duì)上述提出的5點(diǎn)細(xì)節(jié)問題崔梗,我們來改進(jìn)一下UploadHandleServlet夜只,改進(jìn)后的代碼如下:

package me.gacl.web.controller;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.List;
import java.util.UUID;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileUploadBase;
import org.apache.commons.fileupload.ProgressListener;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;

/**
* @ClassName: UploadHandleServlet
* @Description: TODO(這里用一句話描述這個(gè)類的作用)
* @author: 孤傲蒼狼
* @date: 2015-1-3 下午11:35:50
*
*/ 
public class UploadHandleServlet extends HttpServlet {

    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
                //得到上傳文件的保存目錄,將上傳的文件存放于WEB-INF目錄下蒜魄,不允許外界直接訪問扔亥,保證上傳文件的安全
                String savePath = this.getServletContext().getRealPath("/WEB-INF/upload");
                //上傳時(shí)生成的臨時(shí)文件保存目錄
                String tempPath = this.getServletContext().getRealPath("/WEB-INF/temp");
                File tmpFile = new File(tempPath);
                if (!tmpFile.exists()) {
                    //創(chuàng)建臨時(shí)目錄
                    tmpFile.mkdir();
                }
                
                //消息提示
                String message = "";
                try{
                    //使用Apache文件上傳組件處理文件上傳步驟:
                    //1、創(chuàng)建一個(gè)DiskFileItemFactory工廠
                    DiskFileItemFactory factory = new DiskFileItemFactory();
                    //設(shè)置工廠的緩沖區(qū)的大小谈为,當(dāng)上傳的文件大小超過緩沖區(qū)的大小時(shí)旅挤,就會(huì)生成一個(gè)臨時(shí)文件存放到指定的臨時(shí)目錄當(dāng)中。
                    factory.setSizeThreshold(1024*100);//設(shè)置緩沖區(qū)的大小為100KB伞鲫,如果不指定粘茄,那么緩沖區(qū)的大小默認(rèn)是10KB
                    //設(shè)置上傳時(shí)生成的臨時(shí)文件的保存目錄
                    factory.setRepository(tmpFile);
                    //2、創(chuàng)建一個(gè)文件上傳解析器
                    ServletFileUpload upload = new ServletFileUpload(factory);
                    //監(jiān)聽文件上傳進(jìn)度
                    upload.setProgressListener(new ProgressListener(){
                        public void update(long pBytesRead, long pContentLength, int arg2) {
                            System.out.println("文件大小為:" + pContentLength + ",當(dāng)前已處理:" + pBytesRead);
                            /**
                             * 文件大小為:14608,當(dāng)前已處理:4096
                                文件大小為:14608,當(dāng)前已處理:7367
                                文件大小為:14608,當(dāng)前已處理:11419
                                文件大小為:14608,當(dāng)前已處理:14608
                             */
                        }
                    });
                     //解決上傳文件名的中文亂碼
                    upload.setHeaderEncoding("UTF-8"); 
                    //3秕脓、判斷提交上來的數(shù)據(jù)是否是上傳表單的數(shù)據(jù)
                    if(!ServletFileUpload.isMultipartContent(request)){
                        //按照傳統(tǒng)方式獲取數(shù)據(jù)
                        return;
                    }
                    
                    //設(shè)置上傳單個(gè)文件的大小的最大值柒瓣,目前是設(shè)置為1024*1024字節(jié),也就是1MB
                    upload.setFileSizeMax(1024*1024);
                    //設(shè)置上傳文件總量的最大值吠架,最大值=同時(shí)上傳的多個(gè)文件的大小的最大值的和芙贫,目前設(shè)置為10MB
                    upload.setSizeMax(1024*1024*10);
                    //4、使用ServletFileUpload解析器解析上傳數(shù)據(jù)傍药,解析結(jié)果返回的是一個(gè)List<FileItem>集合磺平,每一個(gè)FileItem對(duì)應(yīng)一個(gè)Form表單的輸入項(xiàng)
                    List<FileItem> list = upload.parseRequest(request);
                    for(FileItem item : list){
                        //如果fileitem中封裝的是普通輸入項(xiàng)的數(shù)據(jù)
                        if(item.isFormField()){
                            String name = item.getFieldName();
                            //解決普通輸入項(xiàng)的數(shù)據(jù)的中文亂碼問題
                            String value = item.getString("UTF-8");
                            //value = new String(value.getBytes("iso8859-1"),"UTF-8");
                            System.out.println(name + "=" + value);
                        }else{//如果fileitem中封裝的是上傳文件
                            //得到上傳的文件名稱,
                            String filename = item.getName();
                            System.out.println(filename);
                            if(filename==null || filename.trim().equals("")){
                                continue;
                            }
                            //注意:不同的瀏覽器提交的文件名是不一樣的怔檩,有些瀏覽器提交上來的文件名是帶有路徑的褪秀,如:  c:\a\b\1.txt,而有些只是單純的文件名薛训,如:1.txt
                            //處理獲取到的上傳文件的文件名的路徑部分,只保留文件名部分
                            filename = filename.substring(filename.lastIndexOf("\\")+1);
                            //得到上傳文件的擴(kuò)展名
                            String fileExtName = filename.substring(filename.lastIndexOf(".")+1);
                            //如果需要限制上傳的文件類型仑氛,那么可以通過文件的擴(kuò)展名來判斷上傳的文件類型是否合法
                            System.out.println("上傳的文件的擴(kuò)展名是:"+fileExtName);
                            //獲取item中的上傳文件的輸入流
                            InputStream in = item.getInputStream();
                            //得到文件保存的名稱
                            String saveFilename = makeFileName(filename);
                            //得到文件的保存目錄
                            String realSavePath = makePath(saveFilename, savePath);
                            //創(chuàng)建一個(gè)文件輸出流
                            FileOutputStream out = new FileOutputStream(realSavePath + "\\" + saveFilename);
                            //創(chuàng)建一個(gè)緩沖區(qū)
                            byte buffer[] = new byte[1024];
                            //判斷輸入流中的數(shù)據(jù)是否已經(jīng)讀完的標(biāo)識(shí)
                            int len = 0;
                            //循環(huán)將輸入流讀入到緩沖區(qū)當(dāng)中乙埃,(len=in.read(buffer))>0就表示in里面還有數(shù)據(jù)
                            while((len=in.read(buffer))>0){
                                //使用FileOutputStream輸出流將緩沖區(qū)的數(shù)據(jù)寫入到指定的目錄(savePath + "\\" + filename)當(dāng)中
                                out.write(buffer, 0, len);
                            }
                            //關(guān)閉輸入流
                            in.close();
                            //關(guān)閉輸出流
                            out.close();
                            //刪除處理文件上傳時(shí)生成的臨時(shí)文件
                            //item.delete();
                            message = "文件上傳成功闸英!";
                        }
                    }
                }catch (FileUploadBase.FileSizeLimitExceededException e) {
                    e.printStackTrace();
                    request.setAttribute("message", "單個(gè)文件超出最大值!=橥唷甫何!");
                    request.getRequestDispatcher("/message.jsp").forward(request, response);
                    return;
                }catch (FileUploadBase.SizeLimitExceededException e) {
                    e.printStackTrace();
                    request.setAttribute("message", "上傳文件的總的大小超出限制的最大值!S錾 辙喂!");
                    request.getRequestDispatcher("/message.jsp").forward(request, response);
                    return;
                }catch (Exception e) {
                    message= "文件上傳失敗鸠珠!";
                    e.printStackTrace();
                }
                request.setAttribute("message",message);
                request.getRequestDispatcher("/message.jsp").forward(request, response);
    }
    
    /**
    * @Method: makeFileName
    * @Description: 生成上傳文件的文件名巍耗,文件名以:uuid+"_"+文件的原始名稱
    * @Anthor:孤傲蒼狼
    * @param filename 文件的原始名稱
    * @return uuid+"_"+文件的原始名稱
    */ 
    private String makeFileName(String filename){  //2.jpg
        //為防止文件覆蓋的現(xiàn)象發(fā)生,要為上傳文件產(chǎn)生一個(gè)唯一的文件名
        return UUID.randomUUID().toString() + "_" + filename;
    }
    
    /**
     * 為防止一個(gè)目錄下面出現(xiàn)太多文件渐排,要使用hash算法打散存儲(chǔ)
    * @Method: makePath
    * @Description: 
    * @Anthor:孤傲蒼狼
    *
    * @param filename 文件名炬太,要根據(jù)文件名生成存儲(chǔ)目錄
    * @param savePath 文件存儲(chǔ)路徑
    * @return 新的存儲(chǔ)目錄
    */ 
    private String makePath(String filename,String savePath){
        //得到文件名的hashCode的值,得到的就是filename這個(gè)字符串對(duì)象在內(nèi)存中的地址
        int hashcode = filename.hashCode();
        int dir1 = hashcode&0xf;  //0--15
        int dir2 = (hashcode&0xf0)>>4;  //0-15
        //構(gòu)造新的保存目錄
        String dir = savePath + "\\" + dir1 + "\\" + dir2;  //upload\2\3  upload\3\5
        //File既可以代表文件也可以代表目錄
        File file = new File(dir);
        //如果目錄不存在
        if(!file.exists()){
            //創(chuàng)建目錄
            file.mkdirs();
        }
        return dir;
    }

    public void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        doGet(request, response);
    }
}

針對(duì)上述提出的5點(diǎn)小細(xì)節(jié)問題進(jìn)行改進(jìn)之后驯耻,我們的文件上傳功能就算是做得比較完善了亲族。

三、文件下載

3.1可缚、列出提供下載的文件資源

我們要將Web應(yīng)用系統(tǒng)中的文件資源提供給用戶進(jìn)行下載霎迫,首先我們要有一個(gè)頁面列出上傳文件目錄下的所有文件,當(dāng)用戶點(diǎn)擊文件下載超鏈接時(shí)就進(jìn)行下載操作帘靡,編寫一個(gè)ListFileServlet女气,用于列出Web應(yīng)用系統(tǒng)中所有下載文件。

ListFileServlet的代碼如下:

package me.gacl.web.controller;

import java.io.File;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
* @ClassName: ListFileServlet
* @Description: 列出Web系統(tǒng)中所有下載文件
* @author: 孤傲蒼狼
* @date: 2015-1-4 下午9:54:40
*
*/ 
public class ListFileServlet extends HttpServlet {

    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        //獲取上傳文件的目錄
        String uploadFilePath = this.getServletContext().getRealPath("/WEB-INF/upload");
        //存儲(chǔ)要下載的文件名
        Map<String,String> fileNameMap = new HashMap<String,String>();
        //遞歸遍歷filepath目錄下的所有文件和目錄测柠,將文件的文件名存儲(chǔ)到map集合中
        listfile(new File(uploadFilePath),fileNameMap);//File既可以代表一個(gè)文件也可以代表一個(gè)目錄
        //將Map集合發(fā)送到listfile.jsp頁面進(jìn)行顯示
        request.setAttribute("fileNameMap", fileNameMap);
        request.getRequestDispatcher("/listfile.jsp").forward(request, response);
    }
    
    /**
    * @Method: listfile
    * @Description: 遞歸遍歷指定目錄下的所有文件
    * @Anthor:孤傲蒼狼
    * @param file 即代表一個(gè)文件炼鞠,也代表一個(gè)文件目錄
    * @param map 存儲(chǔ)文件名的Map集合
    */ 
    public void listfile(File file,Map<String,String> map){
        //如果file代表的不是一個(gè)文件,而是一個(gè)目錄
        if(!file.isFile()){
            //列出該目錄下的所有文件和目錄
            File files[] = file.listFiles();
            //遍歷files[]數(shù)組
            for(File f : files){
                //遞歸
                listfile(f,map);
            }
        }else{
            /**
             * 處理文件名轰胁,上傳后的文件是以u(píng)uid_文件名的形式去重新命名的谒主,去除文件名的uuid_部分
                file.getName().indexOf("_")檢索字符串中第一次出現(xiàn)"_"字符的位置,如果文件名類似于:9349249849-88343-8344_阿_凡_達(dá).avi
                那么file.getName().substring(file.getName().indexOf("_")+1)處理之后就可以得到阿_凡_達(dá).avi部分
             */
            String realName = file.getName().substring(file.getName().indexOf("_")+1);
            //file.getName()得到的是文件的原始名稱赃阀,這個(gè)名稱是唯一的霎肯,因此可以作為key,realName是處理過后的名稱榛斯,有可能會(huì)重復(fù)
            map.put(file.getName(), realName);
        }
    }
    
    public void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        doGet(request, response);
    }
}

????????這里簡(jiǎn)單說一下ListFileServlet中l(wèi)istfile方法观游,listfile方法是用來列出目錄下的所有文件的,listfile方法內(nèi)部用到了遞歸驮俗,在實(shí)際開發(fā)當(dāng)中懂缕,我們肯定會(huì)在數(shù)據(jù)庫創(chuàng)建一張表,里面會(huì)存儲(chǔ)上傳的文件名以及文件的具體存放目錄王凑,我們通過查詢表就可以知道文件的具體存放目錄搪柑,是不需要用到遞歸操作的聋丝,這個(gè)例子是因?yàn)闆]有使用數(shù)據(jù)庫存儲(chǔ)上傳的文件名和文件的具體存放位置,而上傳文件的存放位置又使用了散列算法打散存放工碾,所以需要用到遞歸弱睦,在遞歸時(shí),將獲取到的文件名存放到從外面?zhèn)鬟f到listfile方法里面的Map集合當(dāng)中渊额,這樣就可以保證所有的文件都存放在同一個(gè)Map集合當(dāng)中况木。
  在Web.xml文件中配置ListFileServlet:

<servlet>
     <servlet-name>ListFileServlet</servlet-name>
     <servlet-class>me.gacl.web.controller.ListFileServlet</servlet-class>
</servlet>
 
<servlet-mapping>
     <servlet-name>ListFileServlet</servlet-name>
    <url-pattern>/servlet/ListFileServlet</url-pattern>
</servlet-mapping>

展示下載文件的listfile.jsp頁面如下:

<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<!DOCTYPE HTML>
<html>
  <head>
    <title>下載文件顯示頁面</title>
  </head>
  
  <body>
      <!-- 遍歷Map集合 -->
    <c:forEach var="me" items="${fileNameMap}">
        <c:url value="/servlet/DownLoadServlet" var="downurl">
            <c:param name="filename" value="${me.key}"></c:param>
        </c:url>
        ${me.value}<a href="${downurl}">下載</a>
        <br/>
    </c:forEach>
  </body>
</html>

訪問ListFileServlet,就可以在listfile.jsp頁面中顯示提供給用戶下載的文件資源旬迹,如下圖所示:

image

3.2火惊、實(shí)現(xiàn)文件下載

編寫一個(gè)用于處理文件下載的Servlet,DownLoadServlet的代碼如下:

package me.gacl.web.controller;

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.URLEncoder;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

public class DownLoadServlet extends HttpServlet {

    
    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        //得到要下載的文件名
        String fileName = request.getParameter("filename");  //23239283-92489-阿凡達(dá).avi
        fileName = new String(fileName.getBytes("iso8859-1"),"UTF-8");
        //上傳的文件都是保存在/WEB-INF/upload目錄下的子目錄當(dāng)中
        String fileSaveRootPath=this.getServletContext().getRealPath("/WEB-INF/upload");
        //通過文件名找出文件的所在目錄
        String path = findFileSavePathByFileName(fileName,fileSaveRootPath);
        //得到要下載的文件
        File file = new File(path + "\\" + fileName);
        //如果文件不存在
        if(!file.exists()){
            request.setAttribute("message", "您要下載的資源已被刪除2杖ā矗晃!");
            request.getRequestDispatcher("/message.jsp").forward(request, response);
            return;
        }
        //處理文件名
        String realname = fileName.substring(fileName.indexOf("_")+1);
        //設(shè)置響應(yīng)頭,控制瀏覽器下載該文件
        response.setHeader("content-disposition", "attachment;filename=" + URLEncoder.encode(realname, "UTF-8"));
        //讀取要下載的文件宴倍,保存到文件輸入流
        FileInputStream in = new FileInputStream(path + "\\" + fileName);
        //創(chuàng)建輸出流
        OutputStream out = response.getOutputStream();
        //創(chuàng)建緩沖區(qū)
        byte buffer[] = new byte[1024];
        int len = 0;
        //循環(huán)將輸入流中的內(nèi)容讀取到緩沖區(qū)當(dāng)中
        while((len=in.read(buffer))>0){
            //輸出緩沖區(qū)的內(nèi)容到瀏覽器张症,實(shí)現(xiàn)文件下載
            out.write(buffer, 0, len);
        }
        //關(guān)閉文件輸入流
        in.close();
        //關(guān)閉輸出流
        out.close();
    }
    
    /**
    * @Method: findFileSavePathByFileName
    * @Description: 通過文件名和存儲(chǔ)上傳文件根目錄找出要下載的文件的所在路徑
    * @Anthor:孤傲蒼狼
    * @param filename 要下載的文件名
    * @param saveRootPath 上傳文件保存的根目錄,也就是/WEB-INF/upload目錄
    * @return 要下載的文件的存儲(chǔ)目錄
    */ 
    public String findFileSavePathByFileName(String filename,String saveRootPath){
        int hashcode = filename.hashCode();
        int dir1 = hashcode&0xf;  //0--15
        int dir2 = (hashcode&0xf0)>>4;  //0-15
        String dir = saveRootPath + "\\" + dir1 + "\\" + dir2;  //upload\2\3  upload\3\5
        File file = new File(dir);
        if(!file.exists()){
            //創(chuàng)建目錄
            file.mkdirs();
        }
        return dir;
    }
    
    public void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        doGet(request, response);
    }
}

在Web.xml文件中配置DownLoadServlet:

<servlet>
      <servlet-name>DownLoadServlet</servlet-name>
      <servlet-class>me.gacl.web.controller.DownLoadServlet</servlet-class>
</servlet>
 
<servlet-mapping>
      <servlet-name>DownLoadServlet</servlet-name>
      <url-pattern>/servlet/DownLoadServlet</url-pattern>
</servlet-mapping>

點(diǎn)擊【下載】超鏈接鸵贬,將請(qǐng)求提交到DownLoadServlet就行處理就可以實(shí)現(xiàn)文件下載了俗他,運(yùn)行效果如下圖所示:

image

從運(yùn)行結(jié)果可以看到,我們的文件下載功能已經(jīng)可以正常下載文件了阔逼。
關(guān)于JavaWeb中的文件上傳和下載功能的內(nèi)容就這么多兆衅。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市嗜浮,隨后出現(xiàn)的幾起案子羡亩,更是在濱河造成了極大的恐慌,老刑警劉巖危融,帶你破解...
    沈念sama閱讀 217,277評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件畏铆,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡吉殃,警方通過查閱死者的電腦和手機(jī)辞居,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來蛋勺,“玉大人瓦灶,你說我怎么就攤上這事”辏” “怎么了贼陶?”我有些...
    開封第一講書人閱讀 163,624評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我每界,道長(zhǎng)捅僵,這世上最難降的妖魔是什么家卖? 我笑而不...
    開封第一講書人閱讀 58,356評(píng)論 1 293
  • 正文 為了忘掉前任眨层,我火速辦了婚禮,結(jié)果婚禮上上荡,老公的妹妹穿的比我還像新娘趴樱。我一直安慰自己,他們只是感情好酪捡,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評(píng)論 6 392
  • 文/花漫 我一把揭開白布叁征。 她就那樣靜靜地躺著,像睡著了一般逛薇。 火紅的嫁衣襯著肌膚如雪捺疼。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,292評(píng)論 1 301
  • 那天永罚,我揣著相機(jī)與錄音啤呼,去河邊找鬼。 笑死呢袱,一個(gè)胖子當(dāng)著我的面吹牛官扣,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播羞福,決...
    沈念sama閱讀 40,135評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼惕蹄,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了治专?” 一聲冷哼從身側(cè)響起卖陵,我...
    開封第一講書人閱讀 38,992評(píng)論 0 275
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎张峰,沒想到半個(gè)月后泪蔫,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,429評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡挟炬,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評(píng)論 3 334
  • 正文 我和宋清朗相戀三年鸥滨,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片谤祖。...
    茶點(diǎn)故事閱讀 39,785評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡婿滓,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出粥喜,到底是詐尸還是另有隱情凸主,我是刑警寧澤,帶...
    沈念sama閱讀 35,492評(píng)論 5 345
  • 正文 年R本政府宣布额湘,位于F島的核電站卿吐,受9級(jí)特大地震影響旁舰,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜嗡官,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評(píng)論 3 328
  • 文/蒙蒙 一箭窜、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧衍腥,春花似錦磺樱、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至尚骄,卻和暖如春块差,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背倔丈。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評(píng)論 1 269
  • 我被黑心中介騙來泰國打工憨闰, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人乃沙。 一個(gè)月前我還...
    沈念sama閱讀 47,891評(píng)論 2 370
  • 正文 我出身青樓起趾,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國和親警儒。 傳聞我的和親對(duì)象是個(gè)殘疾皇子训裆,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容