1. 项目中多处用到文件批量上传功能,今天正好解决了此问题,在此写出来,以便日后借鉴。
  2. 首先,以下架构下的批量文件上传可能会失败或者不会成功:
  3.   1.android客户端+springMVC服务端:服务端采用org.springframework.web.multipart.MultipartHttpServletRequest作为批量上传接收类,这种搭配下的批量文件上传会失败,最终服务端只会接受到一个文件,即只会接受到第一个文件。可能因为MultipartHttpServletRequestservlet原本的HttpServletRequest类进行封装,导致批量上传有问题。
  4.   2.android客户端+strutsMVC服务端:这种搭配下的多文件上传,本人没有亲自尝试,不做评论。如果有网友采用此种方式上传失败的,原因可能同1
  1. 接下来说明上传成功的方案:
  2. 采用android客户端+ServletHttpServletRequest)进行文件上传。
  3. Servlet端代码如下:
  1. DiskFileItemFactory factory = new DiskFileItemFactory();
  2. ServletFileUpload upload = new ServletFileUpload(factory);
  3. try
  4. {
  5. List items = upload.parseRequest(request);
  6. Iterator itr = items.iterator();
  7. while (itr.hasNext())
  8. {
  9. FileItem item = (FileItem) itr.next();
  10. if (item.isFormField())
  11. {
  12. System.out.println("表单参数名:" + item.getFieldName() + ",表单参数值:" + item.getString("UTF-8"));
  13. }
  14. else
  15. {
  16. if (item.getName() != null && !item.getName().equals(""))
  17. {
  18. System.out.println("上传文件的大小:" + item.getSize());
  19. System.out.println("上传文件的类型:" + item.getContentType());
  20. // item.getName()返回上传文件在客户端的完整路径名称
  21. System.out.println("上传文件的名称:" + item.getName());
  22.  
  23. File tempFile = new File(item.getName());
  24. // 上传文件的保存路径
  25. File file = new File(sc.getRealPath("/") + savePath, tempFile.getName());
  26. item.write(file);
  27. request.setAttribute("upload.message", "上传文件成功!");
  28. } else
  29. {
  30. request.setAttribute("upload.message", "没有选择上传文件!");
  31. }
  32. }
  33. }
  34. }
  35. catch (FileUploadException e)
  36. {
  37. e.printStackTrace();
  38. }
  39. catch (Exception e)
  40. {
  41. e.printStackTrace();
  42. request.setAttribute("upload.message", "上传文件失败!");
  43. }
  44. request.getRequestDispatcher("/uploadResult.jsp").forward(request, response);

