Como chegar XML usando AsyncTask e temporizador?
-
19-09-2019 - |
Pergunta
A fim de obter dados XML a partir de um servidor repetidamente, eu estou tentando usar AsyncTask e temporizador como por Mark Murphy sugestão .
Eu recebo o seguinte erro:
01-07 16:11:26.705: ERROR/AndroidRuntime(729):
Caused by: java.lang.RuntimeException:
Can't create handler inside thread that has not
called Looper.prepare()
Eu estou usando SDK 1.5 com Eclipse no Windows.
Eu olhei na documentação, em StackOverflow e no grupo Desenvolvedores Android, mas eu não estou claro o que está causando o erro ou como corrigi-lo.
Eu posso obter os dados de uma vez - ou seja, sem Async e Timer -. E analisá-lo através de SAX sem problemas
aplicativo completa código abaixo.
Por favor, desculpe quaisquer erros ingênuos:. Eu sou muito novo para Android
package com.foo.bar.myactivity;
import java.net.URL;
import java.util.Timer;
import java.util.TimerTask;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
import org.xml.sax.InputSource;
import org.xml.sax.XMLReader;
import android.app.Activity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
public class MyActivity extends Activity {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
Timer timer;
timer = new Timer();
timer.schedule(new MyTimerTask(), 0, 1000);
}
public class MyAsyncTask extends AsyncTask<String, Integer, MyData> {
protected MyData doInBackground(String... string) {
MyData myData = new MyData();
try {
URL url = new URL("http://www.example.com/my.xml");
SAXParserFactory spf = SAXParserFactory.newInstance();
SAXParser sp = spf.newSAXParser();
XMLReader xr = sp.getXMLReader();
MyHandler myHandler = new MyHandler();
xr.setContentHandler(myHandler);
System.setProperty("http.proxyHost", "www-cache.example.com");
System.setProperty("http.proxyPort", "80");
xr.parse(new InputSource(url.openStream()));
myData = myHandler.getParsedData();
return myData;
} catch (Exception e) {
Log.e(">>>>>>>>>>>> Error getting myData: ", e.getMessage(), e);
return myData;
}
}
protected void onProgressUpdate(Integer... progress) {
// setProgressPercent(progress[0]);
}
protected void onPostExecute(MyData myData) {
Log.d(">>>>>>>>>>>>>My data: ", myData.toString());
}
}
public class MyTimerTask extends TimerTask {
public void run() {
try {
new MyAsyncTask().execute("");
} catch (Exception e) {
Log.e(">>>>>>>>>>>> Error executing MyAsyncTask: ", e.getMessage(), e);
}
}
}
}
Solução
O problema está no uso de TimerTask. TimerTask corrida deve postar em um manipulador, algo como isto:
private Handler mHandler = new Handler();
public class MyTimerTask extends TimerTask {
public void run() {
mHandler.post(
new Runnable() {
public void run() {
new MyAsyncTask().execute("");
}
};
)
}
}
É claro que isso está ficando um pouco feio, por isso recomendo tomar a classe anônima.
Outras dicas
Eu tenho isso funcione graças a resposta de James.
Eu incluí o código abaixo no caso, é útil para qualquer pessoa.
desenvolvedor Aviso! O código funciona para mim, mas podem conter erros.
package com.example.test;
import java.net.URL;
import java.util.Timer;
import java.util.TimerTask;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
import org.xml.sax.InputSource;
import org.xml.sax.XMLReader;
import android.app.Activity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.os.Handler;
import android.util.Log;
public class MyXmlPoller extends Activity {
private Handler handler = new Handler();
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
new Timer().schedule(new MyTimerTask(), 0, 1000);
}
private class MyAsyncTask extends AsyncTask<Integer, Integer, MyData> {
protected MyData doInBackground(Integer... counter) {
MyData myData = new MyData();
try {
URL url = new URL("http://www.example.com/my.xml");
SAXParserFactory spf = SAXParserFactory.newInstance();
SAXParser sp = spf.newSAXParser();
XMLReader xr = sp.getXMLReader();
MySAXHandler mySAXHandler = new Handler();
xr.setContentHandler(mySAXHandler);
xr.parse(new InputSource(url.openStream()));
myData = mySAXHandler.getParsedData();
return myData;
} catch (Exception e) {
Log.e("!!!!!!!!!! MyAsyncTask doInBackground error", e.getMessage(), e);
return myData;
}
}
protected void onPostExecute(MyData myData) {
Log.d("+++++++++++++ MyAsyncTask onPostExecute", myData.toString());
}
} // MyAsyncTask
public class MyTimerTask extends TimerTask {
private Runnable runnable = new Runnable() {
public void run() {
new MyAsyncTask().execute();
}
};
public void run() {
handler.post(runnable);
}
}
}
O problema parece ser que você não tenha chamado Looper.prepare()
. Você poderia ter uma olhada na documentação do para Looper
.
A Handler
precisa de um loop de mensagem em para processar mensagens e uma linha por padrão não tem um.
Parece que AsyncTask
usa um Handler
internamente então adicione Looper.prepare()
no topo do seu método run()
em MyTimerTask
e que deve resolver o seu problema
Eu usaria o serviço com o gerente de alarme
I Intent updateIntent = new Intent(ACTION_UPDATE_ALL);
updateIntent.setClass(this, UpdateService.class);
PendingIntent pendingIntent = PendingIntent.getService(this, 0, updateIntent, 0);
AlarmManager alarmManager = (AlarmManager)getSystemService(Context.ALARM_SERVICE);
alarmManager.set(AlarmManager.RTC, nextUpdate, pendingIntent);
O problema é que você não pode criar o objeto Handler no segmento que doInBackground () é executado em. Gostaria de implementar o método OnPreExecute () e colocar o seu código de configuração lá. Tente mover estas linhas no OnPreExecute () método:
URL url = new URL("http://www.example.com/my.xml");
SAXParserFactory spf = SAXParserFactory.newInstance();
SAXParser sp = spf.newSAXParser();
XMLReader xr = sp.getXMLReader();
MyHandler myHandler = new MyHandler();
xr.setContentHandler(myHandler);