Android Service中获取GPS适配

从 Android 10(API 级别 29)开始,Google 对后台服务获取 GPS 的行为进行了更严格的限制。这些限制在 Android 11(API 级别 30)中仍然有效,并且进一步加强了对隐私和用户数据的保护。

以下是关于 Android 11 中后台服务获取 GPS 的限制以及如何适配的详细说明:


1. 后台位置访问限制

从 Android 10 开始,应用在后台运行时访问位置信息需要额外的权限声明。具体来说:

  • 如果你的应用需要在后台访问位置信息,必须声明 ACCESS_BACKGROUND_LOCATIONFOREGROUND_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_LOCATIONFOREGROUND_SERVICE_LOCATION ,用户仍然需要手动授予“始终允许”权限。如果用户仅授予“仅在使用时允许”,则应用无法在后台访问位置信息。


2. 后台服务限制

Android 11 对后台服务的行为也施加了更多限制:

  • 前台服务要求:如果你的应用需要在后台持续获取位置信息,建议使用前台服务(Foreground Service)。前台服务需要显示一个通知,告知用户服务正在运行。
  • 后台启动限制:从 Android 10 开始,系统限制了后台应用启动活动(Activity)。虽然这不直接影响 GPS 获取,但可能影响用户体验。
2.1 前台服务示例

以下是一个使用前台服务获取位置信息的示例代码:

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;
    }
}
2.2 通知渠道

在 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);
    }
}
2.3 修改 AndroidManifest.xml

AndroidManifest.xml 文件中,为 元素添加 foregroundServiceType 属性,并根据服务的实际用途选择合适的类型。

示例代码:

<service
    android:name=".MyForegroundService"
    android:foregroundServiceType="location" />
2.4 支持多个类型

如果服务需要同时支持多种类型,可以使用按位或运算符(|)组合多个类型。例如:

<service
    android:name=".MyForegroundService"
    android:foregroundServiceType="location|camera" />

3. 动态适配不同版本

如果应用需要兼容 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);
}

你可能感兴趣的:(Android开发,android)