Android端代码如下:

  1. public class SocketHttpRequester {
  2. /**
  3. *多文件上传
  4. * 直接通过HTTP协议提交数据到服务器,实现如下面表单提交功能:
  5. * <FORM METHOD=POST ACTION="http://192.168.1.101:8083/upload/servlet/UploadServlet" enctype="multipart/form-data">
  6. <INPUT TYPE="text" NAME="name">
  7. <INPUT TYPE="text" NAME="id">
  8. <input type="file" name="imagefile"/>
  9. <input type="file" name="zip"/>
  10. </FORM>
  11. * @param path 上传路径(注:避免使用localhost或127.0.0.1这样的路径测试,因为它会指向手机模拟器,你可以使用http://www.iteye.cn或http://192.168.1.101:8083这样的路径测试)
  12. * @param params 请求参数 key为参数名,value为参数值
  13. * @param file 上传文件
  14. */
  15. public static boolean post(String path, Map<String, String> params, FormFile[] files) throws Exception{
  16. final String BOUNDARY = "---------------------------7da2137580612"; //数据分隔线
  17. final String endline = "--" + BOUNDARY + "--\r\n";//数据结束标志
  18.  
  19. int fileDataLength = ;
  20. for(FormFile uploadFile : files){//得到文件类型数据的总长度
  21. StringBuilder fileExplain = new StringBuilder();
  22. fileExplain.append("--");
  23. fileExplain.append(BOUNDARY);
  24. fileExplain.append("\r\n");
  25. fileExplain.append("Content-Disposition: form-data;name=\""+ uploadFile.getParameterName()+"\";filename=\""+ uploadFile.getFilname() + "\"\r\n");
  26. fileExplain.append("Content-Type: "+ uploadFile.getContentType()+"\r\n\r\n");
  27. fileExplain.append("\r\n");
  28. fileDataLength += fileExplain.length();
  29. if(uploadFile.getInStream()!=null){
  30. fileDataLength += uploadFile.getFile().length();
  31. }else{
  32. fileDataLength += uploadFile.getData().length;
  33. }
  34. }
  35. StringBuilder textEntity = new StringBuilder();
  36. for (Map.Entry<String, String> entry : params.entrySet()) {//构造文本类型参数的实体数据
  37. textEntity.append("--");
  38. textEntity.append(BOUNDARY);
  39. textEntity.append("\r\n");
  40. textEntity.append("Content-Disposition: form-data; name=\""+ entry.getKey() + "\"\r\n\r\n");
  41. textEntity.append(entry.getValue());
  42. textEntity.append("\r\n");
  43. }
  44. //计算传输给服务器的实体数据总长度
  45. int dataLength = textEntity.toString().getBytes().length + fileDataLength + endline.getBytes().length;
  46.  
  47. URL url = new URL(path);
  48. int port = url.getPort()==- ? : url.getPort();
  49. Socket socket = new Socket(InetAddress.getByName(url.getHost()), port);
  50. OutputStream outStream = socket.getOutputStream();
  51. //下面完成HTTP请求头的发送
  52. String requestmethod = "POST "+ url.getPath()+" HTTP/1.1\r\n";
  53. outStream.write(requestmethod.getBytes());
  54. String accept = "Accept: image/gif, image/jpeg, image/pjpeg, image/pjpeg, application/x-shockwave-flash, application/xaml+xml, application/vnd.ms-xpsdocument, application/x-ms-xbap, application/x-ms-application, application/vnd.ms-excel, application/vnd.ms-powerpoint, application/msword, */*\r\n";
  55. outStream.write(accept.getBytes());
  56. String language = "Accept-Language: zh-CN\r\n";
  57. outStream.write(language.getBytes());
  58. String contenttype = "Content-Type: multipart/form-data; boundary="+ BOUNDARY+ "\r\n";
  59. outStream.write(contenttype.getBytes());
  60. String contentlength = "Content-Length: "+ dataLength + "\r\n";
  61. outStream.write(contentlength.getBytes());
  62. String alive = "Connection: Keep-Alive\r\n";
  63. outStream.write(alive.getBytes());
  64. String host = "Host: "+ url.getHost() +":"+ port +"\r\n";
  65. outStream.write(host.getBytes());
  66. //写完HTTP请求头后根据HTTP协议再写一个回车换行
  67. outStream.write("\r\n".getBytes());
  68. //把所有文本类型的实体数据发送出来
  69. outStream.write(textEntity.toString().getBytes());
  70. //把所有文件类型的实体数据发送出来
  71. for(FormFile uploadFile : files){
  72. StringBuilder fileEntity = new StringBuilder();
  73. fileEntity.append("--");
  74. fileEntity.append(BOUNDARY);
  75. fileEntity.append("\r\n");
  76. fileEntity.append("Content-Disposition: form-data;name=\""+ uploadFile.getParameterName()+"\";filename=\""+ uploadFile.getFilname() + "\"\r\n");
  77. fileEntity.append("Content-Type: "+ uploadFile.getContentType()+"\r\n\r\n");
  78. outStream.write(fileEntity.toString().getBytes());
  79. if(uploadFile.getInStream()!=null){
  80. byte[] buffer = new byte[];
  81. int len = ;
  82. while((len = uploadFile.getInStream().read(buffer, , ))!=-){
  83. outStream.write(buffer, , len);
  84. }
  85. uploadFile.getInStream().close();
  86. }else{
  87. outStream.write(uploadFile.getData(), , uploadFile.getData().length);
  88. }
  89. outStream.write("\r\n".getBytes());
  90. }
  91. //下面发送数据结束标志,表示数据已经结束
  92. outStream.write(endline.getBytes());
  93.  
  94. BufferedReader reader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
  95. if(reader.readLine().indexOf("")==-){//读取web服务器返回的数据,判断请求码是否为200,如果不是200,代表请求失败
  96. return false;
  97. }
  98. outStream.flush();
  99. outStream.close();
  100. reader.close();
  101. socket.close();
  102. return true;
  103. }
  104.  
  105. /**
  106. *单文件上传
  107. * 提交数据到服务器
  108. * @param path 上传路径(注:避免使用localhost或127.0.0.1这样的路径测试,因为它会指向手机模拟器,你可以使用http://www.itcast.cn或http://192.168.1.10:8080这样的路径测试)
  109. * @param params 请求参数 key为参数名,value为参数值
  110. * @param file 上传文件
  111. */
  112. public static boolean post(String path, Map<String, String> params, FormFile file) throws Exception{
  113. return post(path, params, new FormFile[]{file});
  114. }
  115. }

FormFile类的代码如下

  1. public class FormFile {
  2. /* 上传文件的数据 */
  3. private byte[] data;
  4. private InputStream inStream;
  5. private File file;
  6.  
  7. private int fileSize;
  8. /* 文件名称 */
  9. private String filname;
  10. /* 请求参数名称*/
  11. private String parameterName;
  12. /* 内容类型 */
  13. private String contentType = "application/octet-stream";
  14.  
  15. public FormFile(String filname, byte[] data, String parameterName, String contentType) {
  16. this.data = data;
  17. this.filname = filname;
  18. this.parameterName = parameterName;
  19. if(contentType!=null) this.contentType = contentType;
  20. }
  21.  
  22. public FormFile(String filname, File file, String parameterName, String contentType) {
  23. this.filname = filname;
  24. this.parameterName = parameterName;
  25. this.file = file;
  26. try {
  27. this.inStream = new FileInputStream(file);
  28. } catch (FileNotFoundException e) {
  29. e.printStackTrace();
  30. }
  31. if(contentType!=null) this.contentType = contentType;
  32. }
  33.  
  34. public FormFile(InputStream inStream, int fileSize, String filname,
  35. String parameterName, String contentType) {
  36. super();
  37. this.inStream = inStream;
  38. this.fileSize = fileSize;
  39. this.filname = filname;
  40. this.parameterName = parameterName;
  41. this.contentType = contentType;
  42. }
  43.  
  44. public int getFileSize() {
  45. return fileSize;
  46. }
  47.  
  48. public File getFile() {
  49. return file;
  50. }
  51.  
  52. public InputStream getInStream() {
  53. return inStream;
  54. }
  55.  
  56. public byte[] getData() {
  57. return data;
  58. }
  59.  
  60. public String getFilname() {
  61. return filname;
  62. }
  63.  
  64. public void setFilname(String filname) {
  65. this.filname = filname;
  66. }
  67.  
  68. public String getParameterName() {
  69. return parameterName;
  70. }
  71.  
  72. public void setParameterName(String parameterName) {
  73. this.parameterName = parameterName;
  74. }
  75.  
  76. public String getContentType() {
  77. return contentType;
  78. }
  79.  
  80. public void setContentType(String contentType) {
  81. this.contentType = contentType;
  82. }
  83.  
  84. }

