0

私はロケーションリスナーサービスを使用して、数秒ごとにユーザーの現在のロケーションを更新しています。また、WakeLockを使ってリスナをバックグラウンドで動作させました。今はすべて正常に動作していますが、私は位置リスナを停止することができません。アプリが強制終了されても、リスナーはユーザーの場所をDBに更新しています。ロケーションリスナーサービスを停止する方法は? (現在停止していない)

場所のリスナー:

private class LogOutTimerTask extends TimerTask 
{ 
    @Override 
    public void run() 
    { 
     Intent i = new Intent(MapsActivity.this, Login.class); 
     i.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP); 
     startActivity(i); 
     Intent intent = new Intent(MapsActivity.this, MyLocationService.class); 
     stopService(intent); 
     finish(); 
    } 
} 
0123(mainactivityから):意図を停止

@Override 
protected void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.activity_maps); 


    Intent serviceIntent = new Intent(context , MyLocationService.class); 
    context.startService(serviceIntent ); 



    if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { 
     checkLocationPermission(); 
    } 
    if (!isGooglePlayServicesAvailable()) { 
     finish(); 
    } else { 
     Log.d("onCreate", "Google Play Services available."); 
    } 
    SupportMapFragment mapFragment = (SupportMapFragment) getSupportFragmentManager().findFragmentById(R.id.map); 
    mapFragment.getMapAsync(this); 
} 

(mainactivityから):

import android.app.Service; 
import android.content.Context; 
import android.content.Intent; 
import android.content.pm.PackageManager; 
import android.location.Location; 
import android.location.LocationManager; 
import android.os.Bundle; 
import android.os.IBinder; 
import android.os.PowerManager; 
import android.support.v4.app.ActivityCompat; 
import android.util.Log; 
import android.widget.Toast; 

import com.google.android.gms.maps.model.LatLng; 

public class MyLocationService extends Service { 
private static final String TAG = "MyLocationService"; 
private LocationManager mLocationManager = null; 
private static final int LOCATION_INTERVAL = 3000; 
private static final float LOCATION_DISTANCE = 0f; 
Context context = this; 
PowerManager.WakeLock wakeLock; 
public int e=0; 

LocationListener[] mLocationListeners = new LocationListener[]{ 
     new LocationListener(LocationManager.GPS_PROVIDER), 
     new LocationListener(LocationManager.NETWORK_PROVIDER) 
}; 

@Override 
public IBinder onBind(Intent arg0) { 
    return null; 
} 

@Override 
public int onStartCommand(Intent intent, int flags, int startId) { 
    Log.e(TAG, "onStartCommand"); 
    super.onStartCommand(intent, flags, startId); 
    Toast.makeText(this, "MyService Started.", Toast.LENGTH_SHORT).show(); 
    return START_STICKY; 
} 

@Override 
public void onCreate() { 
    Log.e(TAG, "onCreate"); 
    initializeLocationManager(); 
    PowerManager mgr = (PowerManager)context.getSystemService(Context.POWER_SERVICE); 
    wakeLock= mgr.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "MyWakeLock"); 
    wakeLock.acquire(); 
    try { 
     mLocationManager.requestLocationUpdates(
       LocationManager.NETWORK_PROVIDER, LOCATION_INTERVAL, LOCATION_DISTANCE, 
       mLocationListeners[1]); 
    } catch (java.lang.SecurityException ex) { 
     Log.i(TAG, "fail to request location update, ignore", ex); 
    } catch (IllegalArgumentException ex) { 
     Log.d(TAG, "network provider does not exist, " + ex.getMessage()); 
    } 
    try { 
     mLocationManager.requestLocationUpdates(
       LocationManager.GPS_PROVIDER, LOCATION_INTERVAL, LOCATION_DISTANCE, 
       mLocationListeners[0]); 
    } catch (java.lang.SecurityException ex) { 
     Log.i(TAG, "fail to request location update, ignore", ex); 
    } catch (IllegalArgumentException ex) { 
     Log.d(TAG, "gps provider does not exist " + ex.getMessage()); 
    } 
} 

@Override 
public void onDestroy() { 
    Log.e(TAG, "onDestroy"); 
    wakeLock.release(); 
    super.onDestroy(); 


    if (mLocationManager != null) { 
     for (int i = 0; i < mLocationListeners.length; i++) { 
      try { 
       if (ActivityCompat.checkSelfPermission(this, android.Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED && ActivityCompat.checkSelfPermission(this, android.Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED) { 
        // TODO: Consider calling 
        // ActivityCompat#requestPermissions 
        // here to request the missing permissions, and then overriding 
        // public void onRequestPermissionsResult(int requestCode, String[] permissions, 
        //           int[] grantResults) 
        // to handle the case where the user grants the permission. See the documentation 
        // for ActivityCompat#requestPermissions for more details. 
        return; 
       } 
       mLocationManager.removeUpdates(mLocationListeners[i]); 
      } catch (Exception ex) { 
       Log.i(TAG, "fail to remove location listners, ignore", ex); 
      } 
     } 
    } 
} 

private void initializeLocationManager() { 
    Log.e(TAG, "initializeLocationManager"); 
    if (mLocationManager == null) { 
     mLocationManager = (LocationManager) getApplicationContext().getSystemService(Context.LOCATION_SERVICE); 
    } 
} 

// create LocationListener class to get location updates 
private class LocationListener implements android.location.LocationListener 
{ 
    Location mLastLocation; 
    double latitude; 
    double longitude; 
    LatLng latLngcurrent; 


    public LocationListener(String provider) 
    { 
     Log.e(TAG, "LocationListener " + provider); 
     mLastLocation = new Location(provider); 
    } 

    @Override 
    public void onLocationChanged(Location location) 
    { 
     Log.e(TAG, "onLocationChanged: " + location); 
     mLastLocation.set(location); 
     latitude = location.getLatitude(); 
     longitude = location.getLongitude(); 
     Store.latu=latitude; 
     Store.longu=longitude; 
     latLngcurrent = new LatLng(location.getLatitude(), location.getLongitude()); 
     Toast.makeText(context,"Location " + String.valueOf(e), Toast.LENGTH_SHORT).show(); 
     e++; 
     Toast.makeText(MyLocationService.this,Store.latu+" "+Store.longu, Toast.LENGTH_SHORT).show(); 

    } 

    @Override 
    public void onProviderDisabled(String provider) 
    { 
     Log.e(TAG, "onProviderDisabled: " + provider); 
    } 

    @Override 
    public void onProviderEnabled(String provider) 
    { 
     Log.e(TAG, "onProviderEnabled: " + provider); 
    } 

    @Override 
    public void onStatusChanged(String provider, int status, Bundle extras) 
    { 
     Log.e(TAG, "onStatusChanged: " + provider); 
    } 
} 
} 

は、サービスの意図を開始

ユーザーのログアウト時に停止インテントが呼び出され、サービスクラスのonDestroy()メソッドが呼び出されても、プロセスを停止していない状態になりました。アプリケーションが強制的に閉じるとき、onDestroy()は呼び出されません。どちらの場合もプロセスは再開し、値をDBに更新し続けます。事前におねがいします。

+0

は 'removeUpdates()'が呼び出されていますか?またはメソッド 'IF'のためにメソッドが返されますか?しかし、なぜリスナーを削除するときに許可を確認しているのですか?これは 'initializeLocationManager()'のどこかにあるはずです – Yazan

+0

removeupdates()が呼び出されているかどうかはチェックしていません。私はアンドロイドスタジオが私を推薦していたので、権限を追加しました。 @ Yazan –

+1

私はこの場所が許可を要求するのは良いとは思わないが(サービスを破壊している間に!)、ループに入れるのはあまり良いことではない。この 'Log.i(TAG、"リスナーを削除... ");'のような何かの 'removeUpdates()'の直前のログをlogcatに表示するかどうかを確認します – Yazan

答えて

1

解決策: 私はついにこの解決策を見つけました。 @ Yazanは、ユーザーがログアウトしたときにサービスを手動で停止するのに役立つonDestroy()メソッドからアクセス許可を削除したと述べています。アプリが強制終了されたときにサービスを停止するには、タグを使用しているマニフェストでこの行も追加します。

スニペット:

<service android:name=".MyLocationService" 
android:stopWithTask="true"></service> 

我々はアプリケーションを強制終了する場合stopWithTaskがサービスを停止し、LocationListenerも停止します。

hereはドキュメントです。これを追加した後、私のアプリケーションは正常に動作します。

関連する問題