热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

Android编程使用HTTP协议与TCP协议实现上传文件的方法

这篇文章主要介绍了Android编程使用HTTP协议与TCP协议实现上传文件的方法,结合实例形式较为详细的分析了Android使用HTTP协议与TCP协议的具体步骤与实现文件传输的相关技巧,需要的朋友可以参考下

本文实例讲述了Android编程使用HTTP协议与TCP协议实现上传文件的方法。分享给大家供大家参考,具体如下:

Android上传文件有两种方式,第一种是基于Http协议的HttpURLConnection,第二种是基于TCP协议的Socket。 这两种方式的区别是使用HttpURLConnection上传时内部有缓存机制,如果上传较大文件会导致内存溢出。如果用TCP协议Socket方式上传就会解决这种弊端。

HTTP协议HttpURLConnection

1. 通过URL封装路径打开一个HttpURLConnection
2.设置请求方式以及头字段:Content-Type、Content-Length、Host
3.拼接数据发送

示例:

private static final String BOUNDARY = "---------------------------7db1c523809b2";//数据分割线
public boolean uploadHttpURLConnection(String username, String password, String path) throws Exception {
  //找到sdcard上的文件
  File file = new File(Environment.getExternalStorageDirectory(), path);
  //仿Http协议发送数据方式进行拼接
  StringBuilder sb = new StringBuilder();
  sb.append("--" + BOUNDARY + "\r\n");
  sb.append("Content-Disposition: form-data; name=\"username\"" + "\r\n");
  sb.append("\r\n");
  sb.append(username + "\r\n");
  sb.append("--" + BOUNDARY + "\r\n");
  sb.append("Content-Disposition: form-data; name=\"password\"" + "\r\n");
  sb.append("\r\n");
  sb.append(password + "\r\n");
  sb.append("--" + BOUNDARY + "\r\n");
  sb.append("Content-Disposition: form-data; name=\"file\"; filename=\"" + path + "\"" + "\r\n");
  sb.append("Content-Type: image/pjpeg" + "\r\n");
  sb.append("\r\n");
  byte[] before = sb.toString().getBytes("UTF-8");
  byte[] after = ("\r\n--" + BOUNDARY + "--\r\n").getBytes("UTF-8");
  URL url = new URL("http://192.168.1.16:8080/14_Web/servlet/LoginServlet");
  HttpURLConnection cOnn= (HttpURLConnection) url.openConnection();
  conn.setRequestMethod("POST");
  conn.setRequestProperty("Content-Type", "multipart/form-data; boundary=" + BOUNDARY);
  conn.setRequestProperty("Content-Length", String.valueOf(before.length + file.length() + after.length));
  conn.setRequestProperty("HOST", "192.168.1.16:8080");
  conn.setDoOutput(true);
  OutputStream out = conn.getOutputStream();
  InputStream in = new FileInputStream(file);
  out.write(before);
  byte[] buf = new byte[1024];
  int len;
  while ((len = in.read(buf)) != -1)
    out.write(buf, 0, len);
  out.write(after);
  in.close();
  out.close();
  return conn.getResponseCode() == 200;
}

TCP协议Socket

1.我们可以使用Socket发送TCP请求,将上传数据分段发送

示例:

public boolean uploadBySocket(String username, String password, String path) throws Exception {
  // 根据path找到SDCard中的文件
  File file = new File(Environment.getExternalStorageDirectory(), path);
  // 组装表单字段和文件之前的数据
  StringBuilder sb = new StringBuilder();
  sb.append("--" + BOUNDARY + "\r\n");
  sb.append("Content-Disposition: form-data; name=\"username\"" + "\r\n");
  sb.append("\r\n");
  sb.append(username + "\r\n");
  sb.append("--" + BOUNDARY + "\r\n");
  sb.append("Content-Disposition: form-data; name=\"password\"" + "\r\n");
  sb.append("\r\n");
  sb.append(password + "\r\n");
  sb.append("--" + BOUNDARY + "\r\n");
  sb.append("Content-Disposition: form-data; name=\"file\"; filename=\"" + path + "\"" + "\r\n");
  sb.append("Content-Type: image/pjpeg" + "\r\n");
  sb.append("\r\n");
  // 文件之前的数据
  byte[] before = sb.toString().getBytes("UTF-8");
  // 文件之后的数据
  byte[] after = ("\r\n--" + BOUNDARY + "--\r\n").getBytes("UTF-8");
  URL url = new URL("http://192.168.1.199:8080/14_Web/servlet/LoginServlet");
  // 由于HttpURLConnection中会缓存数据, 上传较大文件时会导致内存溢出, 所以我们使用Socket传输
  Socket socket = new Socket(url.getHost(), url.getPort());
  OutputStream out = socket.getOutputStream();
  PrintStream ps = new PrintStream(out, true, "UTF-8");
  // 写出请求头
  ps.println("POST /14_Web/servlet/LoginServlet HTTP/1.1");
  ps.println("Content-Type: multipart/form-data; boundary=" + BOUNDARY);
  ps.println("Content-Length: " + String.valueOf(before.length + file.length() + after.length));
  ps.println("Host: 192.168.1.199:8080");
  InputStream in = new FileInputStream(file);
  // 写出数据
  out.write(before);
  byte[] buf = new byte[1024];
  int len;
  while ((len = in.read(buf)) != -1)
    out.write(buf, 0, len);
  out.write(after);
  in.close();
  out.close();
  return true;
}

搭建服务器,完成上传功能

package cn.test.web.servlet;
import java.io.File;
import java.io.IOException;
import java.util.List;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileItemFactory;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;
public class LoginServlet extends HttpServlet {
  private static final long serialVersiOnUID= 1L;
  @Override
  public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    doPost(request, response);
  }
  @Override
  public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    boolean isMultipart = ServletFileUpload.isMultipartContent(request);
    if (isMultipart)
      try {
        FileItemFactory factory = new DiskFileItemFactory();
        ServletFileUpload upload = new ServletFileUpload(factory);
        List items = upload.parseRequest(request);
        File dir = new File(request.getSession().getServletContext().getRealPath("/WEB-INF/upload"));
        //创建目录
        dir.mkdir();
        for (FileItem item : items)
          if (item.isFormField())
            System.out.println(item.getFieldName() + ": " + item.getString());
          else{
            item.write(new File(dir,item.getName().substring(item.getName().lastIndexOf("\\")+1)));
          }
      } catch (Exception e) {
        e.printStackTrace();
      }
    else {
      System.out.println(request.getMethod());
      System.out.println(request.getParameter("username"));
      System.out.println(request.getParameter("password"));
    }
  }
}

希望本文所述对大家Android程序设计有所帮助。


推荐阅读
  • PHP 5.5.0rc1 发布:深入解析 Zend OPcache
    2013年5月9日,PHP官方发布了PHP 5.5.0rc1和PHP 5.4.15正式版,这两个版本均支持64位环境。本文将详细介绍Zend OPcache的功能及其在Windows环境下的配置与测试。 ... [详细]
  • 深入理解Cookie与Session会话管理
    本文详细介绍了如何通过HTTP响应和请求处理浏览器的Cookie信息,以及如何创建、设置和管理Cookie。同时探讨了会话跟踪技术中的Session机制,解释其原理及应用场景。 ... [详细]
  • 本文详细介绍了如何在Linux系统上安装和配置Smokeping,以实现对网络链路质量的实时监控。通过详细的步骤和必要的依赖包安装,确保用户能够顺利完成部署并优化其网络性能监控。 ... [详细]
  • 本文介绍了如何在具备多个IP地址的FTP服务器环境中,通过动态地址端口复用和地址转换技术优化网络配置。重点讨论了2Mb/s DDN专线连接、Cisco 2611路由器及内部网络地址规划。 ... [详细]
  • 本文详细介绍了 Dockerfile 的编写方法及其在网络配置中的应用,涵盖基础指令、镜像构建与发布流程,并深入探讨了 Docker 的默认网络、容器互联及自定义网络的实现。 ... [详细]
  • 网络攻防实战:从HTTP到HTTPS的演变
    本文通过一系列日记记录了从发现漏洞到逐步加强安全措施的过程,探讨了如何应对网络攻击并最终实现全面的安全防护。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 360SRC安全应急响应:从漏洞提交到修复的全过程
    本文详细介绍了360SRC平台处理一起关键安全事件的过程,涵盖从漏洞提交、验证、排查到最终修复的各个环节。通过这一案例,展示了360在安全应急响应方面的专业能力和严谨态度。 ... [详细]
  • 本文详细分析了Hive在启动过程中遇到的权限拒绝错误,并提供了多种解决方案,包括调整文件权限、用户组设置以及环境变量配置等。 ... [详细]
  • 本文探讨了如何优化和正确配置Kafka Streams应用程序以确保准确的状态存储查询。通过调整配置参数和代码逻辑,可以有效解决数据不一致的问题。 ... [详细]
  • 解决MongoDB Compass远程连接问题
    本文记录了在使用阿里云服务器部署MongoDB后,通过MongoDB Compass进行远程连接时遇到的问题及解决方案。详细介绍了从防火墙配置到安全组设置的各个步骤,帮助读者顺利解决问题。 ... [详细]
  • 本文详细探讨了HTTP 500内部服务器错误的成因、解决方案及其在Web开发中的影响。通过对具体案例的分析,帮助读者理解并解决此类问题。 ... [详细]
  • 本文介绍了如何使用PHP代码实现微信平台的媒体素材上传功能,详细解释了API接口的使用方法和注意事项,确保文件路径正确以避免常见的错误。 ... [详细]
  • 使用Vultr云服务器和Namesilo域名搭建个人网站
    本文详细介绍了如何通过Vultr云服务器和Namesilo域名搭建一个功能齐全的个人网站,包括购买、配置服务器以及绑定域名的具体步骤。文章还提供了详细的命令行操作指南,帮助读者顺利完成建站过程。 ... [详细]
  • 网络运维工程师负责确保企业IT基础设施的稳定运行,保障业务连续性和数据安全。他们需要具备多种技能,包括搭建和维护网络环境、监控系统性能、处理突发事件等。本文将探讨网络运维工程师的职业前景及其平均薪酬水平。 ... [详细]
author-avatar
ccmmm
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有