【发布时间】:2019-05-15 19:15:22
【问题描述】:
实际上我在我的应用程序中使用 Retrofit2 向我的服务器发送一些文本。
问题是当我在网络中断时使用射频设备时,我正在 onFailure 呼叫应该被取消,但是一旦设备重新连接到网络,它似乎正在重新发送文本或类似的东西该文本仍在队列中,我会处理它。
这是我发送文本的方法(我还尝试在 onFailure 中使用 call.cancel() 但没有效果)
@SuppressLint("DefaultLocale")
public void sendPost() {
@SuppressLint({"SdCardPath", "DefaultLocale"}) final File file = new File("/data/data/app/files/"+String.format("%03d", Integer.valueOf(nTerminalino))+"_"+nTavoli.getText().toString()+".txt");
MultipartBody.Builder builder = new MultipartBody.Builder();
builder.setType(MultipartBody.FORM);
RequestBody fbody = RequestBody.create(MediaType.parse("text/*"), file);
builder.addFormDataPart(String.format("%03d", Integer.valueOf(nTerminalino))+"_"+nTavoli.getText().toString(), file.getName(),fbody);
MultipartBody requestBody = builder.build();
final APIService apiService = ApiUtils.getAPIService(ipCASSA);
final Call<Void> calls = apiService.savePost(requestBody);
calls.enqueue(new Callback<Void>() {
@Override
public void onResponse(@NonNull Call<Void> call, @NonNull Response<Void> response) {
if (response.isSuccessful()) {
Log.i("RESPONSE: ", response.toString());
Print();
file.delete();
dialogLoading.dismiss();
Runtime.getRuntime().gc();
}else {
calls.cancel();
}
}
@Override
public void onFailure(@NonNull Call<Void> call, @NonNull Throwable t) {
Log.e("TAG", t.toString());
MediaPlayer mpFound = MediaPlayer.create(pterm.this, R.raw.errorsound);
mpFound.start();
Vibrator v = (Vibrator) getSystemService(Context.VIBRATOR_SERVICE);
if (v.hasVibrator()) {
v.vibrate(6000);
}
new GlideToast.makeToast(pterm.this, "CONNESSIONE FALLITA!", GlideToast.LENGTHLONG, GlideToast.FAILTOAST).show();
dialogLoading.dismiss();
}
});
}
在我正在构建 OkHttpClient 的班级中,我设置了 retryOnConnectionFailure(false) 但仍然无法解决即使 onFailure 我也收到 POST 请求的问题。
更新:
经过一番研究,我发现 OkHttp 会在连接失败时静默重试 POST。
那么如何防止客户端重试呢?
一些信息:
我的应用程序是服务员的应用程序,所以当我收到 onFailure 消息时,我通知服务员收据尚未被 POS 接收且尚未打印,但 OkHttpClient 仍然重试发送文件,我强迫服务员在onFailure 之后重新发送收据,这会导致打印重复的收据。
这是我的 ApiUtils 类、APIService 接口和 RetrofitClient 类
class ApiUtils {
private ApiUtils() {}
static APIService getAPIService(String ipCASSA) {
String BASE_URL = "http://"+ipCASSA+"/web/";
final OkHttpClient okHttpClient = new OkHttpClient.Builder()
.connectTimeout(10000, TimeUnit.MILLISECONDS)
.writeTimeout(10000, TimeUnit.MILLISECONDS)
.readTimeout(10000, TimeUnit.MILLISECONDS)
.retryOnConnectionFailure(false)
.build();
return RetrofitClient.getClient(BASE_URL,okHttpClient).create(APIService.class);
}
}
public interface APIService {
@POST("CART=PTERM")
Call<Void> savePost(@Body RequestBody text);
}
class RetrofitClient {
private static Retrofit retrofit = null;
static Retrofit getClient(String baseUrl, OkHttpClient okHttpClient) {
if (retrofit==null) {
retrofit = new Retrofit.Builder()
.baseUrl(baseUrl)
.client(okHttpClient)
.addConverterFactory(GsonConverterFactory.create())
.build();
}
return retrofit;
}
}
【问题讨论】:
-
究竟是什么行为让您认为请求被发送了两次?你只在你的服务器上看到它吗?您是否看到
onFailure,当网络重新连接时您是否看到onSuccess? -
@JanosBreuer 实际上我得到了类似于此处medium.com/inloopx/… 在“解决真正的问题”点中描述的内容我正在发送网络中断的文件,所以我没有得到响应,但是文件已发送
标签: android http retrofit retrofit2