Wie mit Hilfe von XML AsyncTask und Timer zu bekommen?
-
19-09-2019 - |
Frage
Um XML-Daten von einem Server wiederholt zu bekommen, ich bin versucht, AsyncTask und Timer zu verwenden, wie pro Mark Murphy Vorschlag .
Ich erhalte den folgenden Fehler:
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()
Ich bin mit SDK 1.5 mit Eclipse unter Windows.
Ich habe in der Dokumentation sieht, auf Stackoverflow und in der Android-Entwickler-Gruppe, aber ich bin mir nicht klar, was den Fehler verursacht oder wie man es beheben.
Ich kann die Daten bekommen einmal - das heißt ohne Async und Timer - und analysieren es über SAX ohne Probleme
.Voll App-Code unten.
Bitte entschuldigen Sie irgendwelche naiven Fehler: Ich Android ganz neu bin
.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);
}
}
}
}
Lösung
Das Problem ist bei der Verwendung von Timertask. Timertask läuft zu einem Handler, so etwas wie diese schreiben sollte:
private Handler mHandler = new Handler();
public class MyTimerTask extends TimerTask {
public void run() {
mHandler.post(
new Runnable() {
public void run() {
new MyAsyncTask().execute("");
}
};
)
}
}
Das ist natürlich immer ein bisschen hässlich, so würde empfehlen, die anonyme Klasse herausnehmen.
Andere Tipps
habe ich dies dank Antwort auf James zu arbeiten.
ich den Code unten in dem Fall aufgenommen haben, es ist nützlich für alle.
Caveat Entwickler! Der Code funktioniert für mich, aber kann Fehler enthalten.
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);
}
}
}
Das Problem scheint zu sein, dass Sie nicht Looper.prepare()
genannt haben. Sie konnten einen Blick auf die Dokumentation für Looper
haben .
Handler
braucht eine Nachrichtenschleife in bestellen Nachrichten und ein Thread standardmäßig zu verarbeiten keinen hat.
Es scheint, dass AsyncTask
verwendet ein Handler
intern so Looper.prepare()
an der Spitze Ihrer run()
Methode in MyTimerTask
hinzufügen und das sollte Ihr Problem lösen
ich den Dienst mit Alarmmanager verwenden würde
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);
Das Problem ist, dass Sie nicht das Handler-Objekt in dem Thread erstellen können, dass doInBackground () läuft. Ich würde die OnPreExecute () -Methode implementieren und Ihren Setup-Code in dort setzen. Versuchen Sie diese Zeilen in die OnPreExecute bewegen () Methode:
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);