Android学习笔记_13_网络通信之多个上传文件

时间:2023-03-08 21:22:34

一、获取HTTP协议:

  建立一个Web项目,建立一个如下所示的jsp界面,用IE捕获表单提交信息。

<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<form action="<%=request.getContextPath() %>/FileUpload" method="post" enctype="multipart/form-data">
<input type="text" name="name" > <br>
<input type="file" name="file1"> <br>
<input type="file" name="file2"> <br>
<input type="submit" value=" submit ">
</form>
</body>
</html>

  通过IE捕获信息如下:

POST /Simple/FileUpload HTTP/1.1
Accept: image/gif, image/jpeg, image/pjpeg, image/pjpeg, application/QVOD, application/QVOD, application/x-shockwave-flash,
application/vnd.ms-excel, application/vnd.ms-powerpoint, application/msword, application/x-ms-application, application/x-ms-xbap,
application/vnd.ms-xpsdocument, application/xaml+xml, */*
Referer: http://192.168.8.103:8080/Simple/1.jsp
Accept-Language: zh-cn
User-Agent: Mozilla/4.0 (compatible; MSIE 8.0; YYGameAll_1.2.167057.92; Windows NT 5.1; Trident/4.0; GTB6.4; .NET CLR 3.0.04506.648;
.NET CLR 3.5.21022; InfoPath.2; .NET CLR 2.0.50727; .NET CLR 3.0.4506.2152; .NET CLR 3.5.30729; .NET4.0C)
Content-Type: multipart/form-data; boundary=---------------------------7dd2ce2338056a
Accept-Encoding: gzip, deflate
Host: 192.168.8.103:8080
Content-Length: 3940
Connection: Keep-Alive
Cache-Control: no-cache
Cookie: JSESSIONID=B497923BC243F5507DEF27912E36A5FF -----------------------------7dd2ce2338056a
Content-Disposition: form-data; name="name" 123456
-----------------------------7dd2ce2338056a
Content-Disposition: form-data; name="file2"; filename="1.txt"
Content-Type: text/plain 1.内容和Content-Type之间有一个换行,分隔符“
-----------------------------7dd2ce2338056a” 所有内存传输完成之后,会加一个结束行“-----------------------------7dd2ce2338056a--”
-----------------------------7dd2ce2338056a--

  二、文件上传代码实现:

  在进行文件上传时,如果用Socket就需要将上面的HTTP请求头写出去,可以通过socket.getOutputStream()获取输出流对象,进而将数据写出去。如果采用HttpURLConnection相对Socket就简单些,可以通过类似con.setRequestProperty("Connection", "Keep-Alive")就可以讲请求头添加进去。下面采用两种方法实现

文件上传,Socket和HttpURLConnection.

package com.example.util;

import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.io.ByteArrayOutputStream;
import java.io.DataOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.net.HttpURLConnection;
import java.net.InetAddress;
import java.net.Socket;
import java.net.URL;
import java.net.URLEncoder;
import java.util.ArrayList;
import java.util.List;
import java.util.Map; import org.apache.http.HttpResponse;
import org.apache.http.NameValuePair;
import org.apache.http.client.HttpClient;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.message.BasicNameValuePair; public class SocketHttpRequester {
/**
* 直接通过HTTP协议提交数据到服务器,实现如下面表单提交功能: <FORM METHOD=POST
* ACTION="http://192.168.0.200:8080/ssi/fileload/test.do"
* enctype="multipart/form-data"> <INPUT TYPE="text" NAME="name"> <INPUT
* TYPE="text" NAME="id"> <input type="file" name="imagefile"/> <input
* type="file" name="zip"/> </FORM>
*
* @param path
* 上传路径(注:避免使用localhost或127.0.0.1这样的路径测试,
* 因为它会指向手机模拟器,你可以使用http://www.itcast.cn或http://192.168.1.10:8080
* 这样的路径测试)
* @param params
* 请求参数 key为参数名,value为参数值
* @param file
* 上传文件
*/
public static boolean post(String path, Map<String, String> params,
FormFile[] files) throws Exception {
final String BOUNDARY = "---------------------------7da2137580612"; // 数据分隔线
final String endline = "--" + BOUNDARY + "--\r\n";// 数据结束标志//注意回车换行部分 int fileDataLength = 0;
for (FormFile uploadFile : files) {// 得到文件类型数据的总长度
StringBuilder fileExplain = new StringBuilder();
fileExplain.append("--");
fileExplain.append(BOUNDARY);
fileExplain.append("\r\n");
// 以上部分用来完成分割线: -----------------------------7dd33a32e07fc
// 注意这里的回车换行
fileExplain.append("Content-Disposition: form-data;name=\""
+ uploadFile.getParameterName() + "\";filename=\""
+ uploadFile.getFilename() + "\"\r\n");
fileExplain.append("Content-Type: " + uploadFile.getContentType()
+ "\r\n\r\n");
// 上面两行用来
/*
* Content-Disposition: form-data; name="videofile";
* filename="2.png" Content-Type: image/jpeg
*/
fileDataLength += fileExplain.length();
/*
* //通过上面就可以计算下面这个部分的总长度了 -----------------------------7dd33a32e07fc
* Content-Disposition: form-data; name="videofile";
* filename="2.png" Content-Type: image/jpeg
*/
if (uploadFile.getInStream() != null) {
fileDataLength += uploadFile.getFile().length();
} else {
fileDataLength += uploadFile.getData().length;
}
fileDataLength += "\r\n".length();
}
StringBuilder textEntity = new StringBuilder();
for (Map.Entry<String, String> entry : params.entrySet()) {// 构造文本类型参数的实体数据
textEntity.append("--");
textEntity.append(BOUNDARY);
textEntity.append("\r\n");
textEntity.append("Content-Disposition: form-data; name=\""
+ entry.getKey() + "\"\r\n\r\n");
textEntity.append(entry.getValue());
textEntity.append("\r\n");
}
// 计算传输给服务器的实体数据总长度
int dataLength = textEntity.toString().getBytes().length
+ fileDataLength + endline.getBytes().length;
URL url = new URL(path);
int port = url.getPort() == -1 ? 80 : url.getPort();
// 这里没有用HttpUrlConnection来做,而是直接用socket来做
// 因为HttpUrlConnection使用了缓冲技术,通过这个类给web,应用发送文件的时候
// 这个时候文件首先写入到缓冲里面,很可能导致内存溢出的情况.
// Socket比HttpUrlConnection底层,没有缓冲区限制
Socket socket = new Socket(InetAddress.getByName(url.getHost()), port);
OutputStream outStream = socket.getOutputStream();
// 下面完成HTTP请求头的发送
String requestmethod = "POST " + url.getPath() + " HTTP/1.1\r\n";
outStream.write(requestmethod.getBytes());
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";
outStream.write(accept.getBytes());
String language = "Accept-Language: zh-CN\r\n";
outStream.write(language.getBytes());
String contenttype = "Content-Type: multipart/form-data; boundary="
+ BOUNDARY + "\r\n";
outStream.write(contenttype.getBytes());
String contentlength = "Content-Length: " + dataLength + "\r\n";
outStream.write(contentlength.getBytes());
String alive = "Connection: Keep-Alive\r\n";
outStream.write(alive.getBytes());
String host = "Host: " + url.getHost() + ":" + port + "\r\n";
outStream.write(host.getBytes());
// 写完HTTP请求头后根据HTTP协议再写一个回车换行
outStream.write("\r\n".getBytes());
// 把所有文本类型的实体数据发送出来
outStream.write(textEntity.toString().getBytes());
// 把所有文件类型的实体数据发送出来
for (FormFile uploadFile : files) {
StringBuilder fileEntity = new StringBuilder();
fileEntity.append("--");
fileEntity.append(BOUNDARY);
fileEntity.append("\r\n");
fileEntity.append("Content-Disposition: form-data;name=\""
+ uploadFile.getParameterName() + "\";filename=\""
+ uploadFile.getFilename() + "\"\r\n");
fileEntity.append("Content-Type: " + uploadFile.getContentType()
+ "\r\n\r\n");
outStream.write(fileEntity.toString().getBytes());
// 大文件
if (uploadFile.getInStream() != null) {
byte[] buffer = new byte[1024];
int len = 0;
while ((len = uploadFile.getInStream().read(buffer, 0, 1024)) != -1) {
outStream.write(buffer, 0, len);
}
uploadFile.getInStream().close();
} else {// 小文件
outStream.write(uploadFile.getData(), 0,
uploadFile.getData().length);
}
outStream.write("\r\n".getBytes());
}
// 下面发送数据结束标志,表示数据已经结束
outStream.write(endline.getBytes());
BufferedReader reader = new BufferedReader(new InputStreamReader(
socket.getInputStream()));
if (reader.readLine().indexOf("200") == -1) {// 读取web服务器返回的数据,判断请求码是否为200,如果不是200,代表请求失败
return false;
}
outStream.flush();
outStream.close();
reader.close();
socket.close();
return true;
} //单个文件
public static boolean post(String path, Map<String, String> params,
FormFile file) throws Exception {
return post(path, params, new FormFile[] { file });
} //单个文件
public static boolean uploadFile(String path,Map<String, String> params,File file,String encode) {
return uploadFile(path, params, new File[]{file}, encode);
}
/**
*
* @param path 访问路径
* @param params 简单参数
* @param files 上传的文件
* @param encode 编码
* @return
*/
public static boolean uploadFile(String path,Map<String, String> params,File[] files,String encode) {
BufferedInputStream inputStream = null;
HttpURLConnection con =null;
DataOutputStream ds = null;
try {
String end = "\r\n";//换行
String twoHyphens = "--";//内容分割线多了两个“--”
String boundary = "---------------------------7dd2ce2338056a";// 分割线
URL url = new URL(path);
con = (HttpURLConnection) url.openConnection();
/* 允许Input、Output,不使用Cache */
con.setDoInput(true);
con.setDoOutput(true);
con.setUseCaches(false);
/* 设置传送的method=POST */
con.setRequestMethod("POST");
/* setRequestProperty */
con.setRequestProperty("Connection", "Keep-Alive");
con.setRequestProperty("Charset", encode);
con.setRequestProperty("Content-Type", "multipart/form-data;boundary="+ boundary);
/* 设置DataOutputStream */
ds = new DataOutputStream(con.getOutputStream());
//简单请求参数
if(params!=null){
for (String paramName : params.keySet()) {
ds.writeBytes(twoHyphens + boundary + end);
ds.writeBytes("Content-Disposition: form-data; name=\""+paramName+"\""+end);
ds.writeBytes(end);
ds.writeBytes(params.get(paramName));
ds.writeBytes(end);
}
}
//上传文件
int count=1;
if(files!=null && files.length>0){
for (File file : files) {
ds.writeBytes(twoHyphens + boundary + end);
ds.writeBytes("Content-Disposition: form-data; "
+ "name=\"file"+count+"\";filename=\"" + file.getName() + "\"" + end);
ds.writeBytes(end);
/* 取得文件的FileInputStream */
inputStream = new BufferedInputStream(new FileInputStream(file));
/* 设置每次写入1024bytes */
byte[] buffer = new byte[1024];
int length = -1;
/* 从文件读取数据至缓冲区 */
while ((length = inputStream.read(buffer)) != -1) {
ds.write(buffer, 0, length);
}
ds.writeBytes(end);
}
count++;
}
ds.writeBytes(twoHyphens + boundary + twoHyphens + end);
if (con.getResponseCode() == 200) {
return true;
}
} catch (Exception e) {
e.printStackTrace();
}finally{
if(inputStream!=null){
try {
inputStream.close();
} catch (IOException e) {
inputStream=null;
}
}
if(ds!=null){
try {
ds.close();
} catch (IOException e) {
ds=null;
}
}
if(con!=null){
con.disconnect();
}
}
return false;
} /**
* HttpClient提交数据到服务器
*
* @param path
* http://192.168.8.101:8080/Simple/FileUpload
* @param params
* 请求参数 key为参数名,value为参数值
* @param encode
* 编码
*/
public static byte[] postFromHttpClient(String path,
Map<String, String> params, String encode) throws Exception {
List<NameValuePair> formparams = new ArrayList<NameValuePair>();// 用于存放请求参数
for (Map.Entry<String, String> entry : params.entrySet()) {
formparams.add(new BasicNameValuePair(entry.getKey(), entry
.getValue()));
}
UrlEncodedFormEntity entity = new UrlEncodedFormEntity(formparams,
encode);
HttpPost httppost = new HttpPost(path);
httppost.setEntity(entity);
HttpClient httpclient = new DefaultHttpClient();// 看作是浏览器
HttpResponse response = httpclient.execute(httppost);// 发送post请求
return readStream(response.getEntity().getContent());
} /**
* HttpURLConnection发送请求,服务器返回流对象,可以用于下载服务器的资源
*
* @param path
* 请求路径
* @param params
* 请求参数 key为参数名称 value为参数值
* @param encode
* 请求参数的编码
*/
public static byte[] post(String path, Map<String, String> params,
String encode) throws Exception {
// String params = "method=save&name="+ URLEncoder.encode("老毕",
// "UTF-8")+ "&age=28&";//需要发送的参数
StringBuilder parambuilder = new StringBuilder("");
if (params != null && !params.isEmpty()) {
for (Map.Entry<String, String> entry : params.entrySet()) {
parambuilder.append(entry.getKey()).append("=")
.append(URLEncoder.encode(entry.getValue(), encode))
.append("&");
}
parambuilder.deleteCharAt(parambuilder.length() - 1);
}
byte[] data = parambuilder.toString().getBytes();
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setDoOutput(true);// 允许对外发送请求参数
conn.setUseCaches(false);// 不进行缓存
conn.setConnectTimeout(5 * 1000);
conn.setRequestMethod("POST");
// 下面设置http请求头
conn.setRequestProperty(
"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, */*");
conn.setRequestProperty("Accept-Language", "zh-CN");
conn.setRequestProperty(
"User-Agent",
"Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 5.2; Trident/4.0; .NET CLR 1.1.4322; .NET CLR 2.0.50727; .NET CLR 3.0.04506.30; .NET CLR 3.0.4506.2152; .NET CLR 3.5.30729)");
conn.setRequestProperty("Content-Type",
"application/x-www-form-urlencoded");
conn.setRequestProperty("Content-Length", String.valueOf(data.length));
conn.setRequestProperty("Connection", "Keep-Alive");
// 发送参数
DataOutputStream outStream = new DataOutputStream(
conn.getOutputStream());
outStream.write(data);// 把参数发送出去
outStream.flush();
outStream.close();
if (conn.getResponseCode() == 200) {
return readStream(conn.getInputStream());
}
return null;
} /**
* 读取流
*
* @param inStream
* @return 字节数组
* @throws Exception
*/
public static byte[] readStream(InputStream inStream) throws Exception {
ByteArrayOutputStream outSteam = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int len = -1;
while ((len = inStream.read(buffer)) != -1) {
outSteam.write(buffer, 0, len);
}
outSteam.close();
inStream.close();
return outSteam.toByteArray();
}
}

 FormFile类

