2013-07-04 102 views
1

我在下面的代码中得到了着名的“无法在未调用Looper.prepare()的线程中创建处理程序”错误。基于我的大部分谷歌搜索结果,当我尝试从我的非用户线程更新用户界面时,会发生这种情况,但是当我从其他助手类调用该方法时,此时不更新用户界面(仅在地址已经回);所以请咨询为什么发生这种情况将不胜感激。错误:无法在未调用Looper.prepare()的线程中创建处理程序

package com.parspake.kookojapost; 

import android.app.ActionBar; 
import android.app.Activity; 
import android.content.Context; 
import android.content.Intent; 
import android.database.sqlite.SQLiteDatabase; 
import android.location.*; 
import android.os.Build; 
import android.os.Bundle; 
import android.os.Handler; 
import android.os.Looper; 
import android.util.Log; 
import android.view.Menu; 
import android.view.MenuItem; 
import android.view.View; 
import android.widget.EditText; 
import android.widget.TextView; 
import android.widget.Toast; 

public class TextShare extends Activity { 


    ConnectionHandler textPageConn; 
    TextView dt2; 

    @Override 
    protected void onCreate(Bundle savedInstanceState) { 
     super.onCreate(savedInstanceState); 
     setContentView(R.layout.text_share); 

     textPageConn = new ConnectionHandler(this); 
     dt2 = (TextView) findViewById(R.id.show_location_text_page); 

     Thread thread1 = new Thread(new Runnable() { 

      @Override 
      public void run() { 

       textPageConn.getLocation(); 
       do { 

        if (textPageConn.mAddress != null) { 
         dt2.setText(textPageConn.mAddress); 
         break; 
        } else { 
         Log.d("debug","no location"); 
        } 

        try { 
         Thread.sleep(1000); 
        } catch (InterruptedException e) { 
         Thread.currentThread().interrupt(); 
        } 
       } while (textPageConn.mAddress == null); 
      } 

     }); 
     thread1.start(); 
    } 
} 

这是我得到的例外:

07-04 23:59:38.730: E/AndroidRuntime(7149): FATAL EXCEPTION: Thread-8482 
07-04 23:59:38.730: E/AndroidRuntime(7149): java.lang.RuntimeException: Can't create handler inside thread that has not called Looper.prepare() 
07-04 23:59:38.730: E/AndroidRuntime(7149):  at android.os.Handler.<init>(Handler.java:121) 
07-04 23:59:38.730: E/AndroidRuntime(7149):  at android.location.LocationManager$ListenerTransport$1.<init>(LocationManager.java:183) 
07-04 23:59:38.730: E/AndroidRuntime(7149):  at android.location.LocationManager$ListenerTransport.<init>(LocationManager.java:183) 
07-04 23:59:38.730: E/AndroidRuntime(7149):  at android.location.LocationManager._requestLocationUpdates(LocationManager.java:661) 
07-04 23:59:38.730: E/AndroidRuntime(7149):  at android.location.LocationManager.requestLocationUpdates(LocationManager.java:486) 
07-04 23:59:38.730: E/AndroidRuntime(7149):  at com.parspake.kookojapost.ConnectionHandler.getLocation(ConnectionHandler.java:136) 
07-04 23:59:38.730: E/AndroidRuntime(7149):  at com.parspake.kookojapost.TextShare$2.run(TextShare.java:69) 
07-04 23:59:38.730: E/AndroidRuntime(7149):  at java.lang.Thread.run(Thread.java:856) 

所以ConnectionHandler.java:136是 - > mLocationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER,10000,50,mlocationListener);

和TextShare.java:69是 - > textPageConn.getLocation();

这是我ConnectionHandler Helper类:

package com.parspake.kookojapost; 

import android.content.Context; 
import android.location.*; 
import android.net.ConnectivityManager; 
import android.net.NetworkInfo; 
import android.os.Bundle; 
import android.util.Log; 
import android.widget.Toast; 

import java.io.IOException; 
import java.util.List; 
import java.util.Locale; 

public class ConnectionHandler { 

    private Context ctx; 
    double mLat; 
    double mLong; 
    String currCity; 
    String currCountry; 
    String mAddress; 
    LocationManager mLocationManager; 
    LocationListener mlocationListener; 

    public ConnectionHandler(Context context) { 
     this.ctx = context; 
    } 


    public boolean locationSourceEnabled() { 
     mLocationManager = (LocationManager) ctx.getSystemService(Context.LOCATION_SERVICE); 
     boolean isInternetLocationAvailable = mLocationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER); 
     boolean isGpsAvailable = mLocationManager.isProviderEnabled(LocationManager.GPS_PROVIDER); 
     if (isInternetLocationAvailable) { 
      return true; 
     } else if (isGpsAvailable) { 
      return true; 
     } 
     return false; 
    } 


    public void getLocation() { 

     mlocationListener = new LocationListener() { 
      @Override 
      public void onLocationChanged(Location location) { 

       mLat = location.getLatitude(); 
       mLong = location.getLongitude(); 

       Geocoder gcd = new Geocoder(ctx, Locale.getDefault()); 
       List<Address> addresses; 
       try { 
        addresses = gcd.getFromLocation(mLat, mLong, 1); 
        if (addresses.size() > 0) { 
         currCity = addresses.get(0).getLocality(); 
         currCountry = addresses.get(0).getCountryName(); 
         mAddress = currCity + " - " + currCountry; 
        } 
       } catch (IOException e) { 
        Log.d("omid debug", e.getMessage()); 
       } 
      } 

      @Override 
      public void onStatusChanged(String s, int i, Bundle bundle) { 
      } 

      @Override 
      public void onProviderEnabled(String s) { 
      } 

      @Override 
      public void onProviderDisabled(String s) { 
      } 
     }; 
     mLocationManager = (LocationManager) ctx.getSystemService(Context.LOCATION_SERVICE); 
     if (locationSourceEnabled()) { 
      if (mLocationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER)) { 
       mLocationManager.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 10000, 50, mlocationListener); 
      } 
      if (mLocationManager.isProviderEnabled(LocationManager.GPS_PROVIDER)) { 
       mLocationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 10000, 50, mlocationListener); 
      } 
     } 
    } 
} 

回答

2

好的问题是requestLocationUpdates是一个异步调用,这意味着你不需要另一个线程上运行它也意味着,如果你尝试,你正在有效地使另一个线程上的处理程序是不允许的。因此,你可以删除线程,只是运行这些上的onCreate()(因为你只是做异步调用)

参考:requestLocationUpdates gives error "Can't create Handler inside thread that has not called Looper.prepare()

相反,你可以把你的电话syncronous在另一个线程...

@Override 
public void onLocationChanged(Location location) { 
    new Thread(new Runnable() { 

     @Override 
     public void run() { 
      mLat = location.getLatitude(); 
      mLong = location.getLongitude(); 

      Geocoder gcd = new Geocoder(ctx, Locale.getDefault()); 
      List<Address> addresses; 
      try { 
       addresses = gcd.getFromLocation(mLat, mLong, 1); 
       if (addresses.size() > 0) { 
        currCity = addresses.get(0).getLocality(); 
        currCountry = addresses.get(0).getCountryName(); 
        mAddress = currCity + " - " + currCountry; 
       } 
      } catch (IOException e) { 
       Log.d("omid debug", e.getMessage()); 
      } 
     } 
    }).start(); 
} 
+0

也尝试了你的代码,并且仍然得到相同的Looper.prepare()异常。我的代码在ConnectionHandler helper类中引用了这一行:mLocationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER,10000,50,mlocationListener); – kevoroid

+0

你可以添加ConnectionHandler的代码吗? –

+0

当然。添加到原始帖子。 – kevoroid

1

您无法更新TextView从非UI线程。

使用runOnUiThread代替:

runOnUiThread(new Runnable() { 

        @Override 
        public void run() { 
         dt2.setText(textPageConn.mAddress); 
        } 
       }); 
+0

我做到了。它没有工作。在“do”之后,我写道:runOnUiThread(new Runnable(){...然后运行()覆盖...... – kevoroid

+1

我建议你注释掉'dt2.setText()'行,然后写'Log.v(“MY_APP “,”here“)'如果你没有得到'Looper.prepare()'异常并且看到日志中的”here“,那么问题出现在'setText()'中。 – azizbekian

+0

我确实评论过仍然有Looper.prepare()异常! – kevoroid

0

它不可能改变在其他线程的UI内容..你可以在基本方法

  • 解决这些问题,使用[R unOnUiThread()方法,其操作简单,但它不是最好的做法
  • 另一种方法是使用处理器与回调处理器(回拨三),其推荐的一个

我会建议使用的处理程序用回调方法。 你可以在这里获得很好的教程。 handler with callback in thread

相关问题