編輯:關於Android編程
package com.example.photo;
import java.io.BufferedReader;
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.MalformedURLException;
import java.net.URL;
import java.util.UUID;
public class HttpAssist {
private static final String TAG = "uploadFile";
private static final int TIME_OUT = 10 * 10000000; // 超時時間
private static final String CHARSET = "utf-8"; // 設置編碼
public static final String SUCCESS = "1";
public static final String FAILURE = "0";
public static String uploadFile(File file) {
String BOUNDARY = UUID.randomUUID().toString(); // 邊界標識 隨機生成
String PREFIX = "--", LINE_END = "\r\n";
String CONTENT_TYPE = "multipart/form-data"; // 內容類型
String RequestURL = "http://192.168.0.100:7080/YkyPhoneService/Uploadfile1";
try {
URL url = new URL(RequestURL);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setReadTimeout(TIME_OUT);
conn.setConnectTimeout(TIME_OUT);
conn.setDoInput(true); // 允許輸入流
conn.setDoOutput(true); // 允許輸出流
conn.setUseCaches(false); // 不允許使用緩存
conn.setRequestMethod("POST"); // 請求方式
conn.setRequestProperty("Charset", CHARSET); // 設置編碼
conn.setRequestProperty("connection", "keep-alive");
conn.setRequestProperty("Content-Type", CONTENT_TYPE + ";boundary="
+ BOUNDARY);
if (file != null) {
/**
* 當文件不為空,把文件包裝並且上傳
*/
OutputStream outputSteam = conn.getOutputStream();
DataOutputStream dos = new DataOutputStream(outputSteam);
StringBuffer sb = new StringBuffer();
sb.append(PREFIX);
sb.append(BOUNDARY);
sb.append(LINE_END);
/**
* 這裡重點注意: name裡面的值為服務器端需要key 只有這個key 才可以得到對應的文件
* filename是文件的名字,包含後綴名的 比如:abc.png
*/
sb.append("Content-Disposition: form-data; name=\"img\"; filename=\""
+ file.getName() + "\"" + LINE_END);
sb.append("Content-Type: application/octet-stream; charset="
+ CHARSET + LINE_END);
sb.append(LINE_END);
dos.write(sb.toString().getBytes());
InputStream is = new FileInputStream(file);
byte[] bytes = new byte[1024];
int len = 0;
while ((len = is.read(bytes)) != -1) {
dos.write(bytes, 0, len);
}
is.close();
dos.write(LINE_END.getBytes());
byte[] end_data = (PREFIX + BOUNDARY + PREFIX + LINE_END)
.getBytes();
dos.write(end_data);
dos.flush();
/**
* 獲取響應碼 200=成功 當響應成功,獲取響應的流
*/
int res = conn.getResponseCode();
if (res == 200) {
return SUCCESS;
}
}
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return FAILURE;
}
}
這是文件上傳的客戶端代碼。 認真讀一下代碼,應該能夠理解上傳文件的格式了。同樣寫好這個頭文件之後采用HttpURLConnection向後台發送。也就是浏覽器所用的http協議。我們只是把這個協議自己手動調用並且手動填寫頭文件內容。而不是通過浏覽器幫我們寫了。看這段代碼的時候我發現都是把字符串轉化成字節流,然後利用DataOutputStream這個類來想後台傳輸。圖片文件也是利用這個類向後台傳。不知道大家有沒有跟我想法一樣的覺得到後台之後通過Request獲取字節流,然後把流寫到相應格式的文件或者圖片裡就收到了。我覺得理論上應該是這樣的。而且網上也有一些相應的文件或者代碼。但是我拿來用的話卻都不成功,以下是一個例子(不能用。)
public class UploadFile extends HttpServlet {
static int i = 0;
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws IOException {
ServletInputStream in = request.getInputStream();
byte[] buffer = new byte[1024];
File file = new File(request.getSession().getServletContext().getRealPath("/img/"),"img_"+i+".jpg");
i++;
FileOutputStream out = new FileOutputStream(file);
int len = in.read(buffer, 0, 1024);
while( len!=-1){
out.write(buffer,0,len);
len = in.read(buffer, 0, 1024);
}
out.close();
in.close();
}
}
我覺得這幾行代碼跟我的想法很相似,但就是運行不成功。向後台上傳一個圖片之後也收到了,但就是打不開,提示文件損壞。而且我覺得這種思路是對的,於是就一直在找類似的實現方法或者,按自己的思路修改一下代碼,帶最終還是不行。所以只好用其他的方法。 服務器接收上傳的文件的方法最後是通過利用apache提供的兩個jar包來實現的。commons-fileupload.jar和commons-io.jar這倆jar包。在服務器端添加這倆包之後,寫一個Servlet來實現文件接收。直接上代碼:
public class Uploadfile1 extends HttpServlet {
@Override
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
request.setCharacterEncoding("utf-8");
//獲得磁盤文件條目工廠。
DiskFileItemFactory factory = new DiskFileItemFactory();
//獲取文件上傳需要保存的路徑,upload文件夾需存在。
String path = request.getSession().getServletContext().getRealPath("/upload");
//設置暫時存放文件的存儲室,這個存儲室可以和最終存儲文件的文件夾不同。因為當文件很大的話會占用過多內存所以設置存儲室。
factory.setRepository(new File(path));
//設置緩存的大小,當上傳文件的容量超過緩存時,就放到暫時存儲室。
factory.setSizeThreshold(1024*1024);
//上傳處理工具類(高水平API上傳處理?)
ServletFileUpload upload = new ServletFileUpload(factory);
try{
//調用 parseRequest(request)方法 獲得上傳文件 FileItem 的集合list 可實現多文件上傳。
List<FileItem> list = (List<FileItem>)upload.parseRequest(request);
for(FileItem item:list){
//獲取表單屬性名字。
String name = item.getFieldName();
//如果獲取的表單信息是普通的文本信息。即通過頁面表單形式傳遞來的字符串。
if(item.isFormField()){
//獲取用戶具體輸入的字符串,
String value = item.getString();
request.setAttribute(name, value);
}
//如果傳入的是非簡單字符串,而是圖片,音頻,視頻等二進制文件。
else{
//獲取路徑名
String value = item.getName();
//取到最後一個反斜槓。
int start = value.lastIndexOf("\\");
//截取上傳文件的 字符串名字。+1是去掉反斜槓。
String filename = value.substring(start+1);
request.setAttribute(name, filename);
/*第三方提供的方法直接寫到文件中。
* item.write(new File(path,filename));*/
//收到寫到接收的文件中。
OutputStream out = new FileOutputStream(new File(path,filename));
InputStream in = item.getInputStream();
int length = 0;
byte[] buf = new byte[1024];
System.out.println("獲取文件總量的容量:"+ item.getSize());
while((length = in.read(buf))!=-1){
out.write(buf,0,length);
}
in.close();
out.close();
}
}
}catch(Exception e){
e.printStackTrace();
}
}
}
代碼同樣轉自網絡,具體地址給忘了,這個真不是故意的。。。 代碼上有相應注釋,應該都能看懂。並且這個開源jar包提供的方法挺強大的,支持多文件上傳之類的。我猜它的源碼同樣是獲取客戶端傳過來的字節流。後面的代碼跟上面提供的思路一樣。只是不知道它的Item究竟是如何獲得這個字節流的。按著源碼看了看 也沒看太明白。以後再慢慢研究吧。有哪位明白的 還請指點。 這樣利用HTTP做到文件的上傳和接收都已經正確運行了。先留下個筆記,以後用的時候可以看看。
Android項目目錄結構分析
項目結構如下圖所示,使用不同sdk版本建立的項目項目結構有所不同,總體相同,高版本增加了一些包結構 1、src 和java項目一樣src存放項目源代碼 2、ge
java/android 設計模式學習筆記(9)---代理模式
這篇博客我們來介紹一下代理模式(Proxy Pattern),代理模式也成為委托模式,是一個非常重要的設計模式,不少設計模式也都會有代理模式的影子。代理在我們日常生活中也
Container之ListView 1
ListView作為一個列表展示的容器,裡面可以包含多個數據項,數據項可以是簡單的只有一個TextView的布局,也可以是復雜的組合布局。 繼承關系如下: 類 List
Android自動化構建之Ant多渠道打包實踐分析(下)
前言上一篇(Android自動化構建之Ant多渠道打包實踐分析(上))已經介紹了Android的apk是如何構建的,本篇博客繼續Ant打包的實踐過程。集成友盟統計SDK這