Android开发——获取WiFi扫描结果

1.三个与WiFi相关的类:WifiManagerWifiInfoScanResult(点击可查看官方文档,需要FQ)

// 获取系统WiFi服务
WifiManager wm = (WifiManager) getApplicationContext().getSystemService(WIFI_SERVICE);
wm.isWifiEnabled(); // 返回WiFi的开关状态
wm.setWifiEnabled(true);  // 打开WiFi
wm.setWifiEnabled(false); // 关闭WiFi
wm.calculateSignalLevel (int rssi,int numLevels); // 将RSSI分为numLevels个等级,返回rssi处在哪一个等级 
wm.getWifiState(); // 获取当前手机WiFi网卡状态,返回值下面五个int型常量之一
WifiManager.WIFI_STATE_ENABLED
WifiManager.WIFI_STATE_DISABLED
WifiManager.WIFI_STATE_ENABLING
WifiManager.WIFI_STATE_DISABLING
WifiManager.WIFI_STATE_UNKNOWN

// 获取当前所连接WiFi的信息
WifiInfo wi = wm.getConnectionInfo();
wi.getSSID();        // 获取当前连接WiFi的名字
wi.getBSSID();       // 获取当前所连WiFi设备的Mac地址,String类型
wi.getMacAddress();  // 获取本机Mac地址
wi.getRssi();        // 获取当前连接WiFi的信号强度,返回一个0~-100之间的int型数据
wi.getLinkSpeed();   // 获取连接速度
WifiInfo.LINK_SPEED_UNITS; // 连接速度单位的字符串常量"Mbps"

// 获取扫描到的所有WiFi信息 
List<ScanResult> scanResults = wm.getScanResults(); 
scanResult.SSID();    // 返回WiFi的SSID
scanResult.BSSID();   // 返回WiFi的BSSID
scanResult.level();   // 返回WiFi的信号强度(原始数据)

2.WiFi扫描所需权限

<!-- 获取WiFi状态的权限 -->
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
<!-- 改变WiFi状态的权限 -->
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE"/>

注1:API文档中提到使用List<ScanResult> getScanResults()时,最好额外添加以下两个权限之一(保护等级:危险),详见链接

<!-- 获取粗略位置的权限-->
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
<!-- 获取精确位置的权限 -->
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>

注2:Android 6.0及以上的版本,获取涉及隐私的权限更加严格。想要获取WiFi扫描列表,需申请位置权限,还要提醒用户打开位置权限。谷歌这么做大概是考虑到WiFi位置指纹可以用于定位,所以通过开启位置权限来提醒用户吧。

3.显示当前所连WiFi信息以及扫描到的所有WiFi的信息

  • AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.lee.wifiscan">
    <!-- 获取WiFi状态的权限 -->
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
    <!-- 改变WiFi状态的权限 -->
    <uses-permission android:name="android.permission.CHANGE_WIFI_STATE"/>
    <!-- 获取精确位置的权限 -->
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
    <!-- 获取粗略位置的权限 -->
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
    <application
        android:allowBackup="false"
        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" />
                <action android:name="android.intent.action.VIEW"/>
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
</manifest>
  • activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="5dp"
        android:text="@string/current_connected_wifi"
        android:textSize="20sp"/>
    <TextView
        android:id="@+id/connected_wifi_info_tv"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="10dp"
        android:text=""/>
    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="5dp"
        android:text="@string/wifi_scan_results"
        android:textSize="20sp"/>
    <ScrollView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="10dp"
        android:fadingEdge="vertical"
        android:scrollbars="vertical">
        <TextView
            android:id="@+id/scan_results_info_tv"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text=""/>
    </ScrollView>

</LinearLayout>
  • MainActivity.java
package com.lee.wifiscan;

import android.Manifest;
import android.app.Activity;
import android.content.pm.PackageManager;
import android.net.wifi.ScanResult;
import android.net.wifi.WifiInfo;
import android.net.wifi.WifiManager;
import android.os.Build;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.widget.TextView;

import java.util.List;
import java.util.Timer;
import java.util.TimerTask;


