【问题标题】:Android WifiManager getScanResult complains Need ACCESS_COARSE_LOCATION or ACCESS_FINE_LOCATION permission although declared permissionAndroid WifiManager getScanResult 抱怨需要 ACCESS_COARSE_LOCATION 或 ACCESS_FINE_LOCATION 权限,尽管声明了权限
【发布时间】:2024-01-09 17:55:01
【问题描述】:

我正在开发一个应用来检查 Wifi 盒子。

我在 wifiManager.getScanResults() 处收到错误“java.lang.SecurityException:需要 ACCESS_COARSE_LOCATION 或 ACCESS_FINE_LOCATION 权限才能获取扫描结果”,即使我已经声明了这些权限。

主要活动

public class MainActivity extends AppCompatActivity {

WifiManager wifiManager;
String[] wifis;
WifiReceiver wifiReceiver;

ListView wifiListView;

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

    wifiListView = (ListView) findViewById(R.id.wifi_list);

    wifiManager = (WifiManager) getSystemService(Context.WIFI_SERVICE);
    wifiReceiver = new WifiReceiver();
    wifiManager.startScan();
}

protected void onPause() {
    unregisterReceiver(wifiReceiver);
    super.onPause();
}

protected void onResume() {
    registerReceiver(wifiReceiver, new IntentFilter(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION));
    super.onResume();
}

private class WifiReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        List<ScanResult> wifiScanList = wifiManager.getScanResults();
        wifis = new String[wifiScanList.size()];

        for (int i = 0; i < wifiScanList.size(); i++) {
            wifis[i] = wifiScanList.get(i).toString();
        }

        wifiListView.setAdapter(new ArrayAdapter<>(getApplicationContext(), android.R.layout.simple_list_item_1, wifis));
    }
}

清单

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.sample">

<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>

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

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

我使用的是 SDK 6.0

我观察到类似的question,但由于我已经声明了许可,因此解决方案不适用。

有人知道可能是什么问题吗?谢谢。

【问题讨论】:

  • 尝试重启你的IDE。
  • 试过重启IDE,卸载应用,清理项目,构建项目,还是一样的错误
  • 嗨@kevin0228ca,我不完全确定我所说的是否适用于您的问题。根据VogellaAndroid 6.0 Marshmallow (API 23) 引入了新的运行时权限模型。如果您的应用面向 Android 6.0,则必须使用新的权限模型。所以我假设这个新模型基于 Requesting Permissions at Run Time 可能会解决您遇到的问题。 Vogella 为我们提供了official documentation 的链接

标签: java android android-wifi wifimanager


【解决方案1】:

在Android M中,每次使用前都需要向用户询问PermissionModel中定义为危险的权限,如下:

private boolean mayRequestLocation() {
        if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
            return true;
        }
        if (checkSelfPermission(ACCESS_FINE_LOCATION) == PackageManager.PERMISSION_GRANTED) {
            return true;
        }
        if (shouldShowRequestPermissionRationale(ACCESS_FINE_LOCATION)) {
            Snackbar.make(mView, R.string.permission_rationale, Snackbar.LENGTH_INDEFINITE)
                    .setAction(android.R.string.ok, new View.OnClickListener() {
                        @Override
                        @TargetApi(Build.VERSION_CODES.M)
                        public void onClick(View v) {
                            requestPermissions(new String[]{ACCESS_FINE_LOCATION}, REQUEST_FINE_LOCATION);
                        }
                    });
        } else {
            requestPermissions(new String[]{ACCESS_FINE_LOCATION}, REQUEST_FINE_LOCATION);
        }
        return false;
    }

将此添加到您的活动中:

private static final int REQUEST_FINE_LOCATION=0

并在运行时加载它:

loadPermissions(Manifest.permission.ACCESS_FINE_LOCATION,REQUEST_FINE_LOCATION);

要评估您的权限请求的结果,您可以重写 onRequestPermissionsResult 方法:

@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
    switch (requestCode) {
        case REQUEST_FINE_LOCATION: {
            // If request is cancelled, the result arrays are empty.
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                // The requested permission is granted.
            }
            else{
                // The user disallowed the requested permission.
            }
            return;
        }

}

【讨论】:

    【解决方案2】:

    MADAO is right:你应该打开GPS来获取WIFI接入点列表。

    但我不确定 PEERS_MAC_ADDRESS。如果您查看the source code(第 957 行):

    /**
     * Return the results of the most recent access point scan, in the form of
     * a list of {@link ScanResult} objects.
     * @return the list of results
     */
    public List<ScanResult> getScanResults(String callingPackage) {
        enforceAccessPermission();
        int userId = UserHandle.getCallingUserId();
        int uid = Binder.getCallingUid();
        boolean canReadPeerMacAddresses = checkPeersMacAddress();
        boolean isActiveNetworkScorer =
                NetworkScorerAppManager.isCallerActiveScorer(mContext, uid);
        boolean hasInteractUsersFull = checkInteractAcrossUsersFull();
        long ident = Binder.clearCallingIdentity();
        try {
            if (!canReadPeerMacAddresses && !isActiveNetworkScorer
                    && !isLocationEnabled()) {
                return new ArrayList<ScanResult>();
            }
            if (!canReadPeerMacAddresses && !isActiveNetworkScorer
                    && !checkCallerCanAccessScanResults(callingPackage, uid)) {
                return new ArrayList<ScanResult>();
            }
            if (mAppOps.noteOp(AppOpsManager.OP_WIFI_SCAN, uid, callingPackage)
                    != AppOpsManager.MODE_ALLOWED) {
                return new ArrayList<ScanResult>();
            }
            if (!isCurrentProfile(userId) && !hasInteractUsersFull) {
                return new ArrayList<ScanResult>();
            }
            return mWifiStateMachine.syncGetScanResultsList();
        } finally {
            Binder.restoreCallingIdentity(ident);
        }
    }
    

    第一个if 正在检查canReadPeerMacAddressescheckPeersMacAddress() 的代码是:

    /**
     * Returns true if the caller holds PEERS_MAC_ADDRESS.
     */
    private boolean checkPeersMacAddress() {
        return mContext.checkCallingOrSelfPermission(
                android.Manifest.permission.PEERS_MAC_ADDRESS) == PackageManager.PERMISSION_GRANTED;
    }
    

    如果您添加权限,您可以绕过if (!canReadPeerMacAddresses &amp;&amp; !isActiveNetworkScorer &amp;&amp; !isLocationEnabled()) {。我已经测试过,但我无法通过仅使用权限和禁用位置来获取 WIFI MAC 列表。

    【讨论】:

    • 你是说你已经验证你去这个权限,当你打电话给mContext.checkCallingOrSelfPermission( android.Manifest.permission.PEERS_MAC_ADDRESS) == PackageManager.PERMISSION_GRANTED时你得到了真实的?如果是这样,可能是调用enforceAccessPermission() 阻止了访问?
    • 没有。正如我所说,我已经添加了 PEERS_MAC_ADDRESS 权限,但在没有打开位置的情况下未能成功获取接入点列表。此外,我将我的应用程序设为系统应用程序,但在不打开位置的情况下仍然无法获取接入点列表。
    • enforceAccessPermission() 的代码如下:private void enforceAccessPermission() { mContext.enforceCallingOrSelfPermission(android.Manifest.permission.ACCESS_WIFI_STATE, "WifiService"); } 我在清单中有ACCESS_WIFI_STATE 权限。
    • 我只是想知道您是否可以在代码中验证这两个权限实际上已被授予。您可能已经请求了它们,但这并不意味着没有出现任何问题导致操作系统拒绝授予它们。
    • 其实没有。我没有调试Android代码本身。但只有我检查过它的来源。你是对的。也许操作系统或供应商特定的操作系统可能会拒绝该权限。现在我很好奇如何在不打开位置的情况下获得接入点。我可以对手机做任何事情:生根,制作系统应用程序等。如果您有任何解决方案,请分享。 Tnx
    【解决方案3】:

    ACCESS_FINE_LOCATIONACCESS_COARSE_LOCATION 是必需的。要获得有效结果,您还必须打开 GPS 或获得PEERS_MAC_ADDRESS 权限,例如Setting

    【讨论】: