私はかなり簡単なコードを持っています。これは定期的に実行され、現在の場所を記録し(ネットワークプロバイダを使用して)、サーバーに送信してからスリープ状態に戻ります。Android:GPSリスナーがキャッシュされた位置データを取得する
私はこれを2種類の携帯電話でテストしています - SGS2には定期的な月額プランがあり、格安ZTEにはプリペイドSIMカード(データはありますが、分は10c /分です)があります。私は、携帯電話とドライブの両方を取ると、SGS2は正常に動作しますが、ZTEは修正を受ける能力を失うようです。
ZTEは目覚めて、リスナーをセットアップし、ロケーションフィックスを取得しますが、実際の現在のロケーションではなく、私の家(最後のWi-Fiベースのフィックスを入手した場所)をポイントします。その場所のタイムスタンプは最新のものなので、場所の更新を受け取ると、その場所が有効かどうか(SGS2のように、またはZTEが家にいるかどうか)や寝かせ(たとえばZTEを運転している)。
誰も以前に同様の問題を抱えていますか?プリペイドカードやZTEの電話機とは関係がありますか?残念ながら、私はSIMカードを交換することはできません(私は携帯電話のルート/アンロックが必要です)ので、私はそれをテストすることはできません。
私は以下のコードを追加しましたが、SGS2上で正常に動作するため、問題はほとんどないと思います。
public class LocationRecorder extends Service {
private volatile Location lastLocation;
private LocationManager locationManager;
private LocationListener locationListener;
private static volatile PowerManager.WakeLock wakeLock = null;
private static synchronized PowerManager.WakeLock getWakeLock(Context context) {
if (wakeLock == null) {
PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
wakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "LocationRecorder");
wakeLock.acquire();
}
return wakeLock;
}
public static void startLocationRecorder(Context context, Intent service) {
getWakeLock(context);
context.startService(service);
}
@Override
public void onCreate() {
Log.d("LocationRecorder", "Starting Location Service");
locationManager = ((LocationManager)getSystemService(LOCATION_SERVICE));
locationListener = new LocationListener() {
@Override
public void onLocationChanged(Location location) {
Log.d("Location Changed", location.toString());
if (location.getExtras()!=null) {
String x = "";
for (String key : location.getExtras().keySet()) {
x+=key+":"+location.getExtras().get(key).toString()+", ";
}
Log.d("Location Changed Extras", x);
}
setLocation(location);
}
public void onStatusChanged(String provider, int status, Bundle extras) {
Log.d("Status Changed", provider+" "+status);
if (extras!=null) {
String x = "";
for (String key : extras.keySet()) {
x+=key+":"+extras.get(key).toString()+", ";
}
Log.d("Status Changed Extras", x);
}
}
public void onProviderEnabled(String provider) {
Log.d("Provider Enabled", provider);
}
public void onProviderDisabled(String provider) {
Log.d("Provider Disabled", provider);
}
};
locationManager.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 0, 0, locationListener);
waitForLocation();
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
return Service.START_REDELIVER_INTENT;
}
@Override
public void onDestroy() {
locationManager.removeUpdates(locationListener);
try {
wakeLock.release();
wakeLock = null;
}
catch (Exception e) {
wakeLock = null;
}
Log.d("LocationRecorder", "Destroying service");
super.onDestroy();
}
protected void waitForLocation() {
new Thread() {
@Override
public void run() {
setLocation(null);
for (int i=0; i<3;i++) {
Log.d("LocationRecorder", "Waiting for location");
try {Thread.sleep(10000);} catch(Exception e) {};
if (getLocation() != null) {
Log.d("LocationRecorder", "Sending new location!");
new Utilities(LocationRecorder.this).updateLocation(getLocation().getLatitude(),
getLocation().getLongitude(), getLocation().getAccuracy());
break;
}
}
stopSelf();
}
}.start();
}
public synchronized void setLocation(Location newLocation) {
lastLocation = newLocation;
}
public synchronized Location getLocation() {
return lastLocation;
}
@Override
public IBinder onBind(Intent intent) {
return null;
}
それは意味がありますが、私が受け取った修正が実際に古いかどうかはどうしたら分かりますか?ロケーションタイムスタンプが最新であることが示されているので、エクストラが「wifi」または「セル」から来ているかどうかを示しているので、ロケーションオブジェクト(そのエクストラを含む)を調べることでそれを把握することはできません。networkLocationSource常にキャッシュされます。 – Martin
「Location」オブジェクトの時刻を追加するのではなく、[location.getTime()](http://developer.android.com/reference/android/location/Location.html#getTime() )。これはキャッシュされた/古いものであっても正確な修正時間を得ることを保証します。 –