Android“ gps ต้องการข้อผิดพลาด ACCESS_FINE_LOCATION” แม้ว่าไฟล์ Manifest ของฉันจะมีสิ่งนี้ก็ตาม


104

ทุกครั้งที่ฉันเรียกใช้แอปพลิเคชัน SecurityException ของฉันจะถูกโยนทิ้งและข้อผิดพลาดจากดีบักเกอร์จะอ่านดังนี้:

java.lang.SecurityException: ผู้ให้บริการตำแหน่ง "gps" ต้องการสิทธิ์ ACCESS_COARSE_LOCATION หรือ ACCESS_FINE_LOCATION

ดูเหมือนจะเป็นข้อผิดพลาดง่ายๆอย่างไรก็ตามไฟล์ Manifest ของฉันถูกต้องสมบูรณ์ นี่คือและนี่คือรหัส MapActivity ของฉันเช่นกัน:

<?xml version="1.0" encoding="utf-8"?>

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="com.google.android.providers.gsf.permission.READ_GSERVICES" />
<uses-permission android:name="com.dev.cromer.jason.coverme.permission.MAPS_RECEIVE" />

<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:theme="@style/AppTheme" >
    <activity
        android:name=".MainActivity"
        android:label="@string/app_name" >
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>

    <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=".MapActivity"
        android:label="@string/title_activity_map" >
    </activity>
</application>

กิจกรรมของฉัน:

    package com.dev.cromer.jason.coverme;

import android.location.Criteria;
import android.location.Location;
import android.location.LocationListener;
import android.location.LocationManager;
import android.support.v4.app.FragmentActivity;
import android.os.Bundle;
import android.util.Log;

import com.google.android.gms.common.ConnectionResult;
import com.google.android.gms.common.api.GoogleApiClient;
import com.google.android.gms.location.LocationServices;
import com.google.android.gms.maps.CameraUpdateFactory;
import com.google.android.gms.maps.GoogleMap;
import com.google.android.gms.maps.SupportMapFragment;
import com.google.android.gms.maps.model.LatLng;
import com.google.android.gms.maps.model.MarkerOptions;

public class MapActivity extends FragmentActivity implements LocationListener {

    private GoogleMap mMap; // Might be null if Google Play services APK is not available.

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

        setUpMapIfNeeded();
    }

    @Override
    protected void onResume() {
        super.onResume();
        setUpMapIfNeeded();
    }



    private void setUpMapIfNeeded() {
        // Do a null check to confirm that we have not already instantiated the map.
        if (mMap == null) {
            // Try to obtain the map from the SupportMapFragment.
            mMap = ((SupportMapFragment) getSupportFragmentManager().findFragmentById(R.id.map))
                    .getMap();

            // Check if we were successful in obtaining the map.
            if (mMap != null) {
                //mMap.setMyLocationEnabled(true);
                //mMap.setOnMyLocationChangeListener(this);
                setUpMap();
            }
        }
    }


    private void setUpMap() {
        mMap.addMarker(new MarkerOptions().position(new LatLng(0, 0)).title("Marker"));
        mMap.setMyLocationEnabled(true);

        LocationManager locationManager = (LocationManager) getSystemService(LOCATION_SERVICE);

        try {
            Location myLocation = locationManager.getLastKnownLocation(LocationManager.GPS_PROVIDER);

            if (myLocation != null) {
                Log.d("TAG", "Not null");
            }
            else {
                Log.d("TAG", "NULL");
                locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0, this);
            }
        }
        catch (SecurityException se) {
            Log.d("TAG", "SE CAUGHT");
            se.printStackTrace();
        }
    }


    @Override
    public void onLocationChanged(Location location) {
        Log.d("CHANGED", "LOCATION UPDATED");

    }

    @Override
    public void onStatusChanged(String provider, int status, Bundle extras) {

    }

    @Override
    public void onProviderEnabled(String provider) {

    }

    @Override
    public void onProviderDisabled(String provider) {

    }
}

คุณกำลังทดสอบ Android เวอร์ชันใดอยู่
CommonsWare

4
ไม่เกี่ยว แต่ถ้าคุณขอตำแหน่งที่ดีคุณไม่จำเป็นต้องขอแบบหยาบ รวมอยู่ด้วย
joey_g216

คำตอบ:


136

ACCESS_COARSE_LOCATION, ACCESS_FINE_LOCATIONและWRITE_EXTERNAL_STORAGEเป็นส่วนหนึ่งของระบบ Android ได้รับอนุญาตรันไทม์ 6.0 นอกเหนือจากการมีไว้ในไฟล์ Manifest แล้วคุณยังต้องขอจากผู้ใช้ในขณะรันไทม์ (ใช้requestPermissions()) และดูว่าคุณมี (ใช้checkSelfPermission()) หรือไม่

วิธีแก้ปัญหาอย่างหนึ่งในระยะสั้นคือการลดลงtargetSdkVersionต่ำกว่า 23

แต่ในที่สุดคุณจะต้องอัปเดตแอปของคุณเพื่อใช้ระบบสิทธิ์รันไทม์

ตัวอย่างเช่นกิจกรรมนี้ทำงานโดยใช้สิทธิ์ห้าสิทธิ์ สี่คือสิทธิ์รันไทม์แม้ว่าปัจจุบันจะจัดการเพียงสาม (ฉันเขียนไว้ก่อนที่จะWRITE_EXTERNAL_STORAGEเพิ่มลงในบัญชีรายชื่อสิทธิ์รันไทม์)

/***
 Copyright (c) 2015 CommonsWare, LLC
 Licensed under the Apache License, Version 2.0 (the "License"); you may not
 use this file except in compliance with the License. You may obtain a copy
 of the License at http://www.apache.org/licenses/LICENSE-2.0. Unless required
 by applicable law or agreed to in writing, software distributed under the
 License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
 OF ANY KIND, either express or implied. See the License for the specific
 language governing permissions and limitations under the License.

 From _The Busy Coder's Guide to Android Development_
 https://commonsware.com/Android
 */

package com.commonsware.android.permmonger;

import android.Manifest;
import android.app.Activity;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.TextView;
import android.widget.Toast;

public class MainActivity extends Activity {
  private static final String[] INITIAL_PERMS={
    Manifest.permission.ACCESS_FINE_LOCATION,
    Manifest.permission.READ_CONTACTS
  };
  private static final String[] CAMERA_PERMS={
    Manifest.permission.CAMERA
  };
  private static final String[] CONTACTS_PERMS={
      Manifest.permission.READ_CONTACTS
  };
  private static final String[] LOCATION_PERMS={
      Manifest.permission.ACCESS_FINE_LOCATION
  };
  private static final int INITIAL_REQUEST=1337;
  private static final int CAMERA_REQUEST=INITIAL_REQUEST+1;
  private static final int CONTACTS_REQUEST=INITIAL_REQUEST+2;
  private static final int LOCATION_REQUEST=INITIAL_REQUEST+3;
  private TextView location;
  private TextView camera;
  private TextView internet;
  private TextView contacts;
  private TextView storage;

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

    location=(TextView)findViewById(R.id.location_value);
    camera=(TextView)findViewById(R.id.camera_value);
    internet=(TextView)findViewById(R.id.internet_value);
    contacts=(TextView)findViewById(R.id.contacts_value);
    storage=(TextView)findViewById(R.id.storage_value);

    if (!canAccessLocation() || !canAccessContacts()) {
      requestPermissions(INITIAL_PERMS, INITIAL_REQUEST);
    }
  }

  @Override
  protected void onResume() {
    super.onResume();

    updateTable();
  }

  @Override
  public boolean onCreateOptionsMenu(Menu menu) {
    getMenuInflater().inflate(R.menu.actions, menu);

    return(super.onCreateOptionsMenu(menu));
  }

  @Override
  public boolean onOptionsItemSelected(MenuItem item) {
    switch(item.getItemId()) {
      case R.id.camera:
        if (canAccessCamera()) {
          doCameraThing();
        }
        else {
          requestPermissions(CAMERA_PERMS, CAMERA_REQUEST);
        }
        return(true);

      case R.id.contacts:
        if (canAccessContacts()) {
          doContactsThing();
        }
        else {
          requestPermissions(CONTACTS_PERMS, CONTACTS_REQUEST);
        }
        return(true);

      case R.id.location:
        if (canAccessLocation()) {
          doLocationThing();
        }
        else {
          requestPermissions(LOCATION_PERMS, LOCATION_REQUEST);
        }
        return(true);
    }

    return(super.onOptionsItemSelected(item));
  }

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

    switch(requestCode) {
      case CAMERA_REQUEST:
        if (canAccessCamera()) {
          doCameraThing();
        }
        else {
          bzzzt();
        }
        break;

      case CONTACTS_REQUEST:
        if (canAccessContacts()) {
          doContactsThing();
        }
        else {
          bzzzt();
        }
        break;

      case LOCATION_REQUEST:
        if (canAccessLocation()) {
          doLocationThing();
        }
        else {
          bzzzt();
        }
        break;
    }
  }

  private void updateTable() {
    location.setText(String.valueOf(canAccessLocation()));
    camera.setText(String.valueOf(canAccessCamera()));
    internet.setText(String.valueOf(hasPermission(Manifest.permission.INTERNET)));
    contacts.setText(String.valueOf(canAccessContacts()));
    storage.setText(String.valueOf(hasPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE)));
  }

  private boolean canAccessLocation() {
    return(hasPermission(Manifest.permission.ACCESS_FINE_LOCATION));
  }

  private boolean canAccessCamera() {
    return(hasPermission(Manifest.permission.CAMERA));
  }

  private boolean canAccessContacts() {
    return(hasPermission(Manifest.permission.READ_CONTACTS));
  }

  private boolean hasPermission(String perm) {
    return(PackageManager.PERMISSION_GRANTED==checkSelfPermission(perm));
  }

  private void bzzzt() {
    Toast.makeText(this, R.string.toast_bzzzt, Toast.LENGTH_LONG).show();
  }

  private void doCameraThing() {
    Toast.makeText(this, R.string.toast_camera, Toast.LENGTH_SHORT).show();
  }

  private void doContactsThing() {
    Toast.makeText(this, R.string.toast_contacts, Toast.LENGTH_SHORT).show();
  }

  private void doLocationThing() {
    Toast.makeText(this, R.string.toast_location, Toast.LENGTH_SHORT).show();
  }
}

(จาก โครงการตัวอย่างนี้ )

สำหรับฟังก์ชัน requestPermissions () พารามิเตอร์ควรเป็นเพียง "ACCESS_COARSE_LOCATION" หรือไม่ หรือฉันควรใส่ชื่อเต็ม "android.permission.ACCESS_COARSE_LOCATION"

ฉันจะใช้ค่าคงที่ที่กำหนดไว้ Manifest.permissionดังที่แสดงด้านบน

นอกจากนี้รหัสคำขอคืออะไร?

ซึ่งจะส่งกลับมาให้คุณเป็นพารามิเตอร์แรกเพื่อonRequestPermissionsResult()ให้คุณสามารถบอกrequestPermissions()สายหนึ่งจากอีกสายหนึ่งได้


1
สำหรับฟังก์ชัน requestPermissions () พารามิเตอร์ควรเป็นเพียง "ACCESS_COARSE_LOCATION" หรือไม่ หรือฉันควรใส่ชื่อเต็ม "android.permission.ACCESS_COARSE_LOCATION"
Jason Cromer

1
ขอบคุณสิ่งนี้กำจัดข้อผิดพลาด ฉันยังคงประสบปัญหาในการเข้าถึงตำแหน่งของฉันเนื่องจาก locationManager ของฉันยังคงส่งคืนตำแหน่งของฉันเป็นโมฆะ แต่นั่นไม่เกี่ยวข้องกับข้อบกพร่องนี้ ขอบคุณสำหรับการแก้ปัญหาของคุณ!
Jason Cromer

@CommonsWare: คุณหมายถึงอะไรเมื่อพูดว่า 'ในที่สุด'? ขอโทษค่ะฉันไม่เข้าใจส่วนนั้น
theapache64

1
@ theapache64: สักวันมีบางอย่างจะทำให้คุณอยากตั้งค่าtargetSdkVersionเป็น 23 หรือสูงกว่า เมื่อถึงจุดนั้นคุณจะต้องใช้ระบบสิทธิ์รันไทม์ จนกว่าจะถึงเวลานั้นคุณสามารถรักษาtargetSdkVersionระดับอายุต่ำกว่า 23 ปีไว้ได้และละเว้นสิทธิ์รันไทม์
CommonsWare

@CommonsWare: ตอนนี้ฉันเข้าใจแล้ว :)
theapache64

39

วิธีง่ายๆของฉันคือสิ่งนี้

if (ContextCompat.checkSelfPermission(this, android.Manifest.permission.ACCESS_FINE_LOCATION) ==
        PackageManager.PERMISSION_GRANTED &&
        ContextCompat.checkSelfPermission(this, android.Manifest.permission.ACCESS_COARSE_LOCATION) ==
        PackageManager.PERMISSION_GRANTED) {
    googleMap.setMyLocationEnabled(true);
    googleMap.getUiSettings().setMyLocationButtonEnabled(true);
} else {
    Toast.makeText(this, R.string.error_permission_map, Toast.LENGTH_LONG).show();
}

หรือคุณสามารถเปิดกล่องโต้ตอบการอนุญาตในรูปแบบอื่นเช่นนี้

} else {
   ActivityCompat.requestPermissions(this, new String[] {
      Manifest.permission.ACCESS_FINE_LOCATION, 
      Manifest.permission.ACCESS_COARSE_LOCATION }, 
      TAG_CODE_PERMISSION_LOCATION);
}

laways ย้ายไปที่ส่วนอื่นครับ :(
Ashana.Jackol

2
เพิ่มกล่องโต้ตอบเพื่อเพิ่มสิทธิ์ใน "อื่น ๆ " นี้และคุณก็พร้อมที่จะไป
Vasil Valchev

นี่เป็นความจริงโดยไม่ต้องสงสัยเลยว่าการแก้ไขสำหรับ Android 6 ควรสังเกตว่าคุณต้องใส่คำขออนุญาตในที่อื่น
Keith Adler

ฉันได้รับข้อผิดพลาดนี้กับ Target SDK เป็น 22 และบน Android 5.1 บนอุปกรณ์ S plus (GiONEE_WBL7511) ฉันสับสนว่าเหตุใดจึงเกิดความผิดพลาดนี้ขึ้น เบาะแสใด ๆ ? java.lang.SecurityException: ไคลเอ็นต์ต้องมีสิทธิ์ ACCESS_FINE_LOCATION เพื่อขอตำแหน่ง PRIORITY_HIGH_ACCURACY
arpitgoyal2008

5

สาเหตุ: "เริ่มต้นใน Android 6.0 (API ระดับ 23) ผู้ใช้ให้สิทธิ์แก่แอปในขณะที่แอปกำลังทำงานไม่ใช่เมื่อพวกเขาติดตั้งแอป" ในกรณีนี้ "ACCESS_FINE_LOCATION" คือ "การอนุญาตที่เป็นอันตรายและด้วยเหตุนี้คุณจึงได้รับ" java.lang.SecurityException: "gps" ผู้ให้บริการตำแหน่งต้องได้รับอนุญาต ACCESS_FINE_LOCATION ข้อผิดพลาด ( https://developer.android.com/training/permissions/requesting.html )

วิธีแก้ปัญหา: การติดตั้งโค้ดที่ให้ไว้ที่https://developer.android.com/training/permissions/requesting.htmlในหัวข้อ "ขอสิทธิ์ที่คุณต้องการ" และ "จัดการการตอบกลับคำขอสิทธิ์"

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.