我在Android中遇到AsyncTask#2致命异常的问题。所以我要做的是单击地图,获取坐标X和Y并将它们传递给AsyncTask()类以执行获取地址的方法。这是我的单击和MyAsyncTask()的代码:
mMapView.setOnSingleTapListener(new OnSingleTapListener() {
public void onSingleTap(float x, float y) {
eventModel.setEventX(String.valueOf(point.getX()));
eventModel.setEventY(String.valueOf(point.getY()));
new MyAsyncTask().execute(eventModel);
CreateEvent.createEventDialog(context, point.getX(),
point.getY(), eventAddress);
Log.i("Addr", eventAddress);
}
});
new MyAsyncTask().execute();
}
public static class MyAsyncTask extends AsyncTask<Event, Integer, Double> {
@Override
protected Double doInBackground(Event... params) {
try {
eventAddress = eventCtrl.getStreetAddressFromGeometry(eventModel.getEventX(), eventModel.getEventY());
eventCtrl.retrieveEventJSON();
if (params.length == 1) {
eventCtrl.createEvent(params[0]);
// Refresh map after successfully added event
eventCtrl.retrieveEventJSON();
eventCtrl.plotEventOnMap(context);
}
} catch (JSONException e) {
e.printStackTrace();
}
return null;
}
protected void onPostExecute(Double result) {
}
protected void onProgressUpdate(Integer... progress) {
}
}
EventController中的方法基于XY获取地址:
public String getStreetAddressFromGeometry(String eventX, String eventY) {
String streetName = "";
try {
URL url = new URL("http://www.onemap.sg/API/services.svc/revgeocode?token=qo/s2TnSUmfLz+32CvLC4RMVkzEFYjxqyti1KhByvEacEdMWBpCuSSQ+IFRT84QjGPBCuz/cBom8PfSm3GjEsGc8PkdEEOEr&location="+eventX+","+eventY+"");
URLConnection conn = (URLConnection) url.openConnection();
BufferedReader br = new BufferedReader(new InputStreamReader(
(conn.getInputStream())));
String responseString;
while ((responseString = br.readLine()) != null) {
try {
JSONObject rawObj = new JSONObject(responseString);
JSONArray searchResults = rawObj.getJSONArray("GeocodeInfo");
for (int i = 0; i < searchResults.length(); i++) {
streetName = searchResults.getJSONObject(i).getString(
"BLOCK")
+ " " + searchResults.getJSONObject(i).getString("ROAD") + " SINGAPORE";
}
} catch (Exception ex) {
ex.printStackTrace();
}
}
} catch (IOException e) {
e.printStackTrace();
}
return streetName;
}
从单击获取地址后,我将其传递给此方法:
public static Event createEventDialog(final Context context,
final double x, final double y, final String eventAddress) {
AlertDialog.Builder AddDialog = new AlertDialog.Builder(context);
AddDialog.setTitle("Add Event");
LayoutInflater li = (LayoutInflater) context
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
View dialogView = li.inflate(R.layout.create_event, null);
txtEventName = (EditText) dialogView.findViewById(R.id.txtEventName);
txtEventDesc = (EditText) dialogView.findViewById(R.id.txtEventDesc);
txtEventAddr = (EditText) dialogView.findViewById(R.id.txtEventAddr);
txtEventAddr.setText(eventAddress);
AddDialog.setView(dialogView);
AddDialog.setPositiveButton("Ok",
new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int whichButton) {
dialog.dismiss();
}
});
AddDialog.setNegativeButton("Cancel",
new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int whichButton) {
dialog.dismiss();
}
});
AddDialog.show();
return addEventModel;
}
然而,使用这些代码,我从单击打印出地址,它显示地址。但是当我提示出对话框时,它应该显示地址。但是,它没有显示任何内容,几秒钟后,应用程序关闭并显示错误消息:
11-11 17:18:28.889: E/AndroidRuntime(4439): FATAL EXCEPTION: AsyncTask #2
11-11 17:18:28.889: E/AndroidRuntime(4439): java.lang.RuntimeException: An error occured while executing doInBackground()
11-11 17:18:28.889: E/AndroidRuntime(4439): at android.os.AsyncTask$3.done(AsyncTask.java:278)
11-11 17:18:28.889: E/AndroidRuntime(4439): at java.util.concurrent.FutureTask$Sync.innerSetException(FutureTask.java:273)
11-11 17:18:28.889: E/AndroidRuntime(4439): at java.util.concurrent.FutureTask.setException(FutureTask.java:124)
11-11 17:18:28.889: E/AndroidRuntime(4439): at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:307)
11-11 17:18:28.889: E/AndroidRuntime(4439): at java.util.concurrent.FutureTask.run(FutureTask.java:137)
11-11 17:18:28.889: E/AndroidRuntime(4439): at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:208)
11-11 17:18:28.889: E/AndroidRuntime(4439): at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1076)
11-11 17:18:28.889: E/AndroidRuntime(4439): at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:569)
11-11 17:18:28.889: E/AndroidRuntime(4439): at java.lang.Thread.run(Thread.java:856)
11-11 17:18:28.889: E/AndroidRuntime(4439): Caused by: android.content.res.Resources$NotFoundException: Resource ID #0x0
11-11 17:18:28.889: E/AndroidRuntime(4439): at android.content.res.Resources.getValue(Resources.java:1019)
11-11 17:18:28.889: E/AndroidRuntime(4439): at android.content.res.Resources.getDrawable(Resources.java:663)
11-11 17:18:28.889: E/AndroidRuntime(4439): at Controller.EventController.plotEventOnMap(EventController.java:101)
11-11 17:18:28.889: E/AndroidRuntime(4439): at nyp.edu.eneighbourhood.ENeighbourhoodActivity$MyAsyncTask.doInBackground(ENeighbourhoodActivity.java:213)
11-11 17:18:28.889: E/AndroidRuntime(4439): at nyp.edu.eneighbourhood.ENeighbourhoodActivity$MyAsyncTask.doInBackground(ENeighbourhoodActivity.java:1)
11-11 17:18:28.889: E/AndroidRuntime(4439): at android.os.AsyncTask$2.call(AsyncTask.java:264)
11-11 17:18:28.889: E/AndroidRuntime(4439): at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:305)
11-11 17:18:28.889: E/AndroidRuntime(4439): ... 5 more
11-11 17:18:35.725: W/SurfaceView(4439): CHECK surface infomation creating=false formatChanged=false sizeChanged=false visible=false visibleChanged=true surfaceChanged=true realSizeChanged=false redrawNeeded=false left=false top=false
11-11 17:18:37.022: E/WindowManager(4439): Activity nyp.edu.eneighbourhood.ENeighbourhoodActivity has leaked window com.android.internal.policy.impl.PhoneWindow$DecorView@4169d570 that was originally added here
11-11 17:18:37.022: E/WindowManager(4439): android.view.WindowLeaked: Activity nyp.edu.eneighbourhood.ENeighbourhoodActivity has leaked window com.android.internal.policy.impl.PhoneWindow$DecorView@4169d570 that was originally added here
11-11 17:18:37.022: E/WindowManager(4439): at android.view.ViewRootImpl.<init>(ViewRootImpl.java:428)
11-11 17:18:37.022: E/WindowManager(4439): at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:267)
11-11 17:18:37.022: E/WindowManager(4439): at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:215)
11-11 17:18:37.022: E/WindowManager(4439): at android.view.WindowManagerImpl$CompatModeWrapper.addView(WindowManagerImpl.java:140)
11-11 17:18:37.022: E/WindowManager(4439): at android.view.Window$LocalWindowManager.addView(Window.java:537)
11-11 17:18:37.022: E/WindowManager(4439): at android.app.Dialog.show(Dialog.java:278)
11-11 17:18:37.022: E/WindowManager(4439): at android.app.AlertDialog$Builder.show(AlertDialog.java:932)
11-11 17:18:37.022: E/WindowManager(4439): at nyp.edu.eneighbourhood.CreateEvent.createEventDialog(CreateEvent.java:124)
11-11 17:18:37.022: E/WindowManager(4439): at nyp.edu.eneighbourhood.ENeighbourhoodActivity$3.onSingleTap(ENeighbourhoodActivity.java:192)
11-11 17:18:37.022: E/WindowManager(4439): at com.esri.android.map.MapOnTouchListener.onSingleTap(Unknown Source)
11-11 17:18:37.022: E/WindowManager(4439): at com.esri.android.map.MapGestureDetector$a.onSingleTapConfirmed(Unknown Source)
11-11 17:18:37.022: E/WindowManager(4439): at android.view.GestureDetector$GestureHandler.handleMessage(GestureDetector.java:393)
11-11 17:18:37.022: E/WindowManager(4439): at android.os.Handler.dispatchMessage(Handler.java:99)
11-11 17:18:37.022: E/WindowManager(4439): at android.os.Looper.loop(Looper.java:137)
11-11 17:18:37.022: E/WindowManager(4439): at android.app.ActivityThread.main(ActivityThread.java:4512)
11-11 17:18:37.022: E/WindowManager(4439): at java.lang.reflect.Method.invokeNative(Native Method)
11-11 17:18:37.022: E/WindowManager(4439): at java.lang.reflect.Method.invoke(Method.java:511)
11-11 17:18:37.022: E/WindowManager(4439): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:982)
11-11 17:18:37.022: E/WindowManager(4439): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:749)
11-11 17:18:37.022: E/WindowManager(4439): at dalvik.system.NativeStart.main(Native Method)
11-11 17:18:38.209: I/Process(4439): Sending signal. PID: 4439 SIG: 9
提前致谢。
修改
AddDialog.setView(dialogView);
AddDialog.setPositiveButton("Ok",
new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int whichButton) {
addEventModel = new Event();
addEventModel = onConfirmAddEventClicked(context, x, y);
new MyAsyncTask().execute(addEventModel);
dialog.dismiss();
}
});
所以这是用户从对话框中选择Okay的部分,它将获得用户输入,在这种情况下,它是在onConfirmEventClicked中完成的。然后它将调用MyAsyncTask来执行数据库插入。
答案 0 :(得分:0)
改为eventCtrl.plotEventOnMap(context);
上的postExecute
。此方法似乎会更改无法在doInBackground
修改强>
AsyncTask完成后,您必须致电CreateEvent.createEventDialog
。为此,我建议创建一个界面并将其传递给您的任务。这将需要一些编码。让我们开始吧:
public static class MyAsyncTask extends AsyncTask<Event, Integer, Double> {
public interface OnRoutineFinished{ //interface
void onFinish();
}
private OnRoutineFinished mCallbacks;
public MyAsyncTask(OnRoutineFinished callback){ //constructor with interface
mCallbacks = callback;
}
public MyAsyncTask(){} //empty constructor to maintain compatibility
@Override
protected Double doInBackground(Event... params) {
try {
eventAddress = eventCtrl.getStreetAddressFromGeometry(eventModel.getEventX(), eventModel.getEventY());
eventCtrl.retrieveEventJSON();
if (params.length == 1) {
eventCtrl.createEvent(params[0]);
// Refresh map after successfully added event
eventCtrl.retrieveEventJSON();
eventCtrl.plotEventOnMap(context);
}
} catch (JSONException e) {
e.printStackTrace();
}
return null;
}
protected void onPostExecute(Double result) {
if(mCallbacks !=null)
mCallbacks.onFinish(); //call interface on finish
}
protected void onProgressUpdate(Integer... progress) {
}
}
接下来,当您创建asyntask时,您希望在异步完成时调用该事件:
new MyAsyncTask(new MyAsyncTask.OnRoutineFinished() {
@Override
public void onFinish() {
CreateEvent.createEventDialog(context, point.getX(),
point.getY(), eventAddress); //this will be called after the task finishes
}
}).execute(eventModel);