2

バックグラウンド サービスで現在地の更新を実行しようとしています。サービスは、ソケット通信など、すでに他の多くのことを行っている独自のワーカースレッドを実行しています。ロケーションの更新も処理したいのですが、これはアクティビティでしか機能しないようです。私が読む限り、これはワーカースレッドでメッセージループが欠落していることが原因です。Looper.prepare()どこかで使用する必要があると思いますが、場所のリクエストを処理するためだけに別のスレッドが必要になるのでしょうか? エミュレーターを地理修正イベントに応答させることができないようです。何か間違ったことをしているに違いありません。

以下は、関連のない部分をすべて取り除いたサービス コードです。

public class MyService extends Service implements LocationListener {
    private Thread runner;
    private volatile boolean keepRunning;
    private LocationManager locationManager = null;


    @Override
    public void onCreate() {
        keepRunning = true;

        runner = new Thread(null, new Runnable() {
            public void run() { workerLoop(); }
        });
        runner.start();
        startGps();
    }

    @Override
    public void onDestroy() {
        keepRunning = false;
        runner.interrupt();
    }

    private void workerLoop() { 
        //Looper.myLooper(); How does this work??
        //Looper.prepare();

        // Main worker loop for the service
        while (keepRunning) {
            try {
                if (commandQueue.notEmpty()) {
                    executeJob();
                } else {
                    Thread.sleep(1000);
                }
            } catch (Exception e) {
            }
        }   
        stopGps();
    }

    public void onLocationChanged(Location location) {
        // Doing something with the position...
    }       
    public void onProviderDisabled(String provider) {}
    public void onProviderEnabled(String provider) {}
    public void onStatusChanged(String provider, int status, Bundle extras) {}

    private void startGps() {
        if (locationManager == null)
            locationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
        if (locationManager != null) {
            Criteria criteria = new Criteria();
            criteria.setAccuracy(Criteria.ACCURACY_FINE);
            criteria.setAltitudeRequired(false);
            criteria.setBearingRequired(true);
            criteria.setCostAllowed(false);
            criteria.setSpeedRequired(true);
            String provider = locationManager.getBestProvider(criteria, true);
            if (provider != null)
                locationManager.requestLocationUpdates(provider, 10, 5, (LocationListener) this);           
        }       
    }

    private void stopGps() {
        if (locationManager != null)
            locationManager.removeUpdates((LocationListener) this);
        locationManager = null;
    }

}
4

1 に答える 1

0

LocationListener は、それがアクティビティまたはサービスのコンテキストにあるかどうかを気にしません。workerLoop() で場所の udates を使用したいですね。ロケーションの更新 (LocationListener の呼び出し) と workerLoop() は、どちらも互いに独立して動作します。workerLoop() への場所の更新を取得するには、2 つのスレッドに参加する必要があります。これを行う 1 つの方法は、黒板パターンを使用することです。LocationListener は、新しい場所をクラスのフィールドに書き込みます (両方が同じクラスのコンテキストにあるため、これは簡単です)。

private Location blackboard = null;
public void onLocationChanged(final Location location) {
    if( location != null )
        this.blackboard = location;
}

private void workerLoop() {
    ...
    if( this.blackboard != null ) {
        final Location locationUpdate = this.blackboard;
        this.blackboard = null;
        // .. do something with the location
    }
    ...
}

上記のコードでは、workerLoop() が場所を読み取ったり消去したりしている間に LocationListener が黒板に書き込むと、競合状態が発生する可能性があります。これは、次のような同期ブロックで黒板へのアクセスを囲むことで解決できます。

synchronized(this) {
        this.blackboard = location
}

同様に workerLoop() で、黒板を volatile と宣言する必要があります。

private volatile Location blackboard = null;

または、ロックを使用することもできます。詳細については、ドキュメントを参照してください: http://docs.oracle.com/javase/tutorial/essential/concurrency/newlocks.html

于 2012-03-12T09:21:37.780 に答える