1. 程式人生 > >裝置向後臺傳送檔案隨記

裝置向後臺傳送檔案隨記

一、安卓端,ios端向後臺post檔案(圖片,text或壓縮檔案)

我們做web開發的時候幾乎都是通過一個表單來實現上傳。並且是post的方式。而且都必須要加個引數enctype = “multipart/form-data”.然後再上傳後臺用各種框架裡的外掛之類的就可以接收了,並沒有關心過這個檔案具體是怎麼傳的。現在用android開發 沒有那些框架了,所以不得不關心一下了。

其實我們這種前後臺的互動是用的HTTP協議。而http協議預設是傳的字串。所以我們上傳檔案的話要加enctype = "multipart/form-data"這個引數來說明我們這傳的是檔案不是字串了。而我們做web開發的時候,瀏覽器是自動解析HTTP協議的。裡面傳的哪些東西我們不用管。只要記住幾個引數就行。而我們要上傳的檔案報文是儲存在請求的標頭檔案裡面的。下面就是上傳檔案標頭檔案的格式:

在這裡插入圖片描述 在這裡插入圖片描述 在這裡插入圖片描述

紅色字型部分就是協議的頭。給伺服器上傳資料時,並非協議頭每個欄位都得說明,其中,content-type是必須的,它包括一個類似標誌性質的名為boundary的標誌,它可以是隨便輸入的字串。對後面的具體內容也是必須的。它用來分辨一段內容的開始。Content-Length: 3693 ,這裡的3693是要上傳檔案的總長度。綠色字型部分就是需要上傳的資料,可以是文字,也可以是圖片等。資料內容前面需要有Content-Disposition, Content-Type以及Content-Transfer-Encoding等說明欄位。最後的紫色部分就是協議的結尾了。   注意這一行:   Content-Type: multipart/form-data; boundary=---------------------------7db372eb000e2     根據 rfc1867, multipart/form-data是必須的.   ---------------------------7db372eb000e2 是分隔符,分隔多個檔案、表單項。其中b372eb000e2 是即時生成的一個數字,用以確保整個分隔符不會在檔案或表單項的內容中出現。Form每個部分用分隔符分割,分隔符之前必須加上"–“著兩個字元(即–{boundary})才能被http協議認為是Form的分隔符,表示結束的話用在正確的分隔符後面新增”–"表示結束。   前面的 ---------------------------7d 是 IE 特有的標誌,Mozila 為---------------------------71.   每個分隔的資料的都可以用Content-Type來表示下面資料的型別,可以參考rfc1341 (

http://www.ietf.org/rfc/rfc1341.txt)

原理就是讓客戶端模仿 web網頁的表單提交檔案的方式 傳送檔案   客戶端的程式碼Demo

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; } }

伺服器端(Node JS伺服器)

//需要引用這個模組
var multiparty = require('multiparty');

 async uploadLog(req, res, next) {
        try{
            var form = new multiparty.Form();
            form.parse(req, function(err, fields, files) {
                var iconFile = files.myfile[0];
                if(iconFile.size !== 0){
                    fs.readFile(iconFile.path, function(err, data){
                        if(err) {
                            return res.json({'status':'failed',
                                              'message':'讀取檔案失敗'});
                        }
                        //接下來對檔案進行儲存操作
                        儲存道leancloud伺服器
                        var id = iconFile.originalFilename; //檔案的原名字
                        var theFile = new AV.File(id , data);
                        // console.log(theFile+'------------Log File Upload------------');
                        theFile.set('_type', 'logData');
                        const update = new AV.Object('DevicelogFile');
                        update.set('id', id);
                        update.set('file', theFile);
                        update.set('type', 'zip');
                        update.save().then(result => {
                            console.log(update.file+'------------Log File Upload------------');
                            return res.json({'status':'success',
                                 'message':'上傳檔案成功'});
                        })
                    });
                } else {
                            return res.json({'status':'failed',
                                             'message':'上傳檔案失敗'});
                }
            });
        } catch (error) {
            next(error);
        }
    }

二、讓陣列資料在HTML頁面一行一行的展示

HTML端只需要設定一個div

 <div id="showurl" class="showurl"></div>

script中的函式

 for (let i = 0;i<success.message.length; i++){
                        var j = i+1;
                        div += "<p>最新的Log檔案"+j+"------------"+success.message[i]+"</p>";
                    }
                    $("#showurl").html(div);
                },