public class MainActivity extends Activity {
    private static final int PERMISSIONS_REQUEST_CODE_ACCESS_FINE_LOCATION = 1000;
    private final int UPDATE_UI_REQUEST_CODE = 1024;
    private TextView mCurrConnTV;   // 显示当前所连WiFi信息的控件
    private TextView mScanResultTV;    // 显示WiFi扫描结果的控件
    private StringBuffer mCurrConnStr;  // 暂存当前所连WiFi信息的字符串
    private StringBuffer mScanResultStr;    // 暂存WiFi扫描结果的字符串
    private WifiManager mWifiManager;   // 调用WiFi各种API的对象
    private Timer mTimer;   // 启动定时任务的对象
    private final int SAMPLE_RATE = 2000; // 采样周期,以毫秒为单位
    private Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            if (msg.what == UPDATE_UI_REQUEST_CODE) {
                updateUI();
            }
        }
    };

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

        mCurrConnTV = findViewById(R.id.connected_wifi_info_tv);
        mScanResultTV = findViewById(R.id.scan_results_info_tv);

        mWifiManager = (WifiManager) getApplicationContext().getSystemService(WIFI_SERVICE);
        getLocationAccessPermission();  // 先获取位置权限

        mTimer = new Timer();
        mTimer.schedule(new TimerTask() {
            @Override
            public void run() {
                scanWifi();
                mHandler.sendEmptyMessage(UPDATE_UI_REQUEST_CODE);
            }
        }, 0, SAMPLE_RATE); // 立即执行任务,每隔2000ms执行一次WiFi扫描的任务
        // 扫描周期不能太快,WiFi扫描所有的AP需要一定时间
    }

    @Override
    protected void onStop() {
        super.onStop();
        mTimer.cancel();    // 取消定时任务
    }

    /**
     * 增加开启位置权限功能,以适应Android 6.0及以上的版本
     */
    private void getLocationAccessPermission() {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M &&
                checkSelfPermission(Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
            requestPermissions(new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, PERMISSIONS_REQUEST_CODE_ACCESS_FINE_LOCATION);
        }
    }

    public void scanWifi() {
        // 如果WiFi未打开,先打开WiFi
        if (!mWifiManager.isWifiEnabled())
            mWifiManager.setWifiEnabled(true);

        // 开始扫描WiFi
        mWifiManager.startScan();
        // 获取并保存当前所连WiFi信息
        mCurrConnStr = new StringBuffer();
        WifiInfo wifiInfo = mWifiManager.getConnectionInfo();
        mCurrConnStr.append("SSID: ").append(wifiInfo.getSSID()).append("\n");
        mCurrConnStr.append("MAC Address: ").append(wifiInfo.getBSSID()).append("\n");
        mCurrConnStr.append("Signal Strength(dBm): ").append(wifiInfo.getRssi()).append("\n");
        mCurrConnStr.append("speed: ").append(wifiInfo.getLinkSpeed()).append(" ").append(WifiInfo.LINK_SPEED_UNITS);

        // 获取并保存WiFi扫描结果
        mScanResultStr = new StringBuffer();
        List<ScanResult> scanResults = mWifiManager.getScanResults();
        for (ScanResult sr : scanResults) {
            mScanResultStr.append("SSID: ").append(sr.SSID).append("\n");
            mScanResultStr.append("MAC Address: ").append(sr.BSSID).append("\n");
            mScanResultStr.append("Signal Strength(dBm): ").append(sr.level).append("\n\n");
        }
    }

    private void updateUI() {
        mCurrConnTV.setText(mCurrConnStr);
        mScanResultTV.setText(mScanResultStr);
    }
}
  • 运行结果:当移动手机位置时,当前所连WiFi延迟大约3秒才刷新;当手机保持静止时,当前所连WiFi延迟大约20秒才刷新;而不管手机状态如何,WiFi扫描结果列表均是15~20s才刷新一次。
  • 结果分析:WiFi刷新频率同设定的2秒有非常大的差距,初步推断是WiFi扫描机制造成的,详见链接链接链接链接

4.改进:getScanResults()之前,主动调用startScan(),取消注释MainActivity.java中第84行代码即可

// 主动扫描
mWifiManager.startScan();
// 获取WiFi扫描结果
mScanResults = mWifiManager.getScanResults();
  • 带来的新问题:调用startScan()后,WiFi在后台扫描,当前线程继续执行,因此获得的结果不是最新的,而是上次扫描的结果

5. 使用广播

  •  MainActivity.java
package com.lee.wifiscan;

import android.app.Activity;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.net.wifi.ScanResult;
import android.net.wifi.WifiManager;
import android.os.Bundle;
import android.widget.TextView;

import java.util.List;

/**
 * 利用广播实现WiFi扫描
 */
public class MainActivity2 extends Activity {
    private TextView mScanResultTV;    // 显示WiFi扫描结果的控件
    private StringBuffer mScanResultStr;    // 暂存WiFi扫描结果的字符串
    WifiManager wifiManager;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        mScanResultTV = findViewById(R.id.scan_results_info_tv);
        wifiManager = (WifiManager) getApplicationContext().getSystemService(WIFI_SERVICE);

        registerReceiver(new BroadcastReceiver() {
            @Override
            public void onReceive(Context c, Intent intent) {
                // 获取并保存WiFi扫描结果
                mScanResultStr = new StringBuffer();
                List<ScanResult> scanResults = wifiManager.getScanResults();
                for (ScanResult sr:scanResults){
                    mScanResultStr.append("SSID: ").append(sr.SSID).append("\n");
                    mScanResultStr.append("MAC Address: ").append(sr.BSSID).append("\n");
                    mScanResultStr.append("Signal Strength(dBm): ").append(sr.level).append("\n\n");
                }
                runOnUiThread(new Runnable() {
                    @Override
                    public void run() {
                        mScanResultTV.setText(mScanResultStr);
                    }
                });
            }
        }, new IntentFilter(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION));

        wifiManager.startScan();
    }
}

 运行结果:WiFi AP的RSS变化时间很不固定,变化时间范围:2s~30s。

尝试用以下代码定时启动WiFi扫描,WiFi AP的RSS变化时间还是不固定

new Timer().schedule(new TimerTask() {
    @Override
    public void run() {
        wifiManager.startScan();
    }
},0,10000);

// TODO

6.Android 6.0以上getScanResults()结果为空,需要开启位置权限,在进行WiFi扫描之前调用以下函数,弹出对话框让用户开启位置权限(参考文章

private void getLocationAccessPermission() {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M &&
            checkSelfPermission(Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
        requestPermissions(new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, PERMISSIONS_REQUEST_CODE_ACCESS_FINE_LOCATION);
    }
}

项目仓库:https://github.com/zlx01/WifiScan

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值