一聚教程网:一个值得你收藏的教程网站

热门教程

jsp 用户上传头像、上传图片、邮件上传附件代码

时间:2022-06-29 02:39:50 编辑:袖梨 来源:一聚教程网

jsp教程 用户上传头像、上传图片、邮件上传附件代码


2. 页面表单的实现
    文件上传表单和普通表单有两个区别

    1) 需要文件上传字段 

    2) form 表单的 enctype 属性需要指定为 multipart/form-data

3. 服务器端解析request
    在 Servlet 中通过 request.getInputStream 获得表单上传数据,会发现数据是分段发送的

由于自己写程序解析有难度,我们可以使用Apache 开发的开源组件Commons-fileupload

需要导入 jar 包Commons-fileupload 和Commons-io

4 . UploadServlet 中处理文件上传程序
// 1. 创建工厂类

DiskFileItemFactory factory = new DiskFileItemFactory();

// 2. 创建FileUpload对象

ServletFileUpload upload = new ServletFileUpload(factory);

// 3. 判断是否是上传表单

boolean b = upload.isMultipartContent(request);

if(!b) {

    // 不是文件上传

request.setAttribute("message", "对不起,不是文件上传表单!");

request.getRequestDispatcher("/message.jsp").forward(request, response);

    return;

}

// 是文件上传表单

// 4. 解析request,获得FileItem项

List fileitems = upload.parseRequest(request);

// 5. 遍历集合

for(FileItem item : fileitems) {

    // 判断是不是普通字段

if(item.isFormField()) {

    String name = item.getFieldName();

    String value = item.getString();

    // 手工的转换了

    value = new String(value.getBytes("iso-8859-1"),"utf-8");

    System.out.println(name + "=" + value);

} else {

    // 文件上传字段

    // 获得文件名

    String filename = item.getName();

    System.out.println(filename);

    filename = filename.substring(filename.lastIndexOf("")+1);

   

    System.out.println(filename);

    // 创建文件

    ServletContext context = getServletContext();

    String dir = context.getRealPath("WEN-INF/upload");

    File file = new File(dir, filename);

    file.createNewFile();

   

    // 获得流,读取数据写入文件

    InputStream in = item.getInputStream();

    FileOutputStream fos = new FileOutputStream(file);

   

    int len;

    byte[] buffer = new byte[1024];

    while((len=in.read(buffer))>0)

       fos.write(buffer,0,len);

    fos.close();

    in.close();

    item.delete();    // 删除临时文件

}

二、 文件上传处理细节
1. 中文乱码问题
    1) 文件名中文乱码问题,解决办法: 告诉文件上传组件以什么编码方式来解码文件名

       ServletUpload.setCharacterEncoding(“utf-8”);

       request. setCharacterEncoding(“utf-8”);

    

    2) 普通字段中文乱码问题

       fileitem.getString(“utf-8”);      

2.   临时文件
    对于大文件不能缓存在内存,需要缓存到硬盘,为了方便管理,我们需要设置临时文件存放目录

    // 设置临时文件的存放位置

factory.setRepository(new File("d:/temp"));

    文件上传完毕需要删除临时文件,否则会导致服务器存在两份上传文件

// 注意,需要先将流进行关闭,否则会导致临时文件无法删除

out.close();

in.close();

// 删除临时文件

fileitem.delete();

3. 文件存放目录
    1) 目录需要隐藏,禁止外界直接访问

    2) 文件名需要保证不重复

    3) 文件应该分目录存放

三、上传进度条
1. 实现进度监听
    需要实现对文件上传进度的监听,需要给FileUpload 对象添加 ProgressListener

    在upload方法中对与进度相关的数据进行处理

    upload.setProgressListener(new ProgressListener() {

    long num = 0;

    public void update(long bytesRead, long contentLength, int items) {

      

            long progress = bytesRead*100/contentLength;

           if(progress==num)

               return;

           num = progress;

           System.out.println("上传进度:" + progress + "%");

      

        //  request.getSession().setAttribute("progress", progress);

        }

    });

2. 在 jsp 页面显示进度
    实验: 

    1) 使用 iframe 发送请求, 请求一个Servlet, 在Servlet 中返回响应,发送自增的num

       此时会发现 iframe 会不停第想Servlet发送请求

    2) 点击文件上传按钮后,iframe立刻停止刷新,直至上传完毕页面跳转至新页面

    3)为了观察实验结果,将form 的 target 指定为 iframe, UploadServlet回送上传完毕的结果

    4) 出现上述问题的原因,浏览器不支持多线程同时访问服务器只能同时发送一个请求,

       这样的访问方式为同步访问

    5) 要在文件上传的同时在iframe中实现进度访问,就需要ie浏览器与服务器进行异步交互

       此时就需要 XMLHttpRequest 对象

       在网页特效中可以直接使用XMLHttpRequest 对象与服务器进行异步通信

       获得XmlHttpRequest 对象的方式有两种

       ie7以上版本

       var xhr = null;

       if(window.XMLHttpRequest)

           xhr = new XMLHttpRequest();

       ie7以下版本

       if(window.ActiveXObject)

           xhr = new ActiveXObject(“Microsoft.XMLHTTP”);

      

       获得对象后需要调用open方法输入请求地址

       注意请求方式, 地址的输入, 并且需要设置为true 指定异步访问该地址

       xhr.open(“get”,”/upload/servlet/UploadServlet”, false)

      

       // 调用send 方法发送请求,post方式需要发送消息体,get方式则不用直接传入null值

       xhr.send(null);

      

       // 访问 responseText 属性获得 Servlet 回送的数据

       document.write(xhr.responseText);

四、 api方法
1. DiskFileItemFactory 对象
    设置缓冲区大小,字节为单位,默认为10K,一般不用修改

    factory.setSizeThreshold(1000);

    设置临时文件存放目录

    factory.setRepository(file);

2. ServletFileUpload 对象
    判断是否为文件上传表单

    boolean b = upload.isMultipartContent(request);

   解析request对象

   List list = upload.parseRequest(request);

   设置上传文件的最大值

    setFileSizeMax(long fileSizeMax)

    设置上传文件总量的最大值

    setSizeMax(long sizeMax)

    设置编码格式

    setHeaderEncoding(java.lang.String encoding)

    注册进度监听器

    setProgressListener(ProgressListener pListener)

3. FileItem 对象
    获得表单字段的属性名

    item.getFieldName();

    获得普通字段的值

    item.getString(charsetName)

    获得文件上传字段的文件名

    item.getName()

    获得文件上传的流

    item.getInputStream()

  

 文件上传时需要注意的问题:

1.如何设置上传文件最大值,并实现超出最大值时给用户一个友好提示
 upload.setFileSizeMax(1024*10);  //设置最大值
 实现超出最大值时给用户一个友好提示:在程序中捕获FileUploadBase.FileSizeLimitExceededException
 只要程序抛出这个异常,代表用户上传的文件超出最大值
 
 
2.上传过程中的乱码问题
 2.1 普通输入项的乱码
  item.getString("码表 ")
 2.2 上传文件名的乱码
  ServletFileUpload.setHeaderEncoding("码表")
 
3.上传文件的安全性问题
 为防止用户直接上传文件,危害服务器安全,程序应禁止用户直接访问上传文件(即把上传文件保存在用户无法直接访问的目录)
 
4.防止文件覆盖(UUID)

5.文件打散存储(一个目录下面不能存超出1000个文件)
 用hash算法生成目录保存
 
6.设置监听器,监听文件上传进度

 upload.setProgressListener(new ProgressListener(){
   public void update(long arg0, long arg1, int arg2) {
    System.out.println("当前已上传" + arg0 + ",当前处理的文件总大小" + arg1);
   }
  });

7.临时文件的删除问题
 处理完每一个文件上传后,一定要记得调用Fileitem.delete方法,删除临时文件
 
8.限定上传文件类型
 判断上传文件后缀名

 

热门栏目