Отправка сообщения обработчику мертвой нити при получении местоположения из 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);
}
И вот копия класса Fedor (слегка измененная: нет необходимости в 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);
}
}
Ответы
Ответ 1
Вы не можете безопасно регистрировать слушателей с IntentService
, так как IntentService
уходит, как только onHandleIntent()
(a.k.a., doWakefulWork()
) завершается. Вместо этого вам нужно будет использовать обычный сервис, а также данные о деталях, такие как тайм-ауты (например, пользователь находится в подвале и не может получить сигнал GPS).
Ответ 2
У меня была аналогичная ситуация, и я использовал средство Android Looper для хорошего эффекта:
http://developer.android.com/reference/android/os/Looper.html
Конкретно, сразу после вызова функции настройки слушателя я вызываю:
Looper.loop();
Это блокирует текущий поток, поэтому он не умирает, но в то же время позволяет обрабатывать события, поэтому у вас будет возможность получить уведомление при вызове слушателя. Простой цикл помешает вам получать уведомление при вызове слушателя.
И когда вызывается слушатель и я обрабатываю его данные, я поставлю:
Looper.myLooper().quit();
Ответ 3
Для других, подобных мне: у меня была аналогичная проблема, связанная с AsyncTask
. Как я понимаю, этот класс предполагает, что он инициализирован в потоке пользовательского интерфейса (main).
Обходной путь (один из нескольких возможных) состоит в том, чтобы поместить следующее в класс 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"
>
)