如何使用改造2将文件和其他参数一起发送

时间:2016-10-12 21:41:29

标签: android retrofit retrofit2 android-networking

我正在寻找一个例子,我如何将文件和其他参数一起发送到服务器。

我必须发送服务器JSON

{
    "title": "title",
    "file": "uploaded file instance",
    "location": {
        "lat": 48.8583,
        "lng": 2.29232,
        "place": "Eiffel Tower"
    }
}

我如何创建Retrofit来处理这种情况?

如果file是一个字符串,我知道如何处理它。如果file是File对象,我不知道如何做到这一点。

2 个答案:

答案 0 :(得分:5)

使用gson并为该位置创建一个模型类。

将以下依赖项添加到build.gradle

compile 'com.squareup.retrofit2:converter-gson:2.0.0'
compile 'com.google.code.gson:gson:2.5'

创建一个模型来表示位置。

public class Location {

    double lat;
    double lng;
    String location;

    public Location(double lat, double lon, String place) {
        this.lat = lat;
        this.lon = long;
        this.place = place;
    } 

}

如果有效负载字段的变量名称与端点的实际所需名称不匹配,则需要添加注释@SerializedName([expected name])

例如:

import com.google.gson.annotations.SerializedName;

public class Location {

    @SerializedName("lat")
    double latitude;
    @SerializedName("lng")
    double longitude;
    @SerializedName("place")
    String location;

    public Location(double lat, double lon, String place) {
        latitude = lat;
        longitude = long;
        location = place;
    } 

}

定义api接口。

public interface Api {

    @POST("upload/")
    @Multipart
    Call<ResponseBody> uploadFile(@Part("title") RequestBody title,
                                  @Part MultipartBody.Part imageFile,
                                  @Part("location") Location location
    );

}

创建一个Retrofit实例并调用api。

File file;
// create retrofit instance
Retrofit retrofit = new Retrofit.Builder()
        .baseUrl("https://baseurl.com/api/")
        .addConverterFactory(GsonConverterFactory.create())
        .build();
// create api instance
Api api = retrofit.create(Api.class);
// create call object
Call<ResponseBody> uploadFileCall = api.uploadFile(
        RequestBody.create(MediaType.parse("text/plain"), "title"),
        MultipartBody.Part.createFormData(
            "file",
            file.getName(),
            RequestBody.create(MediaType.parse("image"), file)),
        new Location(48.8583, 2.29232, "Eiffel Tower"));
// sync call
try {
    ResponseBody responseBody = uploadFileCall.execute().body();
} catch (IOException e) {
    e.printStackTrace();
}
// async call
uploadFileCall.enqueue(new Callback<ResponseBody>() {
    @Override
    public void onResponse(Call<ResponseBody> call, Response<ResponseBody> response) {
        if (response.isSuccessful()) {
            // TODO
        }
    }

    @Override
    public void onFailure(Call<ResponseBody> call, Throwable t) {
        // TODO
    }
});

如果您没有使用图片文件,则需要更改MediaType.parse()来电。

您可以类似地创建自定义响应类型对象,并用它替换ResponseBody以接收反序列化的结果对象。

如果有效,请告诉我。我没有机会在你的确切场景中进行测试,但我相信这应该有效。我不是100%开启的唯一部分是@Part("location") Location location@Body("location") Location location

答案 1 :(得分:2)

截止到02.2020 ,@Abtin Gramian的answer很不错,除了String requiredName = clients.get(scannedID, newclient.clientName);RequestBody.create()在科特林中已被弃用,必须使用:

MediaType.parse()

代替:

file.asRequestBody("image".toMediaTypeOrNull())

此外,我必须手动编写以下内容:

RequestBody.create(MediaType.parse("image"), file)