#android #android-asynctask
#Android #android-asynctask
Вопрос:
Кто-нибудь знает, почему AsyncTask принудительно закрывается, когда я пытаюсь разархивировать файл в фоновом режиме, он работает, когда не используется AsyncTask, поэтому я не уверен, что я делаю неправильно:
public class UnzipFile extends AsyncTask<Void, Integer, String> {
String zipFile = Environment.getExternalStorageDirectory() "/download/" mixtapefilename;
String unzipLocation = Environment.getExternalStorageDirectory() "/download/";
@Override
protected void onPreExecute() {
Toast toast = Toast.makeText(getApplicationContext(), "Unzipping...", Toast.LENGTH_SHORT);
toast.show();
/* Intent intent = new Intent();
final PendingIntent pendingIntent = PendingIntent.getActivity(Download.this, 0, intent, 0);
// configure the notification
notification = new Notification(R.drawable.zipicon, "Unzipping...", System
.currentTimeMillis());
notification.flags = notification.flags | Notification.FLAG_ONGOING_EVENT;
notification.setLatestEventInfo(getApplicationContext(), "Please Wait", "Unzipping...", pendingIntent);
notificationManager2.notify(1, notification);*/
}
@Override
protected String doInBackground(Void... params) {
Decompress d = new Decompress(zipFile, unzipLocation);
d.unzip();
return "success";}
@Override
protected void onPostExecute(String result) {
Toast toast = Toast.makeText(getApplicationContext(), "Download Complete", Toast.LENGTH_SHORT);
toast.show();
File oldzip = new File(zipFile);
boolean deleted = oldzip.delete();
sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse("file://" Environment.getExternalStorageDirectory())));
/*notificationManager2.cancel(1);*/
}
}
и затем вот моя часть распаковки, которую я пытаюсь запустить в фоновом режиме
public class Decompress {
byte[] buffer = new byte[1024];
int length;
private String _zipFile;
private String _location;
public Decompress(String zipFile, String location) {
_zipFile = zipFile;
_location = location;
File a = new File(_location foldername);
if(!a.isDirectory()){
a.mkdirs();
}
_dirChecker("");
}
public void unzip() {
try {
FileInputStream fin = new FileInputStream(_zipFile);
ZipInputStream zin = new ZipInputStream(fin);
ZipEntry ze = null;
while ((ze = zin.getNextEntry()) != null) {
if(ze.isDirectory()) {
_dirChecker(ze.getName());
} else {
FileOutputStream fout = new FileOutputStream(_location ze.getName());
while ((length = zin.read(buffer))>0) {
fout.write(buffer, 0, length);
}
zin.closeEntry();
fout.close();
}
}
zin.close();
Toast toast = Toast.makeText(getApplicationContext(),"Download Complete", Toast.LENGTH_LONG);
toast.show();
} catch(Exception e) {
ProgressDialog dialog;
dialog = new ProgressDialog(Download.this);
dialog.setMessage(e.toString());
dialog.show();
}
}
private void _dirChecker(String dir) {
File f = new File(_location dir);
if(!f.isDirectory()) {
f.mkdirs();
}
}
}
Комментарии:
1. Какое исключение выдает logcat?
Ответ №1:
Поскольку вы используете AsynTask, вы не можете взаимодействовать с основным потоком пользовательского интерфейса (т.Е. Отображать всплывающие сообщения) внутри doInBackground().
Единственный способ связаться с основным потоком пользовательского интерфейса во время выполнения асинхронной задачи — использовать onPreExcute()
или onPostExcute()
Так что просто удалите эту строку:
Toast toast = Toast.makeText(getApplicationContext(),"Download Complete", Toast.LENGTH_LONG);
Комментарии:
1. Спасибо, я понял это сразу после того, как нажал post. Извините за бессмысленный вопрос.
Ответ №2:
Извините, что не согласен, но есть несколько способов выполнить код в потоке пользовательского интерфейса из другого потока: Activity.runOnUiThread(Runnable) View.post (доступный для выполнения) View.postDelayed(выполняемый, длинный) вы можете использовать один из них для публикации чего-либо, что будет выполняться потоком пользовательского интерфейса независимо от потока, в котором вы находитесь в данный момент. Но да, вы могли бы просто убрать их и вместо этого использовать Log.d(строка, строка) (например)
Ответ №3:
Поток пользовательского интерфейса не является потокобезопасным. Поэтому так опасно обращаться к потоку пользовательского интерфейса в другом потоке. Вы должны опубликовать свое сообщение в очереди сообщений пользовательского интерфейса с помощью View.post()
метода