毕业两年了,也工作两年了,但是由于具体的工作内容原因,真正去按照老大的要求去写一个程序,还是第一次。
这次就遇到了一个问题,从来没有接触过的Android网络编程,需求是按照指定的三种方法,以POST方法,分别用HTTP和Socket上传一个文件到服务器。
指定的三种方法为:
第一种:形如 "http://host:port/xxx/xxx/xxx/param?param1=xxx¶m2=xxx"的形式
Params:
param1
param2
Data:
file:要上传的文件
第二种:形如“http://host:port/xxx/xxx/xxx”的形式
Part1:
param1
Part2:
param2
Part3:
file:要上传的文件
第三种:形如“http://host:port/xxx/xxx/xxx”的形式
Part1:
param1
Part2:
param2
Part3:
file:要上传的文件
利用Socket方法来发送
三种方法,首先搞得我头晕脑胀,纠结了整整一天,后来终于理顺了一些,现在把这些方法陈列出来,大家来批评指正,有不对的地方及时通知我来进行修改,大家共勉。
第一种和第二种都是模拟HTTP协议来发送的,使用HttpURLConnection类来发送,第三种使用Socket方法来发送。
首先我定义了一个上传工具类:UploadUtil.java,里面只是定义了一个Static的服务器地址,INTERNAL_HOST。
代码如下
public class UploadUtil { static String INTERNAL_HOST = "http://host:port/xxx/xxx/xxx/upload/"; }
完成第一种方法:
第一种方法的“http://host:port/xxx/xxx/xxx/param?param1=xxx¶m2=xxx”,其实已经给出了POST的参数,param1和param2,我为了弄着方便,就直接写死了参数键值。其实也可以在方法内部,通过传入的参数,进行循环拼装,最后只要能组出类似“http://host:port/xxx/xxx/xxx/param?param1=xxx¶m2=xxx”的字符串就行,最后通过URL来生成一个url对象即可。
public static void queryParam(String fileName) { String BOUNDARY = "---------------------------7db1c523809b2";//数据分割线 File file = new File(fileName); // 要上传的文件 String host = INTERNAL_HOST + "param?param1=xxx¶m2=xxx"; // 这个字符串就是要上传的带参数的服务器地址 try { byte[] after = ("--" + BOUNDARY + "--\r\n").getBytes("UTF-8"); // 构造URL和Connection URL url = new URL(host); HttpURLConnection conn = (HttpURLConnection)url.openConnection(); // 设置HTTP协议的头属性 conn.setRequestMethod("POST"); conn.setRequestProperty("Content-Type", "multipart/form-data; boundary=" + BOUNDARY); conn.setRequestProperty("Content-Length", String.valueOf(file.length())); conn.setRequestProperty("HOST", url.getHost()); conn.setDoOutput(true); // 得到Connection的OutputStream流,准备写数据 OutputStream out = conn.getOutputStream(); InputStream in = new FileInputStream(file); // 写文件数据。因为服务器地址已经带有参数了,所以这里只要直接写入文件部分就可以了。 byte[] buf = new byte[1024]; int len; while ((len = in.read(buf)) != -1) { out.write(buf, 0, len); } // 数据结束标志,整个HTTP报文就构造结束了。 //out.write(after); in.close(); out.close(); Log.d("carter", "queryParam 返回码为: " + conn.getResponseCode()); Log.d("carter", "queryParam 返回信息为: " + conn.getResponseMessage()); } catch (MalformedURLException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } }
大概的步骤:
1.先构造好URL服务器地址
2.初始化一个URL对象,得到里面的HttpURLConnection对象
3.设置connection对象的头部信息,包括POST方法、HOST、Content-Type、Content-Length等属性
4.写入要上传的文件内容
完成第二种方法:
第二种的方法,不想第一种方法那样简单,但基本语法结构都相似,因为都是模仿HTTP协议的形式。
第二种方法,在服务器地址上,只提供了上传的HOST和PATH,具体的参数没有提供,所以要通过在HTTP报文中添加来实现。
public static void multiPart(String fileName) { String BOUNDARY = "---------------------------7db1c523809b2";//数据分割线 File file = new File(fileName); // 要上传的文件 // 构造param参数部分的数据内容,格式都是相同的,依次添加param1和param2 StringBuilder sb = new StringBuilder(); sb.append("--" + BOUNDARY + "\r\n"); sb.append("Content-Disposition: form-data; name=\"param1\"" + "\r\n"); sb.append("\r\n"); sb.append("xxx" + "\r\n"); sb.append("--" + BOUNDARY + "\r\n"); sb.append("Content-Disposition: form-data; name=\"param2\"" + "\r\n"); sb.append("\r\n"); sb.append("xxx" + "\r\n"); // 构造要上传文件的前段参数内容,和普通参数一样,在这些设置后就可以紧跟文件内容了。 sb.append("--" + BOUNDARY + "\r\n"); sb.append("Content-Disposition: form-data; name=\"data\"; filename=\"" + fileName + "\"" + "\r\n"); sb.append("Content-Type: text/plain" + "\r\n"); sb.append("\r\n"); try { byte[] before = sb.toString().getBytes("UTF-8"); byte[] after = ("--" + BOUNDARY + "--\r\n").getBytes("UTF-8"); URL url = new URL(INTERNAL_HOST); // 得到HttpURLConnection对象,设置一些头信息基本属性 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", url.getHost()); 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); } // 写结束符,代表该HTTP组包完毕 out.write(after); // 发送出去 out.flush(); // 关闭流 in.close(); out.close(); Log.d("carter", "multipart 返回码为: " + conn.getResponseCode()); Log.d("carter", "multipart 返回信息为: " + conn.getResponseMessage()); } catch (MalformedURLException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (UnsupportedEncodingException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } }
1.分别把param1等参数通过格式组织起来
2.得到HttpURLConnection对象,设置一些基本头属性,其中的Content-Length是所有param和文件加上尾标志的总长度。
3.发送outputStream。
完成第三种方法:
第三种方法和第二种方法很相似,只是使用了Socket对象,而不是HttpURLConnection方法。
public static void socket(String fileName) { String BOUNDARY = "---------------------------7db1c523809b2";//数据分割线 File file = new File(fileName); // 要上传的文件 // 构造参数内容字符串 StringBuilder textParam = new StringBuilder(); textParam.append("--" + BOUNDARY + "\r\n"); textParam.append("Content-Disposition: form-data; name=\"param1\"" + "\r\n"); textParam.append("\r\n"); textParam.append("xxx" + "\r\n"); textParam.append("--" + BOUNDARY + "\r\n"); textParam.append("Content-Disposition: form-data; name=\"param2\"" + "\r\n"); textParam.append("\r\n"); textParam.append("xxx" + "\r\n"); // 构造文件内容字符串 int fileDataLen = 0; // 文件内容的长度 StringBuilder fileParam = new StringBuilder(); fileParam.append("--" + BOUNDARY + "\r\n"); fileParam.append("Content-Disposition: form-data;name=\"data\";" + "filename=\"" + fileName + "\"" + "\r\n"); fileParam.append("Content-Type: text/plain" + "\r\n\r\n"); fileParam.append("\r\n"); // 得到文件内容前的声明信息长度 fileDataLen += fileParam.length(); // 得到声明信息和文件内容总长度 fileDataLen += file.length(); // HTTP报文数据总长度 int totalDataLen = 0; try { byte[] textEntity = textParam.toString().getBytes(); byte[] after = ("--" + BOUNDARY + "--\r\n").getBytes(); // 得到报文内数据总长度,参数声明信息+文件信息+报文尾标志信息 totalDataLen = textEntity.length + fileDataLen + after.length; // 得到Socket URL url = new URL(INTERNAL_HOST); Socket socket = new Socket(url.getHost(), 80); socket.setSoTimeout(60000); // 设置Socket超时时间 // 得到输出流,靠它去发送报文 OutputStream out = socket.getOutputStream(); // 设置请求方法 String requestMethod = "POST " + url.getPath() + " HTTP/1.1\r\n"; out.write(requestMethod.getBytes()); // 设置接收类型 String accept = "Accept: image/gif, image/jpeg, 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"; out.write(accept.getBytes()); // 设置语言 String language = "Accept-Language: zh-CN\r\n"; out.write(language.getBytes()); // 设置内容类型 String contentType = "Content-Type: multipart/form-data; boundary=" + BOUNDARY + "\r\n"; out.write(contentType.getBytes()); // 设置报文长度 String contentLength = "Content-Length: " + totalDataLen + "\r\n"; out.write(contentLength.getBytes()); // 设置活动 String alive = "Connection: Keep-Alive\r\n"; out.write(alive.getBytes()); // 设置主机 String host = "Host: " + url.getHost() + ":80\r\n"; out.write(host.getBytes()); // 设置一个回车换行 out.write("\r\n".getBytes()); // 添加所有文本类型数据 out.write(textEntity); // 添加所有文件类型数据 StringBuilder fileEntity = new StringBuilder(); fileEntity.append("--" + BOUNDARY + "\r\n"); fileEntity.append("Content-Disposition: form-data;name=\"data\";" + "filename=\"" + fileName + "\"" + "\r\n"); fileEntity.append("Content-Type: text/plain" + "\r\n\r\n"); out.write(fileEntity.toString().getBytes()); // 将文件内容写入报文 FileInputStream fis = new FileInputStream(file); // 以后记得加null判断 byte[] buffer = new byte[1024]; int len = 0; while( (len=fis.read(buffer))!=-1 ) { out.write(buffer, 0, len); } fis.close(); out.write("\r\n".getBytes()); // 数据结束标志,代表数据已经结束 out.write(after); // 输出处理结果 BufferedReader reader = new BufferedReader(new InputStreamReader(socket.getInputStream())); String line = ""; while( (line=reader.readLine())!=null) { Log.d("carter", line); } // 发送报文 out.flush(); // 关闭流 out.close(); reader.close(); socket.close(); } catch (MalformedURLException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (UnsupportedEncodingException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } }
1.构造参数信息
2.构造文件信息
3.构造尾信息
4.通过URL和端口得到一个Socket对象
5.设置Socket的属性
6.得到一个输出流,将所有信息输出
7.输出传送结果
至此,三种方法就都实现了,纠结的内容,但是需要积极的去实现。
以上方法,只要替换其中的某些参数信息,就可以完全直接复用了。