HttpClient实现文件上传功能


Posted in Java/Android onAugust 14, 2022

除了文件下载外,文件上传也是项目开发中经常用到的功能。此时需要用HttpClient类库的扩展包HttpMime中的MultipartEntity类,此类同样实现了HttpEntity接口。需要注意的是HttpClient通过POST来上传文件,而不是通过流的形式。

示例:使用HttpClient实现文件上传功能。

第一步:要求在服务器端使用fileupload组件接收客户端提交内容。新建一个JSP项目zghc,首先在WEB-INF/lib目录下加入commons-fileupload-1.2.2.jar和commons-io-2.4.jar两个jar包,然后再提供一个对用户提交数据(文本、文件)进行处理的Servlet,具体代码如下所示:

@WebServlet("/uploadServlet")
public class UploadServlet extends HttpServlet {    // 使用fileupload组件
    private static final long serialVersionUID = 1L;
    protected void doPost(HttpServletRequest request,
            HttpServletResponse response) throws ServletException, IOException {
        // 检测是不是文件上传的请求
        boolean isMultipart = ServletFileUpload.isMultipartContent(request);
        if (isMultipart) {// 属于上传文件的请求
            // 创建磁盘工厂,该类用来配置上传组件ServletFileUpload
            DiskFileItemFactory factory = new DiskFileItemFactory();
            factory.setSizeThreshold(10 * 1024 * 1024);
            factory.setRepository(new File("D:/")); // 设置存放临时文件的目录
            // 使用磁盘工厂实例化上传组件
            ServletFileUpload upload = new ServletFileUpload(factory);
            upload.setSizeMax(10 * 1024 * 1024); // 设置允许的最大上传尺寸
            upload.setHeaderEncoding("UTF-8");// 明确指定使用UTF-8编码
            PrintWriter out = response.getWriter();
            try {    // 获取客户端提交过来的所有请求参数
                List<FileItem> items = upload.parseRequest(request); // 解析
                for (FileItem item : items) {//对用户上传的所有文件进行遍历
                    if (item.isFormField()) {// 如果是表单字段
                        System.out.println(item.getFieldName()+"  "+
item.getString("UTF-8"));
                    } else {// 如果是文件
                        String path = request.getSession().getServletContext()
                                .getRealPath("/");// 站点根目录的路径
                        String name = item.getName().substring(
                                item.getName().lastIndexOf("/") + 1); // ---②
                        // 保存用户上传的文件到指定目录
                        item.write(new File(path, name)); 
                        System.out.println("上传完毕");
                        response.setCharacterEncoding("UTF-8");
                        out.println("上传完毕");
                    }
                }
            } catch (Exception e) {
                e.printStackTrace();
                out.println("文件超过规定大小");
            }
        }
    }
}

FileItemFactory类的setSizeThreshold()方法用来设置上传文件时用于临时存放文件的内存的大小,超出的部分将临时存放在硬盘,可以使用FileItemFactory类的 setRepository()方法设置临时文件的目录。

在调用item.write()方法写入数据到文件中时,如果文件的名称是中文,有可能会出现乱码;另外需要注意的是Windows系统中item.getName()方法的返回的值是带路径的。

提示:

