1.三个与WiFi相关的类:WifiManager,WifiInfo,ScanResult(点击可查看官方文档,需要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);
}
}