android批量文件上传(android批量图片上传)的更多相关文章

  1. 在SAE上使用Ueditor的图片上传功能

    SAE上是没有文件夹读写权限的,所以要在SAE使用Ueditor的图片上传功能须要借助SAE的Storage服务. 一.开通Storage服务 在SAE控制台开通Storage服务,并新增一个doma ...

  2. 图片上传5-多个图片上传,独立项目Demo和源码

    图片上传,一次性可以上传多个图片,每个图片可以有名字.URL.排序.备注等字段.这是区别于使用百度WebUploader等多图上传工具的地方. 项目相关图片 Jar包管理:Maven用到的框架:Spr ...

  3. Android开发之使用GridView+仿微信图片上传功能(附源代码)

    前言:如果转载文章请声明转载自:https://i.cnblogs.com/EditPosts.aspx?postid=7419021  .另外针对有些网站转载本人的文章结果源码链接不对的问题,本人在 ...

  4. input file实现多选,限制文件上传类型,图片上传前预览功能

    限制上传类型 & 多选:① accept 属性只能与 <input type="file" /> 配合使用.它规定能够通过文件上传进行提交的文件类型. ② mu ...

  5. 用html5文件api实现移动端图片上传&预览效果

    想要用h5在移动端实现图片上传&预览效果,首先要了解html5的文件api相关知识(所有api只列举本功能所需): 1.Blob对象  Blob表示原始二进制数据,Html5的file对象就继 ...

  6. android 布局文件中xmlns:android="http://schemas.android.com/apk/res/android"

    http://blog.163.com/benben_long/blog/static/199458243201411394624170/ xmlns:android="http://sch ...

  7. Ueditor 1.4.3.1 使用 ThinkPHP 3.2.3 的上传类进行图片上传

    在 ThinkPHP 3.2.3 中集成百度编辑器最新版 Ueditor 1.4.3.1,同时将编辑器自带的上传类替换成 ThinkPHP 3.2.3 中的上传类. ① 下载编辑器(下载地址:http ...

  8. 百度上传工具webuploader,图片上传附加参数

    项目中需要上传视频,图片等资源.最先做的是上传图片,开始在网上找了一款野鸡插件,可以实现图片上传预览(无需传到后台).但是最近这个插件出了莫名的问题,不易修复,一怒之下,还是决定找个大点的,靠谱的插件 ...

  9. android 拍照或者图库选择 压缩后 图片 上传

    通过拍照或者从相册里选择图片通过压缩并上传时很多应用的常用功能,记录一下实现过程 一:创建个临时文件夹用于保存压缩后需要上传的图片 /** * path:存放图片目录路径 */ private Str ...

  10. Java用来进行批量文件重命名,批量提取特定类型文件

    原因: 因为在网上下载视频教程,有的名字特别长,一般都是机构或者网站的宣传,不方便直接看到视频的简介,所以做了下面的第一个功能. 因为老师发的课件中,文件夹太多,想把docx都放在同一个文件夹下面,一 ...

随机推荐

  1. Android项目使用support v7时遇到的各种问题

    Android项目使用support v7时遇到的各种问题 点击你的工程右键-->Properties-->Android 1.查看你引用的appcompat_v7包是否引用正确 2.用较 ...

  2. hdu1796:容斥入门题

    简单的容斥入门题.. 容斥基本的公式早就知道了,但是一直不会写. 下午看到艾神在群里说的“会枚举二进制数就会容斥”,后来发现还真是这样.. 然后直接贴代码了 #include <iostream ...

  3. bzoj1047-理想的正方形(二维单调队列)

    题意: 给一个矩阵,给出行列和每个数,再给出一个N,求出所有N*N的子矩阵中最大值最小值之差的最小值解析: 暴力枚举肯定不行,这题可以用二维单调队列做,把同一行的连续N个点缩成一个点保存最大最小值预处 ...

  4. DBA避坑宝典:Oracle运维中的那些事儿

    对于Oracle运维中的那些事儿,我的最终目的:不是比谁更惨,而是能够从中吸取经验和教训. 从我的理解来看,我会从下面的几个方面来进行说明DBA运维中的一些事儿. 每个部分都是非常关键的,缺一不可,而 ...

  5. Linux远程访问windows时,出现"连接被对端重置"错误

    1.sudo apt-get install rdesktop 需要下载 152 kB 的软件包.       解压缩后会消耗掉 512 kB 的额外空间. 2.运行时出现错误 root@oskey- ...

  6. (转)直接保存对象的数据库——db4o

    在实际开发中,数据的存储是必不可少的,常用的有数据库存储和文件存储.数据库目前有关系型数据库和文档型数据库(No-SQL).关系型数据库以字段.类型.约束.表关系来存储和管理数据,比较常见的比如Ora ...

  7. java接口传递数据的实例

    我们要讲E类中的数据变化通知A类,这样通过接口F来实现.具体原理就是E的每次数据改变都让其通知接口:而A类继承接口,所以每次E的调用接口都会触发A类的数据更改事件的触发. 首先创建一个类E: publ ...

  8. Enable-Migrations 在应用程序配置文件中找不到xx连接字符串

    在解决方案中有多个项目时,使用Enable-Migrations 命令进行数据迁移时,出现以下错误: 尝试在Enable-Migrations 命令中指定-projectName也不行,最后将要操作的 ...

  9. Merge Sorted Array 合并数组并排序

    Given two sorted integer arrays nums1 and nums2, merge nums2 into nums1 as one sorted array. Note:Yo ...

  10. silverlight visifire控件图表制作——silverlight 后台方法ControlChart.xaml.cs

    一.构造方法ControlChart 1.前台页面控件赋值 //时间下拉框赋值,下拉框赋选定值                for (int ii = DateTime.Today.Year; ii ...