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 相关文章推荐
SpringCloud Alibaba 基本开发框架搭建过程
Jun 13 Java/Android
Java数据结构之链表相关知识总结
Jun 18 Java/Android
Sleuth+logback 设置traceid 及自定义信息方式
Jul 26 Java/Android
关于Spring配置文件加载方式变化引发的异常详解
Jan 18 Java/Android
java executor包参数处理功能 
Feb 15 Java/Android
详解Spring Security中的HttpBasic登录验证模式
Mar 17 Java/Android
springboot+zookeeper实现分布式锁
Mar 21 Java/Android
教你在 Java 中实现 Dijkstra 最短路算法的方法
Apr 08 Java/Android
详解Flutter网络请求Dio库的使用及封装
Apr 14 Java/Android
Java数组详细介绍及相关工具类
Apr 14 Java/Android
java高级用法JNA强大的Memory和Pointer
Apr 19 Java/Android
SpringBoot使用AOP实现统计全局接口访问次数详解
Jun 16 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数组函数序列之rsort() - 对数组的元素值进行降序排序
2011/11/02 PHP
不错的新闻标题颜色效果
2006/12/10 Javascript
Open and Print a Word Document
2007/06/15 Javascript
js 返回时间戳所对应的具体时间
2010/07/20 Javascript
实现变速回到顶部的JavaScript代码
2011/05/09 Javascript
js查错流程归纳
2012/05/04 Javascript
JavaScript高级程序设计(第3版)学习笔记12 js正则表达式
2012/10/11 Javascript
Jquery绑定事件(bind和live的区别介绍)
2013/08/23 Javascript
jQuery回调函数的定义及用法实例
2014/12/23 Javascript
Javascript中3个需要注意的运算符
2015/04/02 Javascript
vue 注册组件的使用详解
2018/05/05 Javascript
基于vue-element组件实现音乐播放器功能
2018/05/06 Javascript
Vue 组件修改根实例的数据的方法
2019/04/02 Javascript
JavaScript 链表定义与使用方法示例
2020/04/28 Javascript
Python中字符串的常见操作技巧总结
2016/07/28 Python
对pandas通过索引提取dataframe的行方法详解
2019/02/01 Python
python内存动态分配过程详解
2019/07/15 Python
python代码实现逻辑回归logistic原理
2019/08/07 Python
wxpython布局的实现方法
2019/11/01 Python
在Python中使用turtle绘制多个同心圆示例
2019/11/23 Python
基于h5py的使用及数据封装代码
2019/12/26 Python
python框架Django实战商城项目之工程搭建过程图文详解
2020/03/09 Python
安装并免费使用Pycharm专业版(学生/教师)
2020/09/24 Python
python开发一个解析protobuf文件的简单编译器
2020/11/17 Python
CSS3实现简易版的刮刮乐效果
2016/09/27 HTML / CSS
使用Filters滤镜弥补CSS3的跨浏览器问题以及兼容低版本IE
2013/01/23 HTML / CSS
捷克领先的户外服装及配件市场零售商:ALPINE PRO
2018/01/09 全球购物
惊艳的手工时装首饰:Migonne Gavigan
2018/02/23 全球购物
印度排名第一的蛋糕、鲜花和礼品送货:Winni
2019/08/02 全球购物
经典婚礼主持开场白
2014/03/13 职场文书
个人查摆剖析材料
2014/10/16 职场文书
实习班主任自我评价
2015/03/11 职场文书
社区低保工作总结2015
2015/07/23 职场文书
遗嘱格式范本
2015/08/07 职场文书
如何使用php生成zip压缩包
2021/04/21 PHP
Python获取指定日期是"星期几"的6种方法
2022/03/13 Python