1

電話の位置を知る必要があるサービスがあります。サービスの主なアクティビティは、次のようにスレッドで実行されます (処理内容は削除されています:

Semaphore locationAcquired = new Semaphore(1);
LocationFinder finder;

...
public void run() {
            delaySeconds = 60;
            Looper.prepare();
            while (true) {
                try {

                    finder.StartFinder();

                locationAcquired.acquire();
                // do some stuff...
                } catch (InterruptedException e) {
                    if (isDestroy) {
                        Log.d(TAG, "Closing Monitor Thread");
                        break;
                    } // else just wake up and process the location 
                } catch (Exception e) {
                    e.printStackTrace();

                }
            } // end while
        } // end run

LocationFinder クラスの実装 (これも少し簡略化されています):

package com.ksdagile.opengate;

import...

public class LocationFinder {

public static final int ONE_SECOND = 1000;

LocationListener locationListener;
String provider = LocationManager.PASSIVE_PROVIDER; // passive by default
LocationManager locationManager;
public Location currentLocation;
long updateSeconds;
private boolean isLooking = false;
OpenGateService openGateService;

public LocationFinder(LocationManager _lm, OpenGateService _openGateService) {

    openGateService = _openGateService;
    locationManager = _lm;
     // initialize with whatever location might be available
    currentLocation = locationManager.getLastKnownLocation(LocationManager.PASSIVE_PROVIDER); 


    locationListener = new LocationListener() {
        @Override
        public void onLocationChanged(Location location) {
            // A new location update is received.  Do something useful with it.  In this case,
            // we're sending the update to a handler which then updates the UI with the new
            // location.
            currentLocation = location;
            String newLoc = String.format("Found new Location lat:%.2f long:%.2f", currentLocation.getLatitude(), currentLocation.getLongitude());
            Log.d(getClass().getName(), newLoc);
            openGateService.locationAcquired.release();
        }
                    // simple implementations of onProvider<> etc.
    };

}

public void SetProvider(boolean isActive) {
    if (isActive)
        provider = LocationManager.GPS_PROVIDER;
    else
        provider = LocationManager.PASSIVE_PROVIDER;
}

public void SetFrequency(long delay) {
    updateSeconds = delay;
}

public void StartFinder() {
    if (!isLooking) {
        isLooking = true;
        locationManager.requestLocationUpdates(provider, updateSeconds*ONE_SECOND, 10, locationListener);
        Log.d(getClass().getName(), String.format("Request location from %s provider, every %d sec.", provider, updateSeconds));            
    } else
        Log.d(getClass().getName(), "Location request running");
}

public void StopFinder() {
    locationManager.removeUpdates(locationListener);
    isLooking = false;
}

public boolean IsLocating() {
    return isLooking;
}
}

私の問題は、新しい測定値があることを知っていても、onLocationChanged ルーチンが呼び出されないことです。たとえば、パッシブ モードで読み取るように構成されている場合、Waze を実行すると、自分が動いているのが見えます。onLocationChanged の呼び出しがセマフォによってブロックされている可能性はありますか? もしそうなら、どうすればこれを回避できますか?requestLocationUpdate のパラメーターを動的に変更できるようにしたい。

4

1 に答える 1

1

タンブルウィード バッジを取得することは非常に疑わしい名誉だと思いますが、答えは次のとおりです。はい、.acquire() 呼び出しはスレッドをブロックし、onLocationChanged() メソッドは呼び出されません。

解決策は、ハンドラーに次のようなランナブルを実行させることです。

public void onLocationChanged(Location location) {
            xxxService.locHandler.post(xxxService.locationRun);
        }

どこ

Runnable locationRun = new Runnable() {

    @Override
    public void run() {
        // handle stuff for new location
    }
};

一般的な原則は、ポーリング ループを実行するのではなく、イベントを送信して処理することです。非イベントを処理する必要がある場合、たとえば新しい場所を読み取らない場合は、カウントダウン タイマーを設定します。

于 2013-09-02T08:12:44.343 に答える