Отправка сообщения обработчику мертвой нити при получении местоположения из IntentService

Мое приложение нуждается в исправлениях местоположения на регулярной основе, даже если телефон не бодрствует. Для этого я использую IntentService с шаблоном, щедро предоставленным Commonsware. https://github.com/commonsguy/cwac-wakeful

Чтобы получить исправления местоположения, я полагаюсь на следующий код, предоставленный членом под названием Fedor. Каков самый простой и надежный способ получить текущее местоположение пользователя на Android? , Я слегка изменил его, чтобы вернуть null вместо того, чтобы получить последнее известное местоположение

Оба они отлично работают, когда не комбинируются: я могу получить исправление местоположения из класса Fedor в Activity, и я могу сделать некоторые базовые вещи (например, запись в журнал) с использованием класса Commonsware.

Проблема возникает, когда я пытаюсь получить исправления местоположения из подкласса Commonsware WakefulIntentService. Установщики LocationListeners не реагируют на locationUpdates, и я получаю эти предупреждения (у меня пока нет тонкостей потоков, обработчиков, …). Может кто-нибудь помочь мне понять, в чем проблема? Спасибо, и я желаю вам всего очень счастливого нового года 🙂

12-31 14:09:33.664: W/MessageQueue(3264): Handler (android.location.LocationManager$ListenerTransport$1) {41403330} sending message to a Handler on a dead thread 12-31 14:09:33.664: W/MessageQueue(3264): java.lang.RuntimeException: Handler (android.location.LocationManager$ListenerTransport$1) {41403330} sending message to a Handler on a dead thread 12-31 14:09:33.664: W/MessageQueue(3264): at android.os.MessageQueue.enqueueMessage(MessageQueue.java:196) 12-31 14:09:33.664: W/MessageQueue(3264): at android.os.Handler.sendMessageAtTime(Handler.java:473) 12-31 14:09:33.664: W/MessageQueue(3264): at android.os.Handler.sendMessageDelayed(Handler.java:446) 12-31 14:09:33.664: W/MessageQueue(3264): at android.os.Handler.sendMessage(Handler.java:383) 12-31 14:09:33.664: W/MessageQueue(3264): at android.location.LocationManager$ListenerTransport.onLocationChanged(LocationManager.java:193) 12-31 14:09:33.664: W/MessageQueue(3264): at android.location.ILocationListener$Stub.onTransact(ILocationListener.java:58) 12-31 14:09:33.664: W/MessageQueue(3264): at android.os.Binder.execTransact(Binder.java:338) 12-31 14:09:33.664: W/MessageQueue(3264): at dalvik.system.NativeStart.run(Native Method) 

Это мой подкласс WakefulIntentService:

 public class AppService extends WakefulIntentService { public static final String TAG = "AppService"; public AppService() { super("AppService"); } public LocationResult locationResult = new LocationResult() { @Override public void gotLocation(final Location location) { if (location == null) Log.d(TAG, "location could not be retrieved"); else sendMessage(location); } }; @Override protected void doWakefulWork(Intent intent) { PhoneLocation myLocation; myLocation = new PhoneLocation(); myLocation.getLocation(getApplicationContext(), locationResult); } 

И вот экземпляр класса Федора (слегка измененный: нет необходимости в GPS и последнем известном месте)

 public class PhoneLocation { public static String TAG = "MyLocation"; Timer timer1; LocationManager lm; LocationResult locationResult; boolean gps_enabled=false; boolean network_enabled=false; public boolean getLocation(Context context, LocationResult result) { //I use LocationResult callback class to pass location value from MyLocation to user code. locationResult=result; if(lm==null) lm = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE); //exceptions will be thrown if provider is not permitted. try{gps_enabled=lm.isProviderEnabled(LocationManager.GPS_PROVIDER);}catch(Exception ex){} try{network_enabled=lm.isProviderEnabled(LocationManager.NETWORK_PROVIDER);}catch(Exception ex){} //don't start listeners if no provider is enabled if(!gps_enabled && !network_enabled) return false; if(network_enabled) lm.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 0, 0, locationListenerNetwork); timer1=new Timer(); timer1.schedule(new ReturnNullLocation(), 20000); return true; } LocationListener locationListenerNetwork = new LocationListener() { public void onLocationChanged(Location location) { timer1.cancel(); locationResult.gotLocation(location); lm.removeUpdates(this); } public void onProviderDisabled(String provider) {} public void onProviderEnabled(String provider) {} public void onStatusChanged(String provider, int status, Bundle extras) {} }; class ReturnNullLocation extends TimerTask { @Override public void run() { lm.removeUpdates(locationListenerNetwork); locationResult.gotLocation(null); } } public static abstract class LocationResult{ public abstract void gotLocation(Location location); } 

}

Вы не можете безопасно регистрировать слушателей из IntentService , так как IntentService уходит, как только onHandleIntent() (aka, doWakefulWork() ) завершается. Вместо этого вам нужно будет использовать обычный сервис, а также данные о ручках, такие как тайм-ауты (например, пользователь находится в подвале и не может получить сигнал GPS).

У меня была аналогичная ситуация, и я использовал средство андроида Looper для хорошего эффекта: http://developer.android.com/reference/android/os/Looper.html

Конкретно, сразу после вызова функции настройки слушателя я вызываю:

 Looper.loop(); 

Это блокирует текущий поток, поэтому он не умирает, но в то же время позволяет обрабатывать события, поэтому у вас будет возможность получить уведомление при вызове слушателя. Простой цикл помешает вам получать уведомление при вызове слушателя.

И когда вызывается слушатель и я обрабатываю его данные, я добавляю:

 Looper.myLooper().quit(); 

Для других, подобных мне: у меня была аналогичная проблема, связанная с AsyncTask . Насколько я понимаю, этот класс предполагает, что он инициализирован в потоке пользовательского интерфейса (основного).

Обходной путь (один из нескольких возможных) заключается в размещении в классе MyApplication следующего:

 @Override public void onCreate() { // workaround for http://code.google.com/p/android/issues/detail?id=20915 try { Class.forName("android.os.AsyncTask"); } catch (ClassNotFoundException e) {} super.onCreate(); } 

(Не забудьте упомянуть об этом в AndroidManifest.xml :

 <application android:icon="@drawable/ic_launcher" android:label="@string/app_name" android:name="MyApplication" > 

)