Android のAPILocationManager
は、ユーザーの位置を時折大まかに把握するだけでよいアプリケーションでは、使用するのは少し面倒なようです。
私が取り組んでいるアプリは、厳密には位置情報アプリではありませんが、近くのビジネスのリストを表示するために、ユーザーの位置情報を取得する必要があります。ユーザーが移動しているかどうかなどを気にする必要はありません。
私がやりたいことは次のとおりです:
- ユーザーに近くの場所のリストを表示します。
Activity
X で必要になったときに利用できるように、ユーザーの位置情報を事前にロードします。- 更新の精度や頻度については特に気にしていません。大きく外れていない限り、1 つの場所を取得するだけで十分です。凝ったことをしたい場合は、数分ごとに場所を更新するかもしれませんが、それほど優先事項ではありません。
- GPS またはネットワーク ロケーション プロバイダーのいずれかが搭載されている限り、どのデバイスでも動作します。
それほど難しいことではないように思えますが、2 つの異なる位置情報プロバイダー (GPS と NETWORK) を起動し、それぞれのライフサイクルを管理する必要があるようです。それだけでなく、#2 を満たすには、複数のアクティビティで同じコードを複製する必要があります。getBestProvider()
過去に、ソリューションを 1 つの位置情報プロバイダーのみに削減しようとしましたが、実際に最良の結果をもたらすプロバイダーではなく、最良の「理論上の」プロバイダーしか提供されないようです。
これを実現するより簡単な方法はありますか?
ベストアンサー1
私がやっていることは次のとおりです:
- まず、どのプロバイダーが有効になっているかを確認します。デバイス上で無効になっているプロバイダーもあれば、アプリケーション マニフェストで無効になっているプロバイダーもあります。
- 利用可能なプロバイダーがある場合は、ロケーション リスナーとタイムアウト タイマーを開始します。私の例では 20 秒ですが、GPS には十分ではない可能性があるため、時間を増やすことができます。
- ロケーション リスナーから更新を取得した場合、提供された値を使用します。リスナーとタイマーを停止します。
- 更新が取得されずにタイマーが経過した場合は、最後に確認された値を使用する必要があります。
- 利用可能なプロバイダーから最新の既知の値を取得し、その中で最新のものを選択します。
私のクラスの使い方は次のとおりです:
LocationResult locationResult = new LocationResult(){
@Override
public void gotLocation(Location location){
//Got the location!
}
};
MyLocation myLocation = new MyLocation();
myLocation.getLocation(this, locationResult);
MyLocation クラスは次のとおりです。
import java.util.Timer;
import java.util.TimerTask;
import android.content.Context;
import android.location.Location;
import android.location.LocationListener;
import android.location.LocationManager;
import android.os.Bundle;
public class 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(gps_enabled)
lm.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0, locationListenerGps);
if(network_enabled)
lm.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 0, 0, locationListenerNetwork);
timer1=new Timer();
timer1.schedule(new GetLastLocation(), 20000);
return true;
}
LocationListener locationListenerGps = new LocationListener() {
public void onLocationChanged(Location location) {
timer1.cancel();
locationResult.gotLocation(location);
lm.removeUpdates(this);
lm.removeUpdates(locationListenerNetwork);
}
public void onProviderDisabled(String provider) {}
public void onProviderEnabled(String provider) {}
public void onStatusChanged(String provider, int status, Bundle extras) {}
};
LocationListener locationListenerNetwork = new LocationListener() {
public void onLocationChanged(Location location) {
timer1.cancel();
locationResult.gotLocation(location);
lm.removeUpdates(this);
lm.removeUpdates(locationListenerGps);
}
public void onProviderDisabled(String provider) {}
public void onProviderEnabled(String provider) {}
public void onStatusChanged(String provider, int status, Bundle extras) {}
};
class GetLastLocation extends TimerTask {
@Override
public void run() {
lm.removeUpdates(locationListenerGps);
lm.removeUpdates(locationListenerNetwork);
Location net_loc=null, gps_loc=null;
if(gps_enabled)
gps_loc=lm.getLastKnownLocation(LocationManager.GPS_PROVIDER);
if(network_enabled)
net_loc=lm.getLastKnownLocation(LocationManager.NETWORK_PROVIDER);
//if there are both values use the latest one
if(gps_loc!=null && net_loc!=null){
if(gps_loc.getTime()>net_loc.getTime())
locationResult.gotLocation(gps_loc);
else
locationResult.gotLocation(net_loc);
return;
}
if(gps_loc!=null){
locationResult.gotLocation(gps_loc);
return;
}
if(net_loc!=null){
locationResult.gotLocation(net_loc);
return;
}
locationResult.gotLocation(null);
}
}
public static abstract class LocationResult{
public abstract void gotLocation(Location location);
}
}
誰かが私のロジックを変更したい場合もあります。たとえば、ネットワーク プロバイダーから更新を取得した場合、リスナーを停止せずに待機を続けます。GPS はより正確なデータを提供するため、待機する価値があります。タイマーが経過し、ネットワークから更新を取得したが GPS から取得していない場合は、ネットワークから提供された値を使用できます。
もう一つのアプローチはLocationClientを使うことですhttp://developer.android.com/training/location/retrieve-current.htmlただし、ユーザーのデバイスに Google Play 開発者サービス APK がインストールされている必要があります。