package com.example.util;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.InputStream; public class FormFile {
//上传文件的数据
private byte[] data; private InputStream inStream; private File file;
//文件名称
private String filename;
//请求参数名称
private String parameterName;
//内容类型
private String contentType="application/octet-stream";
/**
* 针对上传文件比较小
* @param data 文件内容
* @param filename 文件名称
* @param parameterName 请求参数名称
* @param contentType 内容类型
*/
public FormFile(byte[] data, String filename, String parameterName,
String contentType) {
this.data = data;
this.filename = filename;
this.parameterName = parameterName;
if(contentType!=null)
this.contentType = contentType;
} /**
* 针对比较大的文件
* @param file 传一个文件对象
* @param parameterName 请求参数名称
* @param contentType 内容类型
*/
public FormFile(File file, String parameterName, String contentType) {
this.file = file;
this.filename=file.getName();
this.parameterName = parameterName;
if(contentType!=null)
this.contentType = contentType;
try {
this.inStream = new FileInputStream(file);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
} public byte[] getData() {
return data;
} public void setData(byte[] data) {
this.data = data;
} public InputStream getInStream() {
return inStream;
} public void setInStream(InputStream inStream) {
this.inStream = inStream;
} public File getFile() {
return file;
} public void setFile(File file) {
this.file = file;
} public String getFilename() {
return filename;
} public void setFilename(String filename) {
this.filename = filename;
} public String getParameterName() {
return parameterName;
} public void setParameterName(String parameterName) {
this.parameterName = parameterName;
} public String getContentType() {
return contentType;
} public void setContentType(String contentType) {
this.contentType = contentType;
}
}

  三、单元测试:

     public void testPost2()throws Exception{
String path="http://192.168.8.101:8080/Simple/FileUpload";
Map<String, String> map=new HashMap<String, String>();
map.put("username", "hello web");
FormFile[] formFiles=new FormFile[2];
formFiles[0] = new FormFile(new File(Environment.getExternalStorageDirectory(),"5.jpg"),"userimage","image/jpeg");
formFiles[1] = new FormFile(new File(Environment.getExternalStorageDirectory(),"5.png"),"userimage2","image/jpeg");
SocketHttpRequester.post(path, map, formFiles);
}
public void testPost3()throws Exception{
String path="http://192.168.8.101:8080/Simple/FileUpload";
Map<String, String> params=new HashMap<String, String>();
params.put("username", "hello web");
params.put("age", "12");
File[] files=new File[2];
files[0]=new File(Environment.getExternalStorageDirectory(),"5.jpg");
files[1]=new File(Environment.getExternalStorageDirectory(),"1.txt");
SocketHttpRequester.uploadFile(path,params,files,"UTF-8");
}

  四、通过Get或Post方式提交请求参数到Web:

  对于Get请求方式只需要将数据拼接到URL后面就可以,但是Get提交的数据长度有限,对于数据量比较大时,需要使用Post方式提交。下面是Post方式提交数据的HTTP响应头参数名称及值,对于Post请求需要将数据封装到“请求属性(RequestProperty)”上。从下面请求头信息可以看成,Post请求头和请求参数之间有换行(\r\n)。

下面是封装的Get和Post两种请求方式的封装类:

package com.example.service;

import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.net.HttpURLConnection;
import java.net.URL;
import java.net.URLEncoder;
import java.util.Map; /**
* 向web服务发送数据
* @author Administrator
*/
public class HTTPService { public static String contentType="application/x-www-form-urlencoded"; /**
* 发送post请求
* @param path 路径
* @param params 请求参数
* @param encoding 编码方式
* @return
*/
public static boolean sendPost(String path, Map<String, Object> params,
String encoding) {
try {
StringBuffer sb = buildParams(params, encoding,false);
byte[] data = sb.toString().getBytes();
HttpURLConnection conn = (HttpURLConnection) new URL(path).openConnection();
conn.setConnectTimeout(5000);
conn.setRequestMethod("POST");
// 允许对外输出数据
conn.setDoOutput(true);
conn.setRequestProperty("Content-Type",contentType);
conn.setRequestProperty("Content-Length",String.valueOf(data.length));
OutputStream outputStream = conn.getOutputStream();
outputStream.write(data);
if (conn.getResponseCode() == 200) {
// 可以通过 conn.getInputStream() 接收数据
return true;
}
} catch (Exception e) {
e.printStackTrace();
}
return false;
} /**
* 发送get请求
* @param path
* @param params
* @param encoding
* @return
*/
public static boolean sendGet(String path, Map<String, Object> params,
String encoding) {
try {
StringBuffer sb = buildParams(params, encoding,true);
path += sb.toString();
HttpURLConnection conn = (HttpURLConnection) new URL(path).openConnection();
conn.setConnectTimeout(5000);
conn.setRequestMethod("GET");
if (conn.getResponseCode() == 200) {
// 可以通过 conn.getInputStream() 接收数据
return true;
}
} catch (Exception e) {
e.printStackTrace();
}
return false;
} /**
* 组装参数
*
* @param params
* @param encoding
* @return
* @throws UnsupportedEncodingException
*/
private static StringBuffer buildParams(Map<String, Object> params,
String encoding,boolean isGet) throws UnsupportedEncodingException {
StringBuffer sb = new StringBuffer();
if (isGet) {
sb.append("?");
}
if (params != null) {
for (String key : params.keySet()) {
sb.append(key).append("=");
sb.append(URLEncoder.encode(String.valueOf(params.get(key)),
encoding));
sb.append("&");
}
sb.deleteCharAt(sb.length() - 1);
}
return sb;
}
}

测试:

public class HttpGetPostTest extends AndroidTestCase {

    public void testGet()throws Exception{
String path="http://192.168.8.103:8080/Simple/PersonList";
Map<String, Object> params = new HashMap<String, Object>();
params.put("name", "android get test...");
HTTPService.sendGet(path, params , "utf-8");
}
public void testPost()throws Exception{
String path="http://192.168.8.103:8080/Simple/PersonList";
Map<String, Object> params = new HashMap<String, Object>();
params.put("name", "android post test...");
HTTPService.sendPost(path, params , "utf-8");
} }