最近项目中需要客户端往服务器传输图片,并且还需要附带一些普通参数,研究了几天,把结果记录下。
首先客户端可服务端进行通信一般都是有http请求来发送和接收数据,这里android中有两种HttpClient和HttpURLConnection,这两个都可以和后台服务器进行网络通信,但是如何选择哪个来进行文件传输呢?
谷歌官方是提倡我们使用HttpURLConnection来和服务器进行通信(这个是在android.jar中就有),HttpClient这个是org.apache.http.client.HttpClient下的一个工具类,根据网上的一些测试,得出的一些理论结果是使用HttpURLConnection来传输文件比HttpClient要快一些,在普通参数传输上都差不多。
既然谷歌推荐,所以我么这里就选择HttpURLConnection来实现传输文件。
效果图非常简单
Android客户端逻辑:
1: 点击选图片—>打开系统的相册选择;
2:点击上传图片,把图片上传到服务器,然后显示进度,以及上传完成后显示服务器返回的json数据。首先看看项目的结构
- 显示的界面在MainAcvtitvity中;
- ConfigParamters是http请求的公共参数配置;
- HttpUrlConn是http请求主要类;
- Uploadllinsener用来监听文件上传的进度,以及上传的状态(成功,失败,进度)。这里主要是看看HttpUrlConn请求类:
public class HttpUrlConn {
//标签
private String TAG="HttpUrlConn";
//HttpURLConnection请求
private HttpURLConnection myhttpconnect;
//结束点
private String end = "\r\n";
//分割标记twoHyphens
private String twoHyphens = "--";
//分割标记
private String boundary = "*****";
//数据输出管道
private DataOutputStream ds=null;
//文件上传监听
private UploadLinsener uploadlinsener;
//文件总大小
private double allFilesize=0;
//当前传输了多少
private double curr_ret=0;
/**
* HttpURLConnection连接服务器请求
* @param path :接口地址
* @param param1 :普通参数
* @param paramFiles:需要上传的文件参数
* @param config:连接公共属性配置
* @return
*/
public HttpURLConnection getHttpUrlConnInstances(String path,
HashMap<String, String> paramString,
HashMap<String, File> paramFiles,
ConfigParamters config){
try {
//这里可以放普通参数,(第一种方法)
// StringBuffer sb=new StringBuffer(path);
// if(null!=paramString&¶mString.size()>0){
// sb.append("?");
// for(Entry<String, String> data:paramString.entrySet()){
// sb.append(data.getKey()+"="+data.getValue());
// sb.append("&");
// }
// }
// path=sb.substring(0, sb.lastIndexOf("&"));
URL url=new URL(path);
myhttpconnect=(HttpURLConnection)
url.openConnection();
/* 允许Input、Output,post中不使用Cache */
myhttpconnect.setDoInput(config.isDoInput());
myhttpconnect.setDoOutput(config.isDoOutput());
myhttpconnect.setUseCaches(config.isUseCaches());
/* 设置传送的method=POST */
myhttpconnect.
setRequestMethod(config.getUseMethod());
/*设置属性 setRequestProperty */
myhttpconnect
.setRequestProperty("Connection", "Keep-Alive");
myhttpconnect.setRequestProperty("Charset", "UTF-8");
myhttpconnect.setRequestProperty("Content-Type", "multipart/form-data;boundary=" + boundary);
/* 设置DataOutputStream */
ds = new DataOutputStream(
myhttpconnect.getOutputStream());
/*把普通参数写到服务器*/
if(null!=paramString&¶mString.size()>0){
writeStringParam(ds, paramString);
}
/*把文件参数写到服务器*/
if(null!=paramFiles&¶mFiles.size()>0){
writeFileParam(ds, paramFiles);
}
//添加结尾标志
paramsEnd(ds);
if(myhttpconnect.getResponseCode()
==HttpURLConnection.HTTP_OK){
InputStream is =
myhttpconnect.getInputStream();
String successMsg=InputStremtoString(is);
//成功
if(uploadlinsener!=null){
uploadlinsener.uploadSucsess(successMsg);
}
}else{
uploadlinsener.uploadFail("错误,错误码
ResponseCode:"+
myhttpconnect.getResponseCode());
}
ds.close();
} catch (MalformedURLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
if(uploadlinsener!=null){
uploadlinsener.uploadFail(e.toString());
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
if(uploadlinsener!=null){
uploadlinsener.uploadFail(e.toString());
}
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
if(uploadlinsener!=null){
uploadlinsener.uploadFail(e.toString());
}
}
return myhttpconnect;
}
/**
* 把字节输入流转换成字符串
* @return
*/
private String InputStremtoString(InputStream is)throws Exception{
int ch;
StringBuffer b = new StringBuffer();
byte[] bt=new byte[1024];
while ((ch = is.read(bt)) != -1)
{
b.append(new String(bt, 0, ch, "UTF-8"));
}
return b.toString();
}
/**
* 传普通参数(第二种方法)
* @param parm:参数
*/
private void writeStringParam(DataOutputStream ds,
HashMap<String, String> params){
try{
Set<String> keySet = params.keySet();
for (Iterator<String> it =keySet.iterator();
it.hasNext();){
String name = it.next();
String value = params.get(name);
ds.writeBytes(twoHyphens + boundary + end);
ds.writeBytes("Content-Disposition: form-data; name=\"" + name +"\""+ end);
ds.writeBytes(end);
//URLEncoder.encode(value)
ds.writeBytes(new String(value.getBytes(),"UTF-8")+ end);
}
}catch(Exception e){
e.printStackTrace();
}
}
/***
* 传输文件
* @param ds :数据传输通道
* @param params :参数
*/
private void writeFileParam(DataOutputStream ds,HashMap<String, File> params){
try{
Set<String> keySet = params.keySet();
getFilesSize(params);
for (Iterator<String> it =keySet.iterator();
it.hasNext();){
String name = it.next();
File value = params.get(name);
ds.writeBytes(twoHyphens + boundary + end);
ds.writeBytes("Content-Disposition: form-data; name=\"" + name + "\"; filename=\"" + URLEncoder.encode(value.getName()) + "\""+end);
ds.writeBytes("Content-Type: " + getContentType(value) + end);
ds.writeBytes(end);
ds.write(getBytes(value));
ds.writeBytes(end);
}
}catch(Exception e){
e.printStackTrace();
}
}
/**
* 获取要传输文件总共大小
* @param params
*/
private void getFilesSize(HashMap<String, File> params){
for(Entry<String, File> data:params.entrySet()){
File f=data.getValue();
allFilesize+=f.length();
}
Log.i(TAG, "AllFileSize:"+allFilesize);
}
/**
* 获取文件的上传类型,图片格式为image/png,image/jpg等。非图片为application/octet-stream
* @param f 文件
* @return
* @throws Exception
*/
private String getContentType(File f) throws Exception{
return "application/octet-stream";
}
/**
* 添加结尾数据
* @param ds:数据传输通道
* * @throws Exception
*/
private void paramsEnd( DataOutputStream ds) throws Exception {
ds.writeBytes(twoHyphens + boundary +
twoHyphens + end);
ds.writeBytes(end);
}
/**
* 把文件转换成字节数组
* @param f:文件
* @return
* @throws Exception
*/
private byte[] getBytes(File f) throws Exception{
FileInputStream in = new FileInputStream(f);
ByteArrayOutputStream out = new ByteArrayOutputStream();
byte[] b = new byte[1024];
int n;
while ((n = in.read(b)) != -1)
{
out.write(b, 0, n);
if(allFilesize>0){
curr_ret+=n;
double level=(curr_ret/allFilesize)*100;
Log.i("result", ""+level);
if(uploadlinsener!=null){
uploadlinsener.uploading(level);
}
}
}
in.close();
return out.toByteArray();
}
public void setUploadlinsener(UploadLinsener uploadlinsener) {
this.uploadlinsener = uploadlinsener;
}
}
上面的代码是HttpUrlConn的全部代码。
1: getHttpUrlConnInstances()获取并且请求连接服务器;
-方法中主要是获取连接,并且配置连接,然后拼接参数,发送到服务器,然后通过myhttpconnect.getInputStream()
获取到服务器返回的数据。
2:InputStremtoString(InputStream)把字节流转换成String;
- 这里主要是为了处理服务器返回的数据。(这里注意编码)
3:writeStringParam(DataOutputStream,HashMap<String, String>)
这个方法主要是为了传输普通的参数;
- 把普通参数装入HashMap<String, String>
中然后循环写出数据,这里的数据是需要拼接的,因为我们提交数据是模拟from表单提交。
4:writeFileParam(DataOutputStream,HashMap<String, File>)
这个方法只要是用来传输文件参数;
- 把需要传输的文件都装在HashMap<String, File>
中,循环遍历hashmap中的文件,一一写到服务器。(在这个方法中获取了所有文件的总大小,用于计算当前的传输进度,还有这里的name和服务器的要一样。)
5:getFilesSize(HashMap<String, File>)
获取文件的总大小;
6:getContentType(File f)
获取文件的上传类型,图片格式为image/png,image/jpg等。非图片为application/octet-stream;
7:paramsEnd(DataOutputStream)
添加结尾数据;
8:getBytes(File)
把文件转换成字节数组;
9:setUploadlinsener(UploadLinsener)
设置传输监听;
这里我们使用struts2的一个组件来实现配合客户端文件上传。
用struts,spring,hibernate三大框架搭建好的项目,这里项目搭建过程就不演示了。
这里的核心代码在FileUploadAction中:
public class FileUploadAction extends ActionSupport{
public String phoneinfo="NXJ570";
public void uploadfielandparam() throws Exception{
System.out.println("upload");
HttpServletRequest request=
ServletActionContext.getRequest();
System.out.println(request.getContentType()); System.out.println(request.getCharacterEncoding());
System.out.println(request.getContentLength());
//临时文件夹,在tomcat项目下的temp
String temp = ServletActionContext
.getServletContext()
.getRealPath("")+ "\\temp";
System.out.println(temp);
//上传文件存放地方
String basepath=ServletActionContext
.getServletContext()
.getRealPath("")+"\\myfile";
//获取Response
HttpServletResponse resp=
ServletActionContext.getResponse();
DiskFileItemFactory factory =
new DiskFileItemFactory();
//单个文件缓存最大值
factory.setSizeThreshold(10 * 1024 * 1024);
// 缓冲区
factory.setRepository(new File(temp));
//文件上传组件(关键)
ServletFileUpload upload1=
new ServletFileUpload(factory);
upload1.setFileSizeMax(10000000);
upload1.setSizeMax(480000000);
upload1.setHeaderEncoding("UTF-8");
// 设置一旦文件大小超过getSizeThreshold()
//的值时数据存放在硬盘的目录
// 开始读取上传信息
int index = 0;
List fileItems= null;
try
{
fileItems=upload1.parseRequest(request);
System.out.println("list=" + fileItems);
}
catch (Exception e)
{
e.printStackTrace();
}
Iterator iter = fileItems.iterator();
// 依次处理每个上传的文件
while (iter.hasNext())
{
FileItem item = (FileItem) iter.next();
// 忽略其他不是文件域的所有表单信息
if (!item.isFormField())
{
String name = item.getName();
// 获取上传文件名,包括路径
name = name.substring(
name.lastIndexOf("\\") + 1);
// 从全路径中提取文件名
long size = item.getSize();
if ((name == null
|| name.equals("")) && size == 0)
continue;
int point = name.indexOf(".");
name = (new Date()).getTime()
+ name.substring(point, name.length());
System.out.println("fileName="+name);
index++;
File fNew = new File(basepath, name);
try
{
item.write(fNew);
}
catch (Exception e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
else
// 取出不是文件域的所有表单信息
{
//如果包含中文应写为:(转为UTF-8编码)
String key=item.getFieldName();
String fieldvalue =
.getBytes(),"UTF-8");//UTF-8
System.out
.println("
params:"+key+"="+fieldvalue);
}
}
//把对象转换成json
Gson gson = new Gson();
myobj obj1=new myobj("0", "seccessed",
"服务器已经接收到文件");
String json = gson.toJson(obj1);
System.out.println(json);
//设置格式(必须设置,否则客户端接收到的json中文乱码)
resp.setContentType("text/html; charset=utf-8");
PrintWriter out = resp.getWriter();
out.println(json);
out.flush();
}
//实体对象
class myobj{
private String retcode;
private String retmessage;
private String retdata;
public myobj(String retcode,
String retmessage, String retdata) {
super();
this.retcode = retcode;
this.retmessage = retmessage;
this.retdata = retdata;
}
public String getRetcode() {
return retcode;
}
public void setRetcode(String retcode) {
this.retcode = retcode;
}
public String getRetmessage() {
return retmessage;
}
public void setRetmessage(String retmessage) {
this.retmessage = retmessage;
}
public String getRetdata() {
return retdata;
}
public void setRetdata(String retdata) {
this.retdata = retdata;
}
}
这个类是核心业务类,当客户端发送请求是进入到这个方法中,并且获取到参数数据。
在服务端有两种方式能后获取到android的普通参数,
一是通过HttpServletRequest的getParameter(arg0)方法来获取;
二是从提交的表单中获取:
//获取key
String key=item.getFieldName();
//获取vaule
String fieldvalue =new
String(item.getString().getBytes(),"UTF-8");
经过测试传输速度的确挺快的5MB的图片几秒钟就能传输完成,注意这里如果是在android客户端模拟机中调试程序,那里的请求地址中的ip应该为10.0.0.2:8080…..否则找不到服务器,如果是真机上测试可以直接发布到你电脑的tomcat中,然后把ip改成你点击的ip就可以了(这里电脑和android手机需要连接到同一个网络wifi,也就是内网,当然也可以拿到公司的服务器上测试)。
项目测试通过,效果也挺好的,不过这里还有一个问题没解决,就是通过from表单提交上来的普通参数如果是中文这里好像还是会乱码。
项目地址:http://download.csdn.net/detail/leifengpeng/8614101
android上传图片(及普通参数)到服务器(j2ee后台服务器,ssh框架)
原文地址:http://blog.csdn.net/leifengpeng/article/details/45156047