從Android 8.0開始系統(tǒng)為實現(xiàn)降低功耗,對后臺應(yīng)用獲取用戶位置信息頻率進行了限制杈湾,每小時只允許更新幾次位置信息攘须,詳細信息請參考官方說明。按照官方指引于宙,如果要提高位置更新頻率,需要后臺應(yīng)用提供一個前臺服務(wù)通知告知至会。
所以原來的單單使用locationManager獲得當前位置在后臺情況下無法使用了谱俭。于是打算使用一個前臺服務(wù),當app在后臺時也能獲得當前位置县貌。
查了幾篇博客說前臺服務(wù)需要在service的onStartCommand方法中調(diào)用startForeground(int, Notification)才能開啟前臺服務(wù)凑懂。
但是onStartCommand需要走startservice()的生命周期才會調(diào)用。
我改用了bindservice() 正好需要activity和service交互,當然兩個啟動方法混用也可以塘匣。但是沒有必要兆解。
我需要的只是和控件綁定的service并且不想處理服務(wù)的結(jié)束操作。
- activity / fragment調(diào)用 綁定服務(wù)
Intent serviceIntent = new Intent(this, ForegroundLocationService.class);
bindService(serviceIntent, conn, Service.BIND_AUTO_CREATE);
// 綁定服務(wù)時要求傳入一個ServiceConnection實現(xiàn)類的對象
// 綁定服務(wù)時埠巨,會觸發(fā)服務(wù)的onBind方法现拒,此方法會返回一個Ibinder的對象給activity / fragment的onServiceConnected(),通過這個對象可以訪問服務(wù)中的方法
ServiceConnection conn = new ServiceConnection() {
@Override
public void onServiceDisconnected(ComponentName name) {
}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
}
};
我在onBind()方法中調(diào)用了startForeground(int, Notification)
第一個參數(shù)是一個不為0的正整數(shù)勋桶,代表通知的id侥猬,第二個參數(shù)代表需要顯示的通知。
適配8.0的通知構(gòu)建需要適配退唠,不然會導(dǎo)致你的通知無法顯示(第一次調(diào)用的時候還以為是一加攔截了通知)那么這時候應(yīng)該已經(jīng)實現(xiàn)了前臺服務(wù)瞧预,需要把服務(wù)獲得的位置信息傳遞給activity。(直接調(diào)用locationmanager就可以獲得垢油,這里把位置實現(xiàn)隱去)
public class MyBinder extends Binder {
public ForegroundLocationService getService(){
return ForegroundLocationService.this;
}
}
//通過binder實現(xiàn)調(diào)用者client與Service之間的通信
private MyBinder binder = new MyBinder();
//通過service的onBind()方法返回我們實例化的MyBinder對象,該對象可以獲的當前的Service
@Override
public IBinder onBind(Intent arg0) {
NotificationUtils notificationUtils = new NotificationUtils(this);
startForeground(111, notificationUtils.getNotification("Notice", "Continuous positioning",null));
return binder;
}
- 然后需要進行控件和服務(wù)的交互躯喇,這里就分成了三種方法
- 在得到service的情況下act主動調(diào)用得到數(shù)據(jù)
- 在service中設(shè)置回調(diào)惊楼,service主動傳遞數(shù)據(jù)給act
- 通過廣播傳遞數(shù)據(jù)。
ServiceConnection conn = new ServiceConnection() {
@Override
public void onServiceDisconnected(ComponentName name) {
}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
//通過這個方法可以得到service的實例,通過設(shè)置回調(diào)可以持續(xù)更新
ForegroundLocationService foregroundLocationService = ((ForegroundLocationService.MyBinder) service).getService();
foregroundLocationService.setLocationCallback(new ForegroundLocationService.LocationCallback() {
@Override
public void onLocation(Location location) {
}
})弧可;
}
};
在service中編寫接口,并在獲得位置的回調(diào)方法中調(diào)用棕诵。
public interface LocationCallback {
/**
* 當前位置
*/
void onLocation(Location location);
}
private LocationCallback mLocationCallback;
private class LocationListener implements android.location.LocationListener {
public LocationListener(String provider) {
Logger.e(TAG, "LocationListener " + provider);
}
@Override
public void onLocationChanged(Location location) {
Log.i("location", "onLocationChanged: " + "當前坐標:" + location.getLatitude() + " : " + location.getLongitude());
if(mLocationCallback!=null){
mLocationCallback.onLocation(location);
}
}
}
Service向Activity發(fā)送消息校套,可以使用廣播,當然Activity要注冊相應(yīng)的接收器笛匙。比如Service要向多個Activity發(fā)送同樣的消息的話,用這種方法就更好秋柄,這里就省略不寫了蠢正。具體可以參考下面的文章。