Android “gps yêu cầu lỗi ACCESS_FINE_LOCATION”, mặc dù tệp kê khai của tôi có chứa lỗi này


104

Mỗi lần tôi chạy ứng dụng, SecurityException của tôi sẽ bị ném và lỗi từ trình gỡ lỗi sẽ đọc như vậy:

java.lang.SecurityException: nhà cung cấp vị trí "gps" yêu cầu quyền ACCESS_COARSE_LOCATION hoặc ACCESS_FINE_LOCATION.

Đây có vẻ như là một sai lầm đơn giản, tuy nhiên, tệp kê khai của tôi hoàn toàn chính xác. Đây rồi, và đây là mã MapActivity của tôi nữa:

<?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>

Hoạt động của tôi:

    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) {

    }
}

Bạn đang thử nghiệm điều này trên phiên bản Android nào?
CommonsWare

4
Không liên quan, nhưng nếu bạn yêu cầu vị trí tốt, bạn không cần phải yêu cầu thô. Nó được bao gồm.
joey_g216

Câu trả lời:


136

ACCESS_COARSE_LOCATION, ACCESS_FINE_LOCATIONWRITE_EXTERNAL_STORAGE đều là một phần của Android hệ thống cho phép 6.0 runtime . Ngoài việc có chúng trong tệp kê khai như bạn làm, bạn cũng phải yêu cầu chúng từ người dùng trong thời gian chạy (sử dụng requestPermissions()) và xem liệu bạn có chúng (đang sử dụng checkSelfPermission()) hay không.

Một giải pháp trong ngắn hạn là giảm xuống targetSdkVersiondưới 23 của bạn .

Tuy nhiên, cuối cùng, bạn sẽ muốn cập nhật ứng dụng của mình để sử dụng hệ thống quyền thời gian chạy.

Ví dụ: hoạt động này hoạt động với năm quyền. Bốn là quyền thời gian chạy, mặc dù nó hiện chỉ xử lý ba (tôi đã viết nó trước đây WRITE_EXTERNAL_STORAGEđã được thêm vào danh sách quyền thời gian chạy).

/***
 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();
  }
}

(từ dự án mẫu này )

Đối với hàm requestPermissions (), các tham số có phải là "ACCESS_COARSE_LOCATION" không? Hay tôi nên bao gồm tên đầy đủ "android.permission.ACCESS_COARSE_LOCATION"?

Tôi sẽ sử dụng các hằng số được định nghĩa trên Manifest.permission , như được hiển thị ở trên.

Ngoài ra, mã yêu cầu là gì?

Điều đó sẽ được chuyển lại cho bạn dưới dạng tham số đầu tiên onRequestPermissionsResult(), vì vậy bạn có thể nói một requestPermissions()cuộc gọi từ một cuộc gọi khác.


1
Đối với hàm requestPermissions (), các tham số có phải là "ACCESS_COARSE_LOCATION" không? Hay tôi nên bao gồm tên đầy đủ "android.permission.ACCESS_COARSE_LOCATION"?
Jason Cromer

1
Cảm ơn bạn, điều này đã loại bỏ lỗi. Tôi vẫn gặp sự cố khi truy cập vị trí của mình, vì locationManager của tôi liên tục trả về vị trí của tôi là null, nhưng điều đó không liên quan đến lỗi này. Cảm ơn bạn cho giải pháp của bạn!
Jason Cromer

@CommonsWare: Ý bạn là gì khi nói 'cuối cùng'? Xin lỗi, tôi không hiểu phần đó.
theapache64

1
@ theapache64: Một ngày nào đó, điều gì đó sẽ khiến bạn muốn đặt thành targetSdkVersion23 hoặc cao hơn. Tại thời điểm đó, bạn sẽ cần phải sử dụng hệ thống cho phép thời gian chạy. Cho đến khi khoảng thời gian đó đến, bạn có thể giữ nguyên targetSdkVersiondưới 23 của mình và bỏ qua các quyền thời gian chạy.
CommonsWare

@CommonsWare: Bây giờ tôi đã hiểu. :)
theapache64 Ngày

39

Giải pháp đơn giản của tôi là

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();
}

hoặc bạn có thể mở hộp thoại quyền ở nơi khác như thế này

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

laways di chuyển theo để khác phần bro :(
Ashana.Jackol

2
thêm hộp thoại để thêm quyền trong "khác" này, và bạn đã sẵn sàng.
Vasil Valchev

Trên thực tế, không nghi ngờ gì nữa, đây là bản sửa lỗi cho Android 6. Cần lưu ý rằng bạn phải đặt các yêu cầu quyền vào cái khác.
Keith Adler

Tôi gặp lỗi này với Target SDK là 22 và trên Android 5.1 trên thiết bị S plus (GiONEE_WBL7511). Tôi bối rối về lý do tại sao vụ tai nạn này xảy ra. Bất kì manh mối nào? java.lang.SecurityException: Máy khách phải có quyền ACCESS_FINE_LOCATION để yêu cầu vị trí PRIORITY_HIGH_ACCURACY.
arpitgoyal2008

5

NGUYÊN NHÂN: "Bắt đầu từ Android 6.0 (API cấp 23), người dùng cấp quyền cho ứng dụng khi ứng dụng đang chạy, không phải khi họ cài đặt ứng dụng." Trong trường hợp này, "ACCESS_FINE_LOCATION" là "quyền nguy hiểm và vì lý do đó, bạn nhận được 'java.lang.SecurityException: nhà cung cấp vị trí" gps "này yêu cầu quyền ACCESS_FINE_LOCATION.' lỗi ( https://developer.android.com/training/permissions/requesting.html ).

GIẢI PHÁP: Triển khai mã được cung cấp tại https://developer.android.com/training/permissions/requesting.html trong tiêu đề "Yêu cầu quyền bạn cần" và "Xử lý phản hồi yêu cầu quyền".

Khi sử dụng trang web của chúng tôi, bạn xác nhận rằng bạn đã đọc và hiểu Chính sách cookieChính sách bảo mật của chúng tôi.
Licensed under cc by-sa 3.0 with attribution required.