阿里云產(chǎn)品通用代金券,最高可領(lǐng)1888分享一波阿里云紅包. 阿里云的購(gòu)買入口
本例是我在年前的項(xiàng)目中用到的一個(gè)小例子赞哗,使用Android自帶的HttpURLConnection實(shí)現(xiàn)文件上傳磺樱。網(wǎng)上的資料對(duì)這方面的講解不太多什湘,有兩個(gè)實(shí)例也不太詳細(xì)迁匠,我在開發(fā)中用到了愿伴,覺得挺實(shí)用的柬帕,分享出來产场,希望能對(duì)各位朋友有所幫助唆姐,受水平所限拗慨,實(shí)例中如有謬處,還望各位大神批評(píng)指正奉芦,這個(gè)實(shí)例個(gè)人認(rèn)為適用于項(xiàng)目中的文件上傳比較少的情況赵抢,當(dāng)文件上傳比較多的時(shí)候還是用框架吧(例如okhtp)。
本來想在剛放寒假就整理一下声功,由于年前回家比較晚了烦却,年前年后的事有點(diǎn)多,直到現(xiàn)在才有時(shí)間整理先巴。廢話不多說 直接上代碼:(代碼是我從項(xiàng)目中剪出來的其爵,可能會(huì)有報(bào)錯(cuò),問題不會(huì)太大筹裕,很好改)
import android.os.Handler;
import android.os.Message;
import android.util.Log;
import org.json.JSONException;
import org.json.JSONObject;
import java.io.BufferedReader;
import java.io.DataOutputStream;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
public class ArrivateUpload extends Thread {
private final String BOUNDARYSTR = "--------aifudao7816510d1hq";
private final String END = "\r\n";
private final String LAST = "--";
private String data;//表單數(shù)據(jù)
private FileInputStream fis;//文件輸入流
private Handler handler;
public ArrivateUpload(String data, FileInputStream fis, Handler handler) {
this.data = data;
this.handler = handler;
this.fis=fis;
}
@Override
public void run() {
try {
URL httpUrl=new URL(urlStr);
HttpURLConnection connection= (HttpURLConnection) httpUrl.openConnection();
connection.setRequestMethod("POST");//必須為post
connection.setDoInput(true);
connection.setDoOutput(true);
connection.setRequestProperty("Content-type", "multipart/form-data;boundary=" + BOUNDARYSTR);//固定格式
DataOutputStream dos=new DataOutputStream(connection.getOutputStream());
StringBuffer sb=new StringBuffer();
/**
* 寫入文本數(shù)據(jù)
*/
sb.append(LAST+BOUNDARYSTR+END);
sb.append("Content-Disposition: form-data; name=\"data\""+END+END);
sb.append(data+END);//內(nèi)容
/**
* 循環(huán)寫入文件
*/
sb.append(LAST+BOUNDARYSTR+END);
sb.append("Content-Disposition:form-data;Content-Type:application/octet-stream;name=\"file\";");
sb.append("filename=\""+"map_image.png"+"\""+END+END);
dos.write(sb.toString().getBytes("utf-8"));
if (fis != null) {
byte[] b=new byte[1024];
int len;
while ((len=fis.read(b))!=-1){
dos.write(b,0,len);
}
dos.write(END.getBytes());
}
dos.write((LAST+BOUNDARYSTR+LAST+END).getBytes());
dos.flush();
sb=new StringBuffer();
if (connection.getResponseCode()==200) {//請(qǐng)求成功
BufferedReader br=new BufferedReader(new InputStreamReader(connection.getInputStream()));
String line;
while ((line=br.readLine())!=null){
sb.append(line);
}
Message msg=Message.obtain();
JSONObject object=new JSONObject(sb.toString());
handler.sendMessage(msg);
}
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} catch (JSONException e) {
e.printStackTrace();
}
}
}
說一下實(shí)現(xiàn)思路:用Android系統(tǒng)中提供的HttpURLConnection醋闭,利用http中的multipart協(xié)議實(shí)現(xiàn)文件上傳。httpUrlConnection應(yīng)該不用多說了(不太熟悉的同學(xué)去Google學(xué)習(xí)一下HttpURLConnection和http協(xié)議吧)朝卒,就說幾個(gè)注意點(diǎn)吧证逻,請(qǐng)求模式設(shè)置為POST,打開輸入輸出流(這個(gè)好像默認(rèn)是開啟的抗斤,還是設(shè)置為true比較放心)
重點(diǎn)說一下multipart協(xié)議吧囚企,因?yàn)槠胀ǖ膆ttp post協(xié)議使用的分隔符太簡(jiǎn)單,上傳文件會(huì)造成文件分割歧義瑞眼,所以必須使用multipart協(xié)議龙宏。 寫一個(gè)簡(jiǎn)單的HTML表單文件上傳的例子,對(duì)照學(xué)習(xí)伤疙。
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>upload</title>
</head>
<body>
<form action="a.php" method="post" enctype="multipart/form-data">
<input type="text" name="data"/>
<input type="file" name="file">
<input type="submit">
</form>
</body>
</html>
在瀏覽器中通過調(diào)試我們可以對(duì)照瀏覽器來拼出multipart協(xié)議银酗,瀏覽器會(huì)報(bào)404,a.php 沒有找到辆影,不用管它,這不是我們關(guān)注的重點(diǎn)黍特。下圖就是瀏覽器中的詳細(xì)信息
在1處定義了分隔符蛙讥,分隔符盡量復(fù)雜一點(diǎn),避免造成與文件中的字符產(chǎn)生沖突灭衷。注意了次慢,前方有大坑,注意腳下油門翔曲,老司機(jī)一不留神也會(huì)翻車迫像。仔細(xì)對(duì)比你會(huì)發(fā)現(xiàn)2,3,4處和1處是不同的,2,3,4處比1處多了兩個(gè)“-”,一定要留神瞳遍,這也就是為什么我們?cè)谑褂胋oundary時(shí)要加上“--”的原因了闻妓。sb.append(LAST+BOUNDARYSTR+END); 在4處,整個(gè)body的結(jié)束處要加上一個(gè)“--”傅蹂,沒有why纷闺,協(xié)議就是這樣規(guī)定的。
每一行結(jié)束都要有一個(gè)"\r\n"份蝴,第二行結(jié)束時(shí)(數(shù)據(jù)文件開始前)要有兩個(gè)"\r\n"。對(duì)比圖中會(huì)發(fā)現(xiàn)數(shù)據(jù)文件前有一個(gè)空行氓轰。還有非常重要的一點(diǎn)婚夫,千萬注意引號(hào)(“ ”)的轉(zhuǎn)義問題,如果引號(hào)出問題就會(huì)導(dǎo)致上傳失敗署鸡,而且這個(gè)錯(cuò)誤還不容易發(fā)現(xiàn)案糙。文件寫入需要用輸入流循環(huán)寫入
sb.append("Content-Disposition:form-data;Content-Type:application/octet-stream;name=\"file\";"); //name 之前是固定寫法, 這里的name是服務(wù)端接收時(shí)需要用到的靴庆,
sb.append("filename=\""+"map_image.png"+"\""+END+END); // 這里的filename的值隨便寫时捌,無所謂,但是一定要有炉抒,沒有會(huì)導(dǎo)致上傳失敗奢讨。
dos.write(sb.toString().getBytes("utf-8"));
if (fis != null) {
byte[] b=new byte[1024];
int len;
while ((len=fis.read(b))!=-1){
dos.write(b,0,len);
}
dos.write(END.getBytes());
}
dos.write((LAST+BOUNDARYSTR+LAST+END).getBytes());
dos.flush();
再分享一個(gè)調(diào)試技巧,我們的程序是跑在Android上的焰薄,我們是無法像在瀏覽器中那樣調(diào)試的拿诸,所以開抓包是非常必要的(genymation自帶抓包器,熟悉linux的可以直接使用)塞茅,在模擬器上運(yùn)行程序亩码,電腦開啟抓包器,每次上傳都抓取一個(gè)數(shù)據(jù)包野瘦,分析數(shù)據(jù)包描沟,如果出錯(cuò)了與瀏覽器的格式對(duì)比,這樣調(diào)試不要太酸爽。有次上傳的內(nèi)容有點(diǎn)復(fù)雜吏廉,一直調(diào)試不正確蠢络,最后用抓包對(duì)比搞定了。我用的是wireshark迟蜜,這個(gè)功能太強(qiáng)大了刹孔,直接在網(wǎng)卡安裝驅(qū)動(dòng),只要經(jīng)過網(wǎng)卡的數(shù)據(jù)都能抓到(想象力豐富的同學(xué)可以用它來干點(diǎn)壞事情 此處略去一千字……)娜睛。還有一個(gè)地方需要注意一下髓霞,wireshark不能抓ip為127.0.0.1的數(shù)據(jù)包(用本地電腦做服務(wù)器),因?yàn)檫@時(shí)的數(shù)據(jù)不經(jīng)過網(wǎng)卡畦戒,所以無法抓取方库。
由于水平有限,加之時(shí)間匆忙障斋,如有謬處纵潦,還請(qǐng)各位大神批評(píng)指正