Android Marshmallow 应用程序中有关位置的权限问题

3

我正在学习开发一个安卓应用程序,以获取设备位置,遵循Google开发者论坛:@http://developer.android.com/training/location/retrieve-current.html#last-known @http://developer.android.com/training/location/receive-location-updates.html

示例展示了如何使用Google Play服务来实现此目的。 但是,getLastLocation()函数总是返回null。

Location location = LocationServices.FusedLocationApi.getLastLocation(mGoogleApiClient);

由于最初的位置可能为空,因此我添加了以下行来接收定期位置更新

LocationServices.FusedLocationApi.requestLocationUpdates(mGoogleApiClient, mLocationRequest, this);

它抛出一个异常,带有以下消息

客户端必须具有ACCESS_COARSE_LOCATION或ACCESS_FINE_LOCATION权限才能执行任何位置操作

我已经在清单文件中添加了权限:

<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
我已经打开了 Google 地图来更新当前位置到 Google Play 服务中,但是问题仍然存在。 我是 Android 开发的新手,在发布这个问题之前,我已经进行了大量搜索,但没有找到适合我的解决方案。 我正在使用 Android Studio 1.3.2 进行开发,测试设备为 Android Marshmallow。 < p >AndroidManifest.xml< /p >
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="newray.acpsa1">
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme">
        <meta-data
            android:name="com.google.android.gms.version"
            android:value="@integer/google_play_services_version" />

        <meta-data
            android:name="com.google.android.maps.v2.API_KEY"
            android:value="@string/google_maps_key" />

        <activity
            android:name=".MapsActivity"
            android:label="@string/title_activity_maps">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

    </application>

</manifest>

Activity中的onCreate函数

 @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        mGoogleApiClient = new GoogleApiClient.Builder(this)
                .addConnectionCallbacks(this)
                .addOnConnectionFailedListener(this)
                .addApi(LocationServices.API)
                .build();
        mGoogleApiClient.connect();
        mLocationRequest = LocationRequest.create()
                .setPriority(LocationRequest.PRIORITY_HIGH_ACCURACY)
                .setInterval(10 * 1000)        
                .setFastestInterval(1 * 1000); 
        /* Rest functionality */
    }

onConnected函数

@Override
    public void onConnected(Bundle bundle) {
        try{
        Location location = LocationServices.FusedLocationApi.getLastLocation(mGoogleApiClient);
        LatLng loc = new LatLng(21.356993, 72.826647);
        if (location == null) {
            Log.d(TAG,"location is again null");
            LocationServices.FusedLocationApi.requestLocationUpdates(mGoogleApiClient, mLocationRequest, this);
            Log.d(TAG, "Requested for updates");
        }
        else {
            loc = new LatLng(location.getLatitude(),location.getLongitude());
            handleNewLocation(location);
        }            
        } catch (Exception e) {
            Log.d(TAG, "Error in onConnected.  "+e.getMessage());

        }
    }

日志:

XXXX: Error in onConnected.  Client must have ACCESS_COARSE_LOCATION or ACCESS_FINE_LOCATION permission to perform any location operations.

build.gradle文件中的依赖项:

dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    compile 'com.android.support:appcompat-v7:23.1.0'
    compile 'com.google.android.gms:play-services:8.1.0'
    compile 'com.google.android.gms:play-services-location:8.1.0'
}

1
http://developer.android.com/training/permissions/requesting.html - Daniel Nugent
谢谢@DanielNugent。那解决了我的问题。 - Naveen
2个回答

9

以下是针对您问题的简单解决方案/示例:

package com.my.packagename;

import android.Manifest;
import android.annotation.SuppressLint;
import android.app.Activity;
import android.content.Context;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {

private static final int PERMISSION_REQUEST_CODE_LOCATION = 1;

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    if (checkPermission(Manifest.permission.ACCESS_FINE_LOCATION,getApplicationContext(),CopyOfMap.this)) {
        fetchLocationData();
    }
    else
    {
        requestPermission(Manifest.permission.ACCESS_FINE_LOCATION,PERMISSION_REQUEST_CODE_LOCATION,getApplicationContext(),CopyOfMap.this);
    }
}

public static void requestPermission(String strPermission,int perCode,Context _c,Activity _a){

    if (ActivityCompat.shouldShowRequestPermissionRationale(_a,strPermission)){
                Toast.makeText(getApplicationContext(),"GPS permission allows us to access location data. Please allow in App Settings for additional functionality.",Toast.LENGTH_LONG).show();
            } else {

                ActivityCompat.requestPermissions(_a,new String[]{strPermission},perCode);
            }
    }

public static boolean checkPermission(String strPermission,Context _c,Activity _a){
        int result = ContextCompat.checkSelfPermission(_c, strPermission);
        if (result == PackageManager.PERMISSION_GRANTED){

            return true;

        } else {

            return false;

        }
    }

@Override
public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
    switch (requestCode) {

        case PERMISSION_REQUEST_CODE_LOCATION:
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {

                fetchLocationData();

            } else {

                Toast.makeText(getApplicationContext(),"Permission Denied, You cannot access location data.",Toast.LENGTH_LONG).show();

            }
            break;

    }
}


    private void fetchLocationData()
    {
    //code to use the granted permission (location)
    }
}

我把requestPermission()和checkPermission()方法设为了public static,因为我可以在其他需要请求权限的情况下重用它们,如电话呼叫、写入存储等。希望这能帮助到某些人。 :-)


@Annes U 你觉得 'PERMISSION_REQUEST_CODE_LOCATION' 怎么样? - kadik
虽然有点晚了,但如果能帮到其他人的话,@kadik,你需要在应用程序中定义一个int常量,稍后会在onRequestPermissionsResult函数中传回给你。 - hi0001234d
我正在做类似的事情,仍然遇到这个问题:http://stackoverflow.com/questions/41206885/unable-to-retrieve-location-using-locationmanager-network-provider-even-when-w' - Hammad Nasir

7
在Marshmallow中,Android SDK有很多新的东西。请求权限是其中的一个新更新,每个开发者都应该牢记。如果您的应用程序目标版本为23,则可能需要在开始使用之前向用户授权。只有在用户允许访问的情况下才能使用它。因此,在向用户授权时,开发人员需要注意所有测试用例。
例如,您的应用程序需要访问位置。在早期版本的Android中,在安装时会显示权限对话框,并授予清单文件中定义的权限给应用程序。这些事情在Marshmallow中发生了改变。现在,每个应用程序都必须从用户那里获取访问权限。
查看开发人员站点并学习如何指导用户进入应用程序以请求应用程序中的“运行时权限”的基本步骤。
有很多网站展示了如何在Marshmallow中请求权限的基本教程。以下是一些建议链接:使用运行时权限构建更好的应用程序

权限设计指南

这个链接会为您提供基本信息,以指导用户在请求权限之前进入应用程序。

谢谢。



网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接