Maison >Java >javaDidacticiel >Comment faire des requêtes POST en plusieurs parties avec Volley sans utiliser HttpEntity ?
Volley est une bibliothèque Android populaire pour effectuer des requêtes HTTP. Dans les versions antérieures de l'API, HttpEntity était utilisé conjointement avec Volley pour la soumission de données de formulaire en plusieurs parties. Cependant, avec la dépréciation de HttpEntity dans l'API 22 et sa suppression complète dans l'API 23, les développeurs se retrouvent face à un défi.
Dans cet article, nous présenterons une solution efficace pour effectuer des requêtes POST en plusieurs parties sans utiliser de HttpEntité. Le code fourni vous permet de télécharger plusieurs fichiers avec des données texte.
L'implémentation présentée ici se compose de deux classes : MultipartActivity et MultipartRequest. MultipartActivity gère la préparation des données de formulaire en plusieurs parties, tandis que MultipartRequest étend la classe Request de Volley et remplace les méthodes nécessaires pour gérer le corps en plusieurs parties et traiter la réponse du serveur.
Pour utiliser cette solution, suivez ces étapes :
MultipartActivity.java:
import android.app.Activity; import android.content.Context; import android.os.Bundle; import android.widget.Toast; import com.android.volley.NetworkResponse; import com.android.volley.Response; import com.android.volley.VolleyError; import com.example.multipartvolley.MultipartRequest; import com.example.multipartvolley.VolleySingleton; import java.util.HashMap; import java.util.Map; public class MultipartActivity extends Activity { private Context context = this; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // Assuming you have prepared file data in fileData1 and fileData2 String url = "http://192.168.1.100/api/postfile"; MultipartRequest multipartRequest = new MultipartRequest(url, null, "multipart/form-data", multipartBody, new Response.Listener<NetworkResponse>() { @Override public void onResponse(NetworkResponse response) { Toast.makeText(context, "Upload successfully!", Toast.LENGTH_SHORT).show(); } }, new Response.ErrorListener() { @Override public void onErrorResponse(VolleyError error) { Toast.makeText(context, "Upload failed!\r\n" + error.toString(), Toast.LENGTH_SHORT).show(); } }) { @Override protected Map<String, String> getParams() { Map<String, String> params = new HashMap<>(); params.put("text_field1", "Value for text field 1"); params.put("text_field2", "Value for text field 2"); return params; } @Override protected Map<String, DataPart> getByteData() { Map<String, DataPart> params = new HashMap<>(); params.put("file_name1", new DataPart("file_name1.txt", "file content 1".getBytes(), "text/plain")); params.put("file_name2", new DataPart("file_name2.png", fileData1, "image/png")); return params; } }; VolleySingleton.getInstance(context).addToRequestQueue(multipartRequest); } }
MultipartRequest.java:
import com.android.volley.AuthFailureError; import com.android.volley.NetworkResponse; import com.android.volley.ParseError; import com.android.volley.Request; import com.android.volley.Response; import com.android.volley.VolleyError; import com.android.volley.toolbox.HttpHeaderParser; import java.util.HashMap; import java.util.Map; class MultipartRequest extends Request<NetworkResponse> { private final Response.Listener<NetworkResponse> mListener; private final Response.ErrorListener mErrorListener; private Map<String, String> mHeaders; private Map<String, DataPart> mByteData; MultipartRequest(String url, Map<String, String> headers, String contentType, Map<String, DataPart> byteData, Response.Listener<NetworkResponse> listener, Response.ErrorListener errorListener) { super(Method.POST, url, errorListener); this.mListener = listener; this.mErrorListener = errorListener; this.mHeaders = headers; this.mByteData = byteData; } @Override public Map<String, String> getHeaders() throws AuthFailureError { return mHeaders != null ? mHeaders : super.getHeaders(); } @Override public String getBodyContentType() { return "multipart/form-data; boundary=" + getBoundary(); } @Override public byte[] getBody() throws AuthFailureError { return encodeMultipartData(mByteData, getBoundary()); } private String getBoundary() { return Long.toHexString(System.currentTimeMillis()); } @Override protected Response<NetworkResponse> parseNetworkResponse(NetworkResponse response) { try { return Response.success( response, HttpHeaderParser.parseCacheHeaders(response)); } catch (Exception e) { return Response.error(new ParseError(e)); } } @Override protected void deliverResponse(NetworkResponse response) { mListener.onResponse(response); } @Override public void deliverError(VolleyError error) { mErrorListener.onErrorResponse(error); } protected static byte[] encodeMultipartData(Map<String, DataPart> dataParts, String boundary) { ByteArrayOutputStream bos = new ByteArrayOutputStream(); DataOutputStream dos = new DataOutputStream(bos); try { for (Map.Entry<String, DataPart> entry : dataParts.entrySet()) { dos.writeBytes(twoHyphens + boundary + lineEnd); dos.writeBytes("Content-Disposition: form-data; name=\"" + entry.getKey() + "\"" + "; filename=\"" + entry.getValue().getFileName() + "\"" + lineEnd); dos.writeBytes(String.format("Content-Type: %s%s", entry.getValue().getType(), lineEnd)); dos.writeBytes(lineEnd); dos.write(entry.getValue().getContent()); dos.writeBytes(lineEnd); } dos.writeBytes(lineEnd); dos.writeBytes(twoHyphens + boundary + twoHyphens + lineEnd); return bos.toByteArray(); } catch (IOException e) { e.printStackTrace(); return null; } } class DataPart { private String fileName; private byte[] content; private String type; DataPart(String fileName, byte[] content, String type) { this.fileName = fileName; this.content = content; this.type = type; } String getFileName() { return fileName; } byte[] getContent() { return content; } String getType() { return type; } } }
Ce code gère les formulaires en plusieurs parties requêtes de données où vous pouvez transmettre en toute sécurité les paramètres de fichier et de texte au serveur. Il est essentiel de noter que le code final n'est pas destiné à une utilisation en production et doit être modifié en fonction de vos besoins spécifiques.
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!