Принудительное закрытие AsyncTask

#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() метода