文件传输原理

文件上传和下载

在web应用中,文件上传和下载功能是非常常用的功能。

1. 准备工作

对于文件上传,浏览器在上传过程中是将文件以流车形式提交到服务器端的。
一般选择采用Apache的开源工具common-fileupload这个文件上传组件。
common-fileupload是依赖于common-io这个包的,所以还需要下载这个包。

2. 使用类介绍

【文件上传的注意事项】

  1. 为保证服务器安全,上传文件应该放在外界无法直接访问的目录下,比如放在WEB-INF目录下。
  2. 为防止文件覆盖的现象发生,要为上传文件产生一个唯一的文件名。【解决问题方案:+时间戳,+UUID,+MD5加密,+位运算算法】
  3. 要限制上传文件的最大值。
  4. 可以限制上传文件的类型,在收到上传文件名时,判断后缀名是否合法。
    【需要用到的类详解】
    ServletFileUpload负责处理上传的文件数据,并将表单中每个输入项封装层一个FileItem对象,在使用ServletFileUpload对象解析请求时需要DiskFileFactory对象。所以,我们需要在进行解析工作前构造好DiskFileItemFactory对象,通过ServletFileUpload对象的构造方法或setFileItemFactory()方法设置ServletFileUpload对象的FileItemFactory属性。
    表单中如果包含一个文件上传项的话,这个表单的entype属性必须设置为multipart/form-data
  <%--通过表单上传文件
      get;  上传文件大小有限制
      post; 上传文件大小没有限制
  --%>
  <form action="" enctype="multipart/form-data" method="post">
    上传用户:<input type="text" name="username"><br/>
    <p><input type="file" name="file1"></p>
    <p><input type="file" name="file2"></p>
    <p><input type="submit"> | <input type="reset"></p>
  </form>

Servlet doGet方法

public class FileServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException{

        // 判断上传的表单是普通表单还是带文件的表单
        if (ServletFileUpload.isMultipartContent(request)){
            return; // 终止方法运行,说明这是一个普通表单,直接返回
        }

        // 创建上传文件的保存路径,建议在WEB-INF路径下,安全,用户无法直接访问上传的文件;
        String uploadPath = this.getServletContext().getRealPath("/WEB-INF/upload");
        File uploadfile = new File(uploadPath);
        if (!uploadfile.exists()){
            uploadfile.mkdir(); // 创建这个目录
        }

        // 缓存,临时文件
        // 临时路径,加入文件超过了预期的大小,我们就把它放在一个临时文件中,过几天自动删除,或者提醒用户转存为永久
        String tmpPath = this.getServletContext().getRealPath("/WEB-INF/tmp");
        File tmpFile = new File(tmpPath);
        if (!uploadfile.exists()){
            tmpFile.mkdir(); // 创建这个目录
        }

        // 处理上传的文件,一般都需要通过流来获取,我们可以使用request.getInputStream(),原生态的文件上传流获取,十分麻烦
        // 但是我们都建议使用Apache的文件上传组件来实现,common-fileupload,它需要依赖于commons-io组件;

        // 1. 创建DiskFileFactory对象,处理文件上传路径或者大小限制的;
        DiskFileItemFactory factory = new DiskFileItemFactory();
        // 通过这个工厂设置一个缓冲区,当上传的文件大于这个缓冲区的时候,将它放在临时文件中;
        factory.setSizeThreshold(1024*1024);    // 缓存区大小为1M
        factory.setRepository(tmpFile); // 临时目录的保存目录,需要一个File

        // 2. 获取ServletFileUpload
        ServletFileUpload upload = new ServletFileUpload(factory);

        // 监听文件上传进度;
        upload.setProgressListener(new ProgressListener() {
            @Override
//            pByteRead:已经读取到的文件大小
//            pContentLength:文件大小
            public void update(long pByteRead, long pContentLength, int i) {
                System.out.println("总大小:" + pContentLength + "已上传:" + pByteRead);
            }
        });

        // 处理乱码问题
        upload.setHeaderEncoding("UTF-8");
        // 设置总共能够上传文件的大小
        // 1024 = 1kb * 1024 = 1M * 10 = 10M
        upload.setSizeMax(1024*1024*10);

        // 3. 处理上传文件
        // 把前端请求解析,封装层一个FileItem对象,需要
        try {
            List<FileItem> fileItems = upload.parseRequest(request);
//            FileItem 每一个表单对象
            for (FileItem fileItem : fileItems) {
                // 判断上传的文件时普通的表单还是带文件的表单
                if (fileItem.isFormField()){
                    // getFiledName指的是前端表单控件的name
                    String name = fileItem.getFieldName();
                    String value = fileItem.getString("UTF-8");     // 处理乱码
                    System.out.println(name+":"+value);
                }else{  // 文件表单
//                    ==================处理文件=================
                    String uploadFileName = fileItem.getName();
                    if (uploadFileName.trim().equals("")||uploadFileName==null){
                        continue;
                    }
                    // 获取上传的文件名 /images/girl/n.png
                    String fileName = uploadFileName.substring(uploadFileName.lastIndexOf("/") + 1);
                    // 获取文件的后缀名
                    String fileExtName = uploadFileName.substring(uploadFileName.lastIndexOf(".") + 1);
                    // 可以使用UUID(唯一识别的通用码),保证文件名唯一;
                    // UUID.randomUUID(), 随机生成一个唯一识别的通用码;

                    // 网络传输中的东西,都需要序列化
                    // POJO ,实体类,如果想要在多个电脑上运行,需要传输(把对象都序列化)
                    // JNI = Java Native Interface
                    // implements Serializable:标记接口,JVM--> Java栈 本地方法栈 native --》C++

                    String uuidPath = UUID.randomUUID().toString();
//                    ==================存放地址=================
                    // 存在哪儿?uploadPath
                    // 文件真是存在的路径
                    String realPath = uploadPath + "/" +uuidPath;
                    // 给每个文件创建一个对应的文件夹
                    File realPathFile = new File(realPath);
                    if (!realPathFile.exists()){
                        realPathFile.mkdir();
                    }
//                    ==================文件传输=================
                    // 获得文件上传的流
                    InputStream inputStream = fileItem.getInputStream();
                    
                    // 创建一个文件输出流
                    // realPath = 真实的文件夹
                    // 差了一个文件;加上输出文件的名字+"/"+uuidFileName
                    FileOutputStream fos = new FileOutputStream(realPath + "/" + fileName);
                    
                    // 创建爱你一个缓冲区
                    byte[] buffer = new byte[1024*1024];
                    
                    // 判断是否读取完毕
                    int len = 0;
                    // 如果大于0说明还存在数据;
                    while ((len=inputStream.read(buffer))>0){
                        fos.write(buffer,0,len);
                    }
                    
                    // 关闭流
                    fos.close();
                    inputStream.close();
                    
                    msg = "文件上传成功";
                    fileItem.delete();  // 上传成功,清除临时文件
                }
                return msg;
            }
        } catch (FileUploadException e) {
            e.printStackTrace();
        }
    }

    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doGet(request, response);
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值