Лента за напредък на изчакване на Android Не може да се създаде манипулатор в нишка, която не е извикала грешка Looper.prepare().

В моето приложение за Android копирам някои данни от сървърна база данни в локална база данни. Може да отнеме над 15 минути. За по-бавни връзки може да бъде надвишено. Така че искам да покажа лента за напредък на чакане.

Кодът е публикуван по-долу.

refresh.setOnClickListener(new OnClickListener(){
public void onClick(View v) {
final ProgressDialog progressDialog = ProgressDialog.show(login.this, "", "Please wait...");
new Thread() {
public void run() {
try{
mySQLiteAdapter.openToWrite();
mySQLiteAdapter.deleteAll();
radapter.openToWrite();
radapter.deleteAll();
uadapter.openToWrite();
uadapter.deleteAll();
        try
            {
                 HttpClient httpclient = new DefaultHttpClient();
                 HttpPost httppost = new HttpPost("http://xxxxxxxxxxxxxxxxxxxxx");
                 HttpResponse response = httpclient.execute(httppost);
                 HttpEntity entity = response.getEntity();
                  is = entity.getContent();
            } catch (UnsupportedEncodingException e) {
                e.printStackTrace();
            } catch (ClientProtocolException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }
            try {
                BufferedReader reader = new BufferedReader(new InputStreamReader(
                        is, "iso-8859-1"), 8);
                StringBuilder sb = new StringBuilder();
                String line = null;
                while ((line = reader.readLine()) != null) {
                    sb.append(line + "\n");
                }
                is.close();
                json = sb.toString();
            } catch (Exception e) {
            Toast.makeText(getBaseContext(), e.toString(), Toast.LENGTH_LONG).show();
            }

            try {
                jObj = new JSONObject(json);
                contacts = jObj.getJSONArray("get");
                for(int i = 0; i < contacts.length(); i++){
                JSONObject c = contacts.getJSONObject(i);
                    if(!c.isNull("rname"))// || c.getString("rcode").equals(null))
                    {
                        rname = c.getString("rname");
                        rcode = c.getString("rcode");
                        radapter.insert(rcode, rname);
                    }
                    else
                    {
                        rname = "";
                        rcode = c.getString("rcode");
                        radapter.insert(rcode, rname);
                    }
                }
                Toast.makeText(getBaseContext(), " Ryot BackUp completed", Toast.LENGTH_SHORT).show();

            } catch (JSONException e) {
                Toast.makeText(getBaseContext(), "Error"+e.toString(), Toast.LENGTH_LONG).show();
            }
                   progressDialog.dismiss();
               }
            }.start();
            adapter.deleteAll();
            oadapter.deleteAll();
            e2.setText("Back Up completed");
        }
            catch (Exception e) {
               Log.e("tag", e.getMessage());
               }
                   progressDialog.dismiss();
               }
            }.start();
       }

   });

Когато изпълнявам приложението си, лентата за напредъка се показва само за няколко секунди и блокът Catch се изпълнява, показвайки грешката на logcat като

Logcat:

02-26 14:26:07.151: E/tag(301): Can't create handler inside thread that has not called Looper.prepare()

Може ли някой да обясни каква е грешката в моя код и как да я реша


person Linga    schedule 26.02.2013    source източник
comment
типичен случай на опит за достъп до потребителския интерфейс от друга нишка. Решението използва асинхронна задача за фонови неща и достъп до потребителския интерфейс в onProgressUpdate и onPostExecute.   -  person njzk2    schedule 26.02.2013


Отговори (2)


Toast.makeText(getBaseContext(), " Ryot BackUp completed", Toast.LENGTH_SHORT).show();

Проблемът е причинен от горния код, този метод ще промени потребителския интерфейс, не можете да актуализирате потребителския интерфейс в нишка, която не е потребителски интерфейс. Така че трябва да го извикате от основната потребителска нишка.

Тъй като това отнема много време, горещо ви препоръчвам да използвате AsyncTask в Android.

Но ако искате само да направите малка промяна в съществуващия си код, опитайте да използвате

context.runOnUiThread(new Runnable() {
  public void run() {
    Toast.makeText(...).show();
  }
});

или манипулатор, можете да разгледате Как работи манипулаторът в Android. Основен пример:

public Handler mHandler = new Handler() {

    public void handleMessage(Message msg) {
           //show toast here
    }
}

protected void startLongRunningOperation() {

    Thread t = new Thread() {
        public void run() {
            //do something here
            //update the result
            mHandler.postDelayed(mUpdateResults, 200);}
            mHandler.post(mUpdateResults);
        }
    };
    t.start();
}
person StarPinkER    schedule 26.02.2013
comment
Къде трябва да използвам това? само за показване на тост - person Linga; 26.02.2013

Вместо да създавате нова нишка, направете HandlerThread - http://developer.android.com/reference/android/os/HandlerThread.html

Просто заменете "new Thread()" с "new HandlerThread("any string)" и трябва да работи.

person azertiti    schedule 26.02.2013