如果使用tomcat6,采用传统的在web.xml方式中配置Servlet,编号②处的代码需要改为:item.getName().lastIndexOf("\")
第二步:提供一个JSP文件upload.jsp,对上面的Servlet进行测试。注意在此JSP文件中需要设置form表单的enctype的值为multipart/form-data。

<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
    <head>
        <title>My JSP 'index.jsp' starting page</title>
    </head>
    <body>
        <form action="uploadServlet" method="post" enctype="multipart/form-data">
            标题:<input type="text" name="name"/>
            文件:<input type="file" name="imgFile"/>
            <input type="submit" value="提交"/>
        </form>
    </body>
</html>

为了能上传文件,必须将表单的method属性设置为POST、enctype属性设置为multipart/form-data,只有这样,浏览器才会把用户选择文件的二进制数据发送给服务器。

第四步:将上面的JSP项目部署到tomcat中,然后在打开的upload.jsp页面中输入文件名,选中要上传的文件,单击【提交】按钮,然后在tomcat下的webapps\zghc目录下确实发现上传的文件,表明文件上传成功。

第五步:新建一个Android项目,将httpmime_XXXX.jar文件添加到当前项目的classpath路径下面,然后在项目中添加一个实现上传功能的工具类,具体代码如下所示:

public class HttpClientUtil {
    public static HttpClient httpClient = new DefaultHttpClient();
    public static String sendPost(String url,HashMap<String,String> map,File file){
        String result = null;
        HttpPost post = new HttpPost(url);// 创建HttpPost对象
        // 如果传递参数个数比较多的话可以对传递的参数进行封装
        MultipartEntity entity = new MultipartEntity();
        try {
            for (String key : map.keySet()) {    // 封装请求参数
StringBody value = new StringBody(map
                        .get(key), Charset.forName("UTF-8"));//避免传递汉字出现乱码
                entity.addPart(new FormBodyPart(key,value));
            }
            if(file != null)
                entity.addPart("myfile", new FileBody(file));
            post.setEntity(entity);// 设置请求参数
            synchronized (httpClient) {
                HttpResponse response = httpClient.execute(post);// 发送POST请求
                if (response.getStatusLine().getStatusCode() == 200){
                    HttpEntity resEntity = response.getEntity();
                    result = EntityUtils.toString(resEntity, "UTF-8");
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
        return result;
    }
}

第六步:为了提高用户体验,我们提供一个线程类来实现文件文件上传的功能,具体代码如下所示:

public class UploadThread extends Thread {
    private String url;
    private HashMap<String, String> map;
    private File file;
    private Handler handler;
    public UploadThread(String url, HashMap<String, String> map, File file,
            Handler handler) {
        this.url = url;
        this.map = map;
        this.file = file;
        this.handler = handler;
    }
    @Override
    public void run() {
        String result = HttpClientUtil.sendPost(url, map, file);  //具体上传代码
        System.out.println("aaaaa" + result);
        if ("上传完毕".equals(result.trim())) {
            handler.sendEmptyMessage(1);
        } else {
            handler.sendEmptyMessage(0);
        }
    }
}

第七步:在主布局文件中提供一个onClick属性值为upload的Button,然后修改MainAcvitity类的代码如下所示:

public class MainActivity extends Activity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
    private Handler handler = new Handler() {
        public void handleMessage(Message msg) {
            if (msg.what == 1) {
                Toast.makeText(MainActivity.this, "提交数据成功", 1).show();
            } else {
                Toast.makeText(MainActivity.this, "提交数据失败", 1).show();
            }
        }
    };
    public void upload(View view) {
        HashMap<String, String> map = new HashMap<String, String>();
        map.put("username", "music");
        File sdPath = Environment.getExternalStorageDirectory();
        File file = new File(sdPath + "/zbjbxf.mp3");
        String url = "http://10.0.2.2:8080/zghc/uploadServlet";
        UploadThread thread = new UploadThread(url, map, file, handler);
        thread.start();
    }
}

在保证第一步创建的JSP项目已经部署到tomcat中且tomcat已经启动的前提下,运行本程序,发现当点击客户端主界面中的Button按钮后,在tomcat下的webapps/zghc目录下确实看到了刚才上传的文件。

大家可以将前面章节中学到Android中制作文件管理器的知识和本示例程序结合起来,实现一个能够通过图形化方式选择文件的文件上传软件。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Java/Android 相关文章推荐
Java 将PPT幻灯片转为HTML文件的实现思路
Jun 11 Java/Android
ConstraintValidator类如何实现自定义注解校验前端传参
Jun 18 Java/Android
Java基于字符界面的简易收银台
Jun 26 Java/Android
JavaWeb Servlet实现网页登录功能
Jul 04 Java/Android
浅谈spring boot使用thymeleaf版本的问题
Aug 04 Java/Android
java如何实现socket连接方法封装
Sep 25 Java/Android
关于MybatisPlus配置双数据库驱动连接数据库问题
Jan 22 Java/Android
剑指Offer之Java算法习题精讲二叉树的构造和遍历
Mar 21 Java/Android
Android Flutter实现图片滑动切换效果
Apr 07 Java/Android
Qt数据库应用之实现图片转pdf
Jun 01 Java/Android
Spring Cloud OpenFeign模版化客户端
Jun 25 Java/Android
Android实现获取短信验证码并自动填充
May 21 Java/Android
SpringBoot Http远程调用的方法
Aug 14 #Java/Android
基于Android10渲染Surface的创建过程
Aug 14 #Java/Android
OpenFeign实现远程调用
Aug 14 #Java/Android
httpclient调用远程接口的方法
Aug 14 #Java/Android
Java Spring读取和存储详细操作
Aug 05 #Java/Android
AndroidStudio图片压缩工具ImgCompressPlugin使用实例
Aug 05 #Java/Android
Java代码规范与质量检测插件SonarLint的使用
Aug 05 #Java/Android
You might like
php登陆页的密码处理方式分享
2013/10/14 PHP
php通过修改header强制图片下载的方法
2015/03/24 PHP
Zend Framework动作助手Json用法实例分析
2016/03/05 PHP
PHP正则表达式匹配替换与分割功能实例浅析
2017/02/04 PHP
PHP自动补全表单的两种方法
2017/03/06 PHP
xss文件页面内容读取(解决)
2010/11/28 Javascript
JavaScript中的apply()方法和call()方法使用介绍
2012/07/25 Javascript
JS解决ie6下png透明的方法实例
2013/08/02 Javascript
javascript轻松实现当鼠标移开时已弹出子菜单自动消失
2013/12/29 Javascript
js获取客户端网卡的IP地址、MAC地址
2014/03/26 Javascript
完美兼容各大浏览器的jQuery仿新浪图文淡入淡出间歇滚动特效
2014/11/12 Javascript
Javascript实现多彩雪花从天降散落效果的方法
2015/02/02 Javascript
JS基于FileSystemObject创建一个指定路径的TXT文本文件
2015/08/05 Javascript
jQuery计算文本框字数及限制文本框字数的方法
2016/03/01 Javascript
jquery对象访问是什么及使用方法介绍
2016/05/03 Javascript
jQuery 全选 全不选 事件绑定的实现代码
2017/01/23 Javascript
JS实现的简单四则运算计算器功能示例
2017/09/27 Javascript
Vue.js通用应用框架-Nuxt.js的上手教程
2017/12/25 Javascript
vue树形结构获取键值的方法示例
2018/06/21 Javascript
JS删除String里某个字符的方法
2021/01/06 Javascript
纯Python开发的nosql数据库CodernityDB介绍和使用实例
2014/10/23 Python
详解Python编程中基本的数学计算使用
2016/02/04 Python
Python读取图片属性信息的实现方法
2016/09/11 Python
Python 获取numpy.array索引值的实例
2019/12/06 Python
Python+OpenCV实现旋转文本校正方式
2020/01/09 Python
Python3+Selenium+Chrome实现自动填写WPS表单
2020/02/12 Python
html5 Canvas画图教程(6)—canvas里画曲线之arcTo方法
2013/01/09 HTML / CSS
Canvas图片分割效果的实现
2019/07/29 HTML / CSS
什么是动态端口(Dynamic Ports)?动态端口的范围是多少?
2014/12/12 面试题
测试驱动开发的主要步骤是什么
2014/12/10 面试题
生物技术毕业生自荐信
2013/10/23 职场文书
毕业证丢失证明
2014/01/15 职场文书
老师给学生的表扬信
2014/01/17 职场文书
初三学生个人自我评定
2014/04/06 职场文书
商务英语专业求职信
2014/06/26 职场文书
门球健将观后感
2015/06/16 职场文书