[英]Upload file using both Multipart and JSON Key value pairs with Retrofit2
目前,我們正在通過使用簡單的 JSON 轉換String bytes
來加載文件(視頻、音頻、文本等),其中包括一些其他值及其鍵值對。 就像下面一樣:
一些標頭值:
{
"header": {
"geoDate": {
"point": {
"longitude": 77.56246948242188,
"latitude": 12.928763389587403
},
"date": "2020-02-25T18:26:00Z"
},
"version": "1.35.00.001",
"businessId": "178"
}
}
和文件信息:
JSONObject data = new JSONObject();
data.put("name", params.name);
data.put("mimeType", params.mimeType);
data.put("fileSize", params.fileSize);
data.put("inputData", params.data);
requestJSON.put("data", data);
這里params.data
是一個簡單的字節轉換String bytes = Base64.encodeToString(bos.toByteArray(), Base64.DEFAULT);
它正在工作,但我們希望通過將文件通過 MultiPart 發送到服務器來通過 Retrofit 來實現,這也將提高性能。 但問題在於它在 JSON 結構中,服務器無法更改它的程序,我們(應用程序)只需要做一些使用 Retrofit Multipart 發送文件的事情,包括其他值和鍵( inputData
)。
我正在尋找一種方法來做到這一點。 而且我想知道我們是否也可以發送,服務器是否必須更改 API 結構的任何內容,例如當前它接受的字節字符串,我們要將其更改為用於inputData
文件。
對我來說很好用(這是我的代碼,只需使其適應您的業務邏輯):
界面:
@Multipart
@POST("{projectName}/log")
Call<LogRp> uploadFile(
@Path("projectName") String project,
@PartMap Map<String, RequestBody> mp,
@Part MultipartBody.Part file
);
服務:
private MultipartBody.Part buildFilePart(File file, FileType type) {
return MultipartBody.Part.createFormData("file", file.getName(),
RequestBody.create(MediaType.parse(type.value.get()), file));
}
private Map<String, RequestBody> buildJsonPart(LogRq logRq) throws JsonProcessingException {
return Collections.singletonMap("json_request_part", RequestBody.create(
MediaType.parse("application/json"),
new ObjectMapper().writeValueAsString(logRq))
);
}
然后簡單地:
client.uploadFile(
project,
buildJsonPart(logRq),
buildFilePart(file, type)
)
LogRp 和 LogRq 是響應和請求 POJO。 如果需要幫助,請聯系我。
您可以使用 VolleyMultipartRequest 上傳帶有文本的文件。
VolleyMultipartRequest multipartRequest = new VolleyMultipartRequest(Request.Method.PUT, url, new Response.Listener<NetworkResponse>() {
@Override
public void onResponse(NetworkResponse response) {
String resultResponse = new String(response.data);
try {
JSONObject result = new JSONObject(resultResponse);
} catch (JSONException e) {
e.printStackTrace();
}
}
}, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError error) {
NetworkResponse networkResponse = error.networkResponse;
String errorMessage = "Unknown error";
if (networkResponse == null) {
if (error.getClass().equals(TimeoutError.class)) {
errorMessage = "Request timeout";
} else if (error.getClass().equals(NoConnectionError.class)) {
errorMessage = "Failed to connect server";
}
} else {
String result = new String(networkResponse.data);
try {
JSONObject response = new JSONObject(result);
String status = response.getString("status");
String message = response.getString("message");
Log.e("Error Status", status);
Log.e("Error Message", message);
if (networkResponse.statusCode == 404) {
errorMessage = "Resource not found";
} else if (networkResponse.statusCode == 401) {
errorMessage = message + " Please login again";
} else if (networkResponse.statusCode == 400) {
errorMessage = message + " Check your inputs";
} else if (networkResponse.statusCode == 500) {
errorMessage = message + " Something is getting wrong";
}
} catch (JSONException e) {
e.printStackTrace();
}
}
Log.i("Error", errorMessage);
error.printStackTrace();
}
}) {
@Override
protected Map<String, String> getParams() {
Map<String, String> params = new HashMap<>();
params.put("Some_text", "text");
return params;
}
@Override
protected Map<String, DataPart> getByteData() {
Map<String, DataPart> params = new HashMap<>();
// file name could found file base or direct access from real path
// for now just get bitmap data from ImageView
params.put("pofile_pic", new DataPart("file_avatar.jpg", getFileDataFromDrawable(bitmap), "image/jpeg"));
return params;
}
};
VolleySingleton.getInstance(getActivity()).addToRequestQueue(multipartRequest);
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.