Android Looper.prepare异常在外部服务中使用LocationManager
当我尝试在外部服务中运行的自定义类中使用LocationManager时,出现以下异常:Android Looper.prepare异常在外部服务中使用LocationManager,android,service,locationmanager,looper,Android,Service,Locationmanager,Looper,当我尝试在外部服务中运行的自定义类中使用LocationManager时,出现以下异常: *** Uncaught remote exception! (Exceptions are not yet supported across processes.) java.lang.RuntimeException: Can't create handler inside thread that has not called Looper.prepare() at an
*** Uncaught remote exception! (Exceptions are not yet supported across processes.)
java.lang.RuntimeException: Can't create handler inside thread that has not called Looper.prepare()
at android.os.Handler.<init>(Handler.java:200)
at android.os.Handler.<init>(Handler.java:114)
at android.location.LocationManager$GpsStatusListenerTransport$1.<init>(LocationManager.java:1464)
at android.location.LocationManager$GpsStatusListenerTransport.<init>(LocationManager.java:1464)
at android.location.LocationManager.addGpsStatusListener(LocationManager.java:1503)
at org.poseidon_project.contexts.hardware.GPSIndoorOutdoorContext.start(GPSIndoorOutdoorContext.java:97)
at org.poseidon_project.context.management.ContextManager.addObserverRequirement(ContextManager.java:97)
at org.poseidon_project.context.reasoner.ContextMapper.registerIndoorOutdoorsContext(ContextMapper.java:260)
at org.poseidon_project.context.reasoner.ContextMapper.registerContext(ContextMapper.java:197)
at org.poseidon_project.context.ContextReasonerCore.addContextRequirement(ContextReasonerCore.java:70)
at org.poseidon_project.context.ContextReasonerService$1.addContextRequirement(ContextReasonerService.java:74)
at org.poseidon_project.context.IContextReasoner$Stub.onTransact(IContextReasoner.java:74)
at android.os.Binder.execTransact(Binder.java:446)
但这最终不会导致在有更新时调用回调(onLocationChanged(Location-Location))
该类实现LocationListener,它还调用LocationManager方法:
public abstract class LocationContext extends ContextObserver implements LocationListener {
protected LocationManager mLocationManager;
private int mMinTime = 3000;
private int mMinDistance = 10;
private String mProvider = LocationManager.GPS_PROVIDER;
private String mIdealProvider = LocationManager.GPS_PROVIDER;
public LocationContext (Context c) {
super(c);
mLocationManager = (LocationManager) mContext.getSystemService(Context.LOCATION_SERVICE);
}
public LocationContext (Context c, ContextReceiver cr) {
super(c, cr);
mLocationManager = (LocationManager) mContext.getSystemService(Context.LOCATION_SERVICE);
}
public LocationContext (Context c, ContextReceiver cr, int minTime, int minDistance, String name) {
super(c, cr, name);
mMinTime = minTime;
mMinDistance = minDistance;
mLocationManager = (LocationManager) mContext.getSystemService(Context.LOCATION_SERVICE);
}
public LocationContext (Context c, ContextReceiver cr, int minTime, int minDistance, String provider, String name) {
super(c, cr, name);
mMinTime = minTime;
mMinDistance = minDistance;
mProvider = provider;
mIdealProvider = provider;
mLocationManager = (LocationManager) mContext.getSystemService(Context.LOCATION_SERVICE);
}
@Override
public boolean start() {
//new Thread(new Runnable() {
// @Override
// public void run() {
//Looper.prepare();
mLocationManager.requestLocationUpdates(mProvider, mMinTime, mMinDistance, this);
//handler.sendEmptyMessage(0);
//Looper.loop();
// }
//}).start();
mIsRunning = true;
//Looper.loop();
return true;
}
@Override
public boolean pause() {
return stop();
}
@Override
public boolean resume() {
return start();
}
@Override
public boolean stop() {
mLocationManager.removeUpdates(this);
mIsRunning = false;
return true;
}
@Override
public void onLocationChanged(Location location) {
checkContext(location);
}
protected abstract void checkContext(Location location);
@Override
public void onProviderDisabled(String provider) {
if (provider.equals(mIdealProvider)) {
mProvider = LocationManager.GPS_PROVIDER;
if (! mLocationManager.isProviderEnabled(mProvider)) {
Intent gpsOptionIntent = new Intent (android.provider.Settings.ACTION_LOCATION_SOURCE_SETTINGS);
mContext.startActivity(gpsOptionIntent);
}
}
}
@Override
public void onProviderEnabled(String provider) {
if ((provider.equals(mIdealProvider)) && (! provider.equals(mProvider))) {
mLocationManager.removeUpdates(this);
mProvider = provider;
mLocationManager.requestLocationUpdates(mProvider, mMinTime, mMinDistance, this);
}
}
@Override
public void onStatusChanged(String provider, int status, Bundle extras) {
// TODO Auto-generated method stub
}
public int getMinTime() {
return mMinTime;
}
public void setMinTime(int mMinTime) {
this.mMinTime = mMinTime;
}
public int getMinDistance() {
return mMinDistance;
}
public void setMinDistance(int mMinDistance) {
this.mMinDistance = mMinDistance;
}
在我的情况下,我真的不知道如何使用活套。有人能帮忙吗?我理解“在UI线程中运行”的答案,但这是一个单一的服务应用程序,没有UI,所以我认为我不能在UI线程中运行
****找到更新****解决方案******
我相信我找到了解决办法。所讨论的类是一个抽象类,我通过几个执行各种基于位置的操作的类来扩展它
在我使用的LocationContext抽象类中:
mLocationManager.requestLocationUpdates(mProvider, mMinTime, mMinDistance,this, Looper.getMainLooper());
在一个实现类(例如用于分析GPS卫星状态的类)中,我将其放在一个新线程中:
new Thread(new Runnable() {
@Override
public void run() {
Looper.prepare();
GPSIndoorOutdoorContext.super.start();
mLocationManager.addGpsStatusListener(gpsListener);
Looper.loop();
}
}).start();
mLocationManager.requestLocationUpdate(mProvider、mMinTime、mMinDistance、this)代码>
正在从非UI线程调用。确保在UI线程中调用init或方法。您可能正在从非UI线程启动LocationContext
或调用start
方法,这是您无法做到的。要请求位置更新,必须从UI线程调用它。问题是,这是一个在没有UI的情况下运行的服务。它应该是其他应用程序(使用UI)调用的服务。我不知道如何在UI线程上实现它?如果我在同一个应用程序中尝试使用测试活动,则此代码可以正常工作,但当从不同的应用程序调用此代码时,会引发异常。我不明白如何才能这样做,因为我说过,此服务通常与任何具有活动的应用程序处于单独的进程中。RunUnuithRead是一种活动方法。。。有什么办法可以让我用活套绕过这个问题吗?
new Thread(new Runnable() {
@Override
public void run() {
Looper.prepare();
GPSIndoorOutdoorContext.super.start();
mLocationManager.addGpsStatusListener(gpsListener);
Looper.loop();
}
}).start();