我有一个带有 httprequest 的应用程序。我想为该请求设置超时,并且由于我无法控制 de DNS 超时,stackoverflow 中的用户建议创建一个带有计时器的线程,以便在一定时间后取消请求,这就是我正在做的。
在我的新线程中,我使用 Thread.sleep 等待 30 秒,当该时间结束时,如果请求尚未结束,则取消请求,但我的线程永远不会醒来。这是我的代码:
private Thread timeout_request = new Thread(new Runnable() {
public void run() {
try{
Log.d(TAG, "Thread start");
Thread.sleep(30000);
Log.d(TAG, "Thread awake");
if(!httprequest_finished){
request_aborted = true;
mHttpClient.getConnectionManager().shutdown();
}
}catch(Exception e){
e.printStackTrace();
}
}
});
public String sendData(String token){
...
timeout_request.start();
Log.i(TAG, "Sending request");
final HttpResponse resp = httpClient.execute(post);
Log.i(TAG, "Execute finished");
if(request_aborted) return null;
httprequest_finished = true;
...
}
如果我在没有互联网连接的情况下执行 sendData 函数,日志会显示“发送请求”和“线程启动”,但不会显示“线程唤醒”或“执行完成”(我已经等了 5 分钟)。这是为什么?
- - - 编辑 - - -
顺便说一句,我不知道这是否重要,但我在连接中使用了 ThreadSafeClientConnManager。
我测试了一些东西:
1- 将 HttpResponse resp = httpClient.execute(post) 替换为 while(!request_aborted)。线程唤醒(它工作)。
2- 使用 mHandler.postDelayed(r, 30000) 而不是线程。我仍在执行 httpClient.execute(post)。结果:Runnable 未启动。
private Handler mHandler = new Handler();
private Runnable r = new Runnable() {
public void run() {
Log.d(TAG, "Runnable()");
if(!httprequest_finished){
request_aborted = true;
mHttpClient.getConnectionManager().shutdown();
}
}
};
3- 使用 mHandler.postDelayed(r, 30000) 而不是线程。将 HttpResponse resp = httpClient.execute(post) 替换为 while(!request_aborted)。结果:Runnable 未启动。
因此,如果我不执行 http 请求,线程就可以工作,但处理程序永远不会工作。我在日志中注意到这些行:
threadid=3: reacting to signal 3
Wrote stack traces to '/data/anr/traces.txt'
threadid=3: reacting to signal 3
Wrote stack traces to '/data/anr/traces.txt'
这会在执行线程或设置处理程序后 6 或 7 秒出现。我试过使用:
Thread.setDefaultUncaughtExceptionHandler(new Thread.UncaughtExceptionHandler() {
public void uncaughtException(Thread paramThread, Throwable paramThrowable) {
Log.e("Alert","Lets See if it Works !!!");
}
});
但是日志从不显示“让我们看看它是否有效!!!”。我查看了 traces.txt 文件,但我不明白它在说什么-.-