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 feign 首次调用100%失败的问题
Jun 23 Java/Android
Java中使用Filter过滤器的方法
Jun 28 Java/Android
Springboot配置suffix指定mvc视图的后缀方法
Jul 03 Java/Android
SpringBoot SpringEL表达式的使用
Jul 25 Java/Android
spring boot中nativeQuery的用法
Jul 26 Java/Android
Java使用Unsafe类的示例详解
Sep 25 Java/Android
java协程框架quasar和kotlin中的协程对比分析
Feb 24 Java/Android
解析探秘fescar分布式事务实现原理
Feb 28 Java/Android
Java字符缓冲流BufferedWriter
Apr 09 Java/Android
零基础学java之循环语句的使用
Apr 10 Java/Android
Java+swing实现抖音上的表白程序详解
Jun 25 Java/Android
利用Java连接Hadoop进行编程
Jun 28 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
人大复印资料处理程序_补充篇
2006/10/09 PHP
php面向对象全攻略 (一) 面向对象基础知识
2009/09/30 PHP
ThinkPHP实现静态缓存和动态缓存示例代码
2017/05/02 PHP
kindeditor 加入七牛云上传的实例讲解
2017/11/12 PHP
List Information About the Binary Files Used by an Application
2007/06/18 Javascript
jQuery模拟超链接点击效果代码
2013/04/21 Javascript
Jquery选中或取消radio示例
2013/09/29 Javascript
JSON+HTML实现国家省市联动选择效果
2014/05/18 Javascript
JavaScript中实现无缝滚动、分享到侧边栏实例代码
2016/04/06 Javascript
针对BootStrap中tabs控件的美化和完善(推荐)
2016/07/06 Javascript
AngularJS基础 ng-switch 指令简单示例
2016/08/03 Javascript
简单实现Vue的observer和watcher
2016/12/21 Javascript
js实现交通灯效果
2017/01/13 Javascript
基于JS实现仿百度百家主页的轮播图效果
2017/03/06 Javascript
vue的Virtual Dom实现snabbdom解密
2017/05/03 Javascript
微信小程序实现流程进度的图样式功能
2018/01/16 Javascript
详解JavaScript中操作符和表达式
2018/09/12 Javascript
详解在不使用ssr的情况下解决Vue单页面SEO问题
2018/11/08 Javascript
vue项目首屏打开速度慢的解决方法
2019/03/31 Javascript
Node.js HTTP服务器中的文件、图片上传的方法
2019/09/23 Javascript
JavaScript进阶(二)词法作用域与作用域链实例分析
2020/05/09 Javascript
python之import机制详解
2014/07/03 Python
Python使用cx_Oracle模块将oracle中数据导出到csv文件的方法
2015/05/16 Python
Python实现删除当前目录下除当前脚本以外的文件和文件夹实例
2015/07/27 Python
pandas带有重复索引操作方法
2018/06/08 Python
对python读写文件去重、RE、set的使用详解
2018/12/11 Python
解决在Python编辑器pycharm中程序run正常debug错误的问题
2019/01/17 Python
Python:__eq__和__str__函数的使用示例
2020/09/26 Python
CSS的background属性及CSS3的背景图片设置总结
2016/06/13 HTML / CSS
CSS实现半透明边框与多重边框的场景分析
2019/11/13 HTML / CSS
html5 http的轮询和Websocket原理
2018/10/19 HTML / CSS
机关保密承诺书
2014/06/03 职场文书
2016幼儿园教师节新闻稿
2015/11/25 职场文书
小学英语教学反思范文
2016/02/15 职场文书
Redis基于Bitmap实现用户签到功能
2021/06/20 Redis
Python创建SQL数据库流程逐步讲解
2022/09/23 Python