从 Android 10(API 级别 29)开始,Google 对后台服务获取 GPS 的行为进行了更严格的限制。这些限制在 Android 11(API 级别 30)中仍然有效,并且进一步加强了对隐私和用户数据的保护。
以下是关于 Android 11 中后台服务获取 GPS 的限制以及如何适配的详细说明:
从 Android 10 开始,应用在后台运行时访问位置信息需要额外的权限声明。具体来说:
ACCESS_BACKGROUND_LOCATION
和 FOREGROUND_SERVICE_LOCATION
权限。在 AndroidManifest.xml
文件中,你需要声明以下权限:
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_BACKGROUND_LOCATION" />
<uses-permission android:name="android.permission.FOREGROUND_SERVICE_LOCATION" />
即使你在清单文件中声明了 ACCESS_BACKGROUND_LOCATION
和 FOREGROUND_SERVICE_LOCATION
,用户仍然需要手动授予“始终允许”权限。如果用户仅授予“仅在使用时允许”,则应用无法在后台访问位置信息。
Android 11 对后台服务的行为也施加了更多限制:
以下是一个使用前台服务获取位置信息的示例代码:
public class LocationService extends Service {
private FusedLocationProviderClient fusedLocationClient;
private LocationCallback locationCallback;
@Override
public void onCreate() {
super.onCreate();
// 初始化 FusedLocationProviderClient
fusedLocationClient = LocationServices.getFusedLocationProviderClient(this);
// 定义位置回调
locationCallback = new LocationCallback() {
@Override
public void onLocationResult(LocationResult locationResult) {
if (locationResult == null) return;
for (Location location : locationResult.getLocations()) {
// 处理位置更新
Log.d("LocationService", "Latitude: " + location.getLatitude() + ", Longitude: " + location.getLongitude());
}
}
};
// 启动前台服务
Notification notification = new NotificationCompat.Builder(this, "location_channel")
.setContentTitle("Location Service")
.setContentText("Running in the background")
.setSmallIcon(R.drawable.ic_location)
.build();
startForeground(1, notification);
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
requestLocationUpdates();
return START_STICKY;
}
private void requestLocationUpdates() {
LocationRequest locationRequest = LocationRequest.create();
locationRequest.setInterval(10000); // 每 10 秒更新一次
locationRequest.setFastestInterval(5000); // 最快每 5 秒更新一次
locationRequest.setPriority(LocationRequest.PRIORITY_HIGH_ACCURACY);
fusedLocationClient.requestLocationUpdates(locationRequest, locationCallback, Looper.getMainLooper());
}
@Override
public void onDestroy() {
super.onDestroy();
fusedLocationClient.removeLocationUpdates(locationCallback);
}
@Override
public IBinder onBind(Intent intent) {
return null;
}
}
在 Android 8.0(API 级别 26)及以上版本中,前台服务需要创建通知渠道:
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
NotificationChannel channel = new NotificationChannel(
"location_channel",
"Location Service Channel",
NotificationManager.IMPORTANCE_LOW
);
NotificationManager manager = getSystemService(NotificationManager.class);
if (manager != null) {
manager.createNotificationChannel(channel);
}
}
AndroidManifest.xml
在 AndroidManifest.xml
文件中,为
元素添加 foregroundServiceType
属性,并根据服务的实际用途选择合适的类型。
示例代码:
<service
android:name=".MyForegroundService"
android:foregroundServiceType="location" />
如果服务需要同时支持多种类型,可以使用按位或运算符(|
)组合多个类型。例如:
<service
android:name=".MyForegroundService"
android:foregroundServiceType="location|camera" />
如果应用需要兼容 Android 10 以下版本,可以在代码中动态检查 API 级别,并仅在必要时设置 foregroundServiceType
。
示例代码:
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
startForeground(1, notification, ServiceInfo.FOREGROUND_SERVICE_TYPE_LOCATION);
} else {
startForeground(1, notification);
}