Как заставить асинхронную задачу выполняться повторно через некоторый интервал времени, как и таймер... На самом деле я разрабатываю приложение, которое будет автоматически загружать все последние непрочитанные приветствия с сервера, и для этой цели я должен проверять наличие обновлений с сервера после некоторого фиксированного временные интервалы... Я знаю, что это можно легко сделать с помощью таймера, но я хочу использовать асинхронную задачу, которая, как мне кажется, более эффективна для приложений Android.
Как повторно выполнять асинхронную задачу через фиксированные промежутки времени
Ответы (5)
Вы можете просто обработчик:
private int m_interval = 5000; // 5 seconds by default, can be changed later
private Handle m_handler;
@Override
protected void onCreate(Bundle bundle)
{
...
m_handler = new Handler();
}
Runnable m_statusChecker = new Runnable()
{
@Override
public void run() {
updateStatus(); //this function can change value of m_interval.
m_handler.postDelayed(m_statusChecker, m_interval);
}
}
void startRepeatingTask()
{
m_statusChecker.run();
}
void stopRepeatingTask()
{
m_handler.removeCallback(m_statusChecker);
}
Но я бы рекомендовал вам проверить эту структуру: http://code.google.com/intl/de-DE/android/c2dm/ Другой подход: сервер уведомит телефон, когда что-то будет готово (таким образом, сэкономив пропускную способность и производительность :))
не было бы более эффективно создать службу и запланировать ее через Alarm Manager?
Принятый ответ проблематичен. Использование TimerTask() для активации асинхронной задачи через обработчик — плохая идея. при изменении ориентации вы должны не забыть также отменить вызовы таймера и обработчика. в противном случае он будет вызывать асинхронную задачу снова и снова при каждом вращении. Это приведет к тому, что приложение взорвет сервер (если это остаточный http-запрос на получение) вместо времени X - в конечном итоге вызовы будут вызывать много вызовов каждую секунду. (потому что будет много таймеров по количеству поворотов экрана). Это может привести к сбою приложения, если активность и задача, выполняемая в фоновом потоке, тяжелы. если вы используете таймер, сделайте его мембером класса и отмените его onStop():
TimerTask mDoAsynchronousTask;
@Override
public void onStop(){
super.onStop();
mDoAsynchronousTask.cancel();
mHandler.removeCallbacks(null);
...
}
public void callAsynchronousTask(final boolean stopTimer) {
Timer timer = new Timer();
mDoAsynchronousTask = new TimerTask() {
@Override
public void run() {
mHandler.post(new Runnable() {
...
Вместо этого старайтесь избегать асинхронной задачи, и если вам необходимо использовать службу планировщика для запуска асинхронной задачи. или класс приложения, как в этой хорошей идее: https://fattybeagle.com/2011/02/15/android-asynctasks-during-a-screen-rotation-part-ii/
Или используйте простой обработчик (без таймера, просто используйте postDelayed), а также хорошей практикой является вызов отмены асинхронной задачи onStop(). этот код отлично работает с использованием postDelayed:
public class MainActivity extends AppCompatActivity {
MyAsync myAsync = new MyAsync();
private final Handler mSendSSLMessageHandler = new Handler();
private final Runnable mSendSSLRunnable = new Runnable(){
..
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ConnectivityManager connMgr = (ConnectivityManager)
getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo networkInfo = connMgr.getActiveNetworkInfo();
if (networkInfo != null && networkInfo.isConnected()) {
mSendSSLMessageHandler.post(mSendSSLRunnable);
}else
..
@Override
public void onStop(){
super.onStop();
if ( progressDialog!=null && progressDialog.isShowing() ){
progressDialog.dismiss();
}
mSendSSLMessageHandler.removeCallbacks(mSendSSLRunnable);
myAsync.cancel(false);
}
private final Runnable mSendSSLRunnable = new Runnable(){
@Override
public void run(){
try {
myAsync = new MyAsync();
myAsync.execute();
} catch (Exception e) {
// TODO Auto-generated catch block
}
mSendSSLMessageHandler.postDelayed(mSendSSLRunnable, 5000);
}
};
class MyAsync extends AsyncTask<Void, Void, String> {
boolean running = true;
@Override
protected void onPreExecute() {
super.onPreExecute();
progressDialog = ProgressDialog.show
(MainActivity.this, "downloading", "please wait");
}
@Override
protected String doInBackground(Void... voids) {
if (!running) {
return null;
}
String result = null;
try{
URL url = new URL("http://192...");
HttpURLConnection urlConnection = (HttpURLConnection)
url.openConnection();
InputStream in = new BufferedInputStream (urlConnection.getInputStream());
result = inputStreamToString(in);
}catch(Exception e){
e.printStackTrace();
}
return result;
}
@Override
protected void onCancelled() {
boolean running = false;
}
@Override
protected void onPostExecute(String s) {
super.onPostExecute(s);
progressDialog.dismiss();
try {
..
} catch (JSONException e) {
textView.append("json is invalid");
e.printStackTrace();
}
}
}
TimerTask
в onPause
, чтобы Task
не продолжал работать после выхода из активности?
- person theblang; 18.06.2014