Android WifiManager::getScanResults() 仍然返回空列表

Posted

技术标签:

【中文标题】Android WifiManager::getScanResults() 仍然返回空列表【英文标题】:Android WifiManager::getScanResults() still returns empty list 【发布时间】:2018-10-07 22:13:41 【问题描述】:

作为 android 编程的新手,我正在尝试使用 WifiManager 的 getScanResults() 方法获取 SSID 列表,但它仍然为空,即使我已授予它 ACCESS_COARSE_LOCATION 权限以及 CHANGE_WIFI_STATE(对于startScan() 方法),无论是在清单中还是在运行时检查/请求它。

SCAN_RESULTS_AVAILABLE_ACTION 的广播接收器中,我什至使用键EXTRA_RESULTS_UPDATED 检查意图的额外字段,结果返回true。

然而,list.size() 总是返回 0。

我在这里缺少什么?我正在 Android 7.0 API 24 设备上进行测试。

编辑:代码如下:

MainActivity.java:

public class MainActivity extends Activity implements View.OnClickListener 
    private static final String TAG="WiFiDemo";

    WifiManager wifiManager;
    WifiBroadcastReceiver wifiReceiver;

    TextView textView;
    Button btn;

    @Override
    protected void onCreate(Bundle savedInstanceState)
    
        Log.d(TAG, "onCreate()");
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        //Check for permissions
        if ((ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_COARSE_LOCATION)
                != PackageManager.PERMISSION_GRANTED)
            || (ContextCompat.checkSelfPermission(this, Manifest.permission.CHANGE_WIFI_STATE)
                    != PackageManager.PERMISSION_GRANTED))
        
            Log.d(TAG, "Requesting permissions");

            //Request permission
            ActivityCompat.requestPermissions(this,
                    new String[]Manifest.permission.ACCESS_COARSE_LOCATION,
                                 Manifest.permission.ACCESS_FINE_LOCATION,
                                 Manifest.permission.ACCESS_WIFI_STATE,
                                 Manifest.permission.CHANGE_WIFI_STATE,
                                 Manifest.permission.ACCESS_NETWORK_STATE,
                    123);
        
        else
            Log.d(TAG, "Permissions already granted");

        wifiManager = (WifiManager) getSystemService(Context.WIFI_SERVICE);

        textView = findViewById(R.id.text);
        textView.setTextAlignment(View.TEXT_ALIGNMENT_TEXT_START);

        btn = findViewById(R.id.btn);
        btn.setOnClickListener(this);

        //Instantiate broadcast receiver
        wifiReceiver = new WifiBroadcastReceiver();

        //Register the receiver
        registerReceiver(wifiReceiver, new IntentFilter(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION));
    

    @Override
    public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults)
    
        Log.d(TAG, "onRequestPermissionsResult");

        switch (requestCode)
        
            case 123:
            
                // If request is cancelled, the result arrays are empty.
                if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED)
                
                    // permission was granted
                    Log.d(TAG, "permission granted: " + permissions[0]);
                
                else
                
                    // permission denied, boo! Disable the
                    // functionality that depends on this permission.
                    Log.d(TAG, "permission denied: " + permissions[0]);
                

                return;
            

            // other 'case' lines to check for other
            // permissions this app might request.
        
    

    //Define class to listen to broadcasts
    class WifiBroadcastReceiver extends BroadcastReceiver
    
        @Override
        public void onReceive(Context context, Intent intent)
        
            Log.d(TAG, "onReceive()");
            Toast.makeText(getApplicationContext(), "Scan complete!", Toast.LENGTH_SHORT).show();

            boolean ok = intent.getBooleanExtra(WifiManager.EXTRA_RESULTS_UPDATED, false);

            if (ok)
            
                Log.d(TAG, "scan OK");

                //StringBuffer buffer = new StringBuffer();
                List<ScanResult> list = wifiManager.getScanResults();

                Toast.makeText(getApplicationContext(), Integer.toString(list.size()), Toast.LENGTH_SHORT).show();

                for (ScanResult scanResult : list)
                
                    //buffer.append(scanResult);
                    textView.append(scanResult.toString());
                
            
            else
                Log.d(TAG, "scan not OK");
        
    

    @Override
    protected void onStop()
    
        // TODO Auto-generated method stub
        unregisterReceiver(wifiReceiver);
        super.onStop();
    

    @Override
    public void onClick(View view)
    
        // TODO Auto-generated method stub
        //Toast.makeText(getApplicationContext(), "All Network seached !!",0).show();
        if(view.getId()==R.id.btn)
        
            Log.d(TAG, "onCreate() wifi.startScan()");

            //if (!wifiManager.isWifiEnabled())
            //    wifiManager.setWifiEnabled(true);

            wifiManager.startScan();
        
    

在清单中:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.wifi2ir.umagi.wifitest2">

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        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>

    <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_NETWORK_STATE" />
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
</manifest>

【问题讨论】:

你的代码在哪里? 我根据要求编辑了帖子以添加代码。 您是否启用了定位服务?从 Android 6 开始,仅授予权限不足以获得结果。 天哪! @sagar 的答案是正确的。虽然我以前见过,但直到现在(终于!)我才意识到这项服务与位置权限完全不同,我需要在 Android 设置 | 中明确启用它。连接 |位置 [位置服务设置][1] 我一直在查看我的应用程序的权限设置:[位置权限设置][2] 感谢您消除困惑! [1]:i.stack.imgur.com/yFAm7.png [2]:i.stack.imgur.com/XsOWF.png 太棒了!我会将其更新为答案,帮助我批准,以便其他人也可以受益。 【参考方案1】:

从 Android 6 开始,您必须启用位置服务才能获得所需的结果。授予权限只是完成了一半的工作。

您还可以触发将用户重定向到此设置的意图:

Intent myIntent = new Intent( Settings.ACTION_LOCATION_SOURCE_SETTINGS);
startActivity(myIntent);

定位服务是必要的,因为当您访问扫描结果时,您可以访问诸如 BSSID(接入点地址)之类的信息。此信息还可用于检测设备的位置。通过要求位置服务,操作系统可确保用户在使用您的应用时了解他们的位置信息正在被访问。

【讨论】:

我使用的是 Android 8.0.1。我还在我的代码中添加了位置权限处理。但是完成扫描后我得到了空列表。在 Android > 8.0 中我们需要额外做的任何事情 有没有办法确定位置服务被禁用? 更新:在***.com/questions/10311834/… 找到了解决方案。也许值得将此添加到答案中。

以上是关于Android WifiManager::getScanResults() 仍然返回空列表的主要内容,如果未能解决你的问题,请参考以下文章

Android 逆向Android 权限 ( Android 逆向中使用的 android.permission 权限 | Android 系统中的 Linux 用户权限 )

android 21 是啥版本

Android逆向-Android基础逆向(2-2)

【Android笔记】android Toast

图解Android - Android核心机制

Android游戏开发大全的目录