1 概述
需要实现一个需求,Android手机A发射热点,获取BSSID,发送到另一台Android设备,在环境中存在多个同样SSID和密码的情况下,设备只会连接手机A。
1.1 概念
服务集识别码(英语:Service Set Identifier,SSID)是一个或一组基础架构模式无线网络的标识,依照标识方式又可细分为两种:
- 基本服务集识别码(BSSID),表示的是AP的数据链路层的MAC地址
- 扩展服务集识别码(ESSID),一个最长32字节区分大小写的字符串,表示无线网络的名称
1.2 相关文件
- frameworks/base/core/java/android/widget/WifiAdmin.java
- frameworks/base/wifi/java/android/net/wifi/WifiManager.java
- frameworks/base/wifi/java/android/net/wifi/WifiConfiguration.java
- frameworks/base/core/java/com/android/internal/util/AsyncChannel.java
- frameworks/base/services/java/com/android/server/wifi/WifiService.java
- frameworks/base/wifi/java/android/net/wifi/WifiStateMachine.java
- frameworks/base/wifi/java/android/net/wifi/NetworkUpdateResult.java
- frameworks/base/wifi/java/android/net/wifi/WifiConfigStore.java
2 解决方法
APP获取WLAN MAC。注意从6.0开始不能用旧的方法获取MAC。用新的方法后,发现华为、vivo等机型有效,但是小米6和小米8都不行。分析一下,小米的手机支持WiFi和热点双开,所以它的网卡应该有两个。调试可以看到有两个设备,一般手机用的wlan0在这里只能用于手机连无线路由器,发射热点用的是softap0。
-
4C:49:E3:XX:XX:XX
小米6(WLAN0 不能连接) -
4E:49:E3:XX:XX:XX
小米6(softap0 其他手机连这个)
代码如下:
public static String getMacAddress(Context context) {
String macAddress = "";
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
try {
WifiManager wifiManager = (WifiManager) context.getApplicationContext().getSystemService(Context.WIFI_SERVICE);
WifiInfo wifiInfo = null;
if (wifiManager != null) {
wifiInfo = wifiManager.getConnectionInfo();
}
macAddress = wifiInfo.getMacAddress();
} catch (Exception e) {
e.printStackTrace();
}
} else {
try {
Enumeration<NetworkInterface> interfaces = NetworkInterface.getNetworkInterfaces();
while (interfaces.hasMoreElements()) {
NetworkInterface iF = interfaces.nextElement();
byte[] addr = iF.getHardwareAddress();
if (addr == null || addr.length == 0) {
continue;
}
StringBuilder buf = new StringBuilder();
for (byte b : addr) {
buf.append(String.format("%02X:", b));
}
if (buf.length() > 0) {
buf.deleteCharAt(buf.length() - 1);
}
String mac = buf.toString();
Log.d(TAG, "interfaceName=" + iF.getName() + ", mac=" + mac);
if (TextUtils.equals(iF.getName(), "wlan0")) {
macAddress = mac;
}
// 适配小米手机
if (TextUtils.equals(iF.getName(), "softap0")) {
return mac;
}
}
} catch (SocketException e) {
e.printStackTrace();
return macAddress;
}
}
return macAddress;
}
修改Android设备的源代码WifiAdmin.java,mBssid保存手机A的WLAN MAC。
private WifiConfiguration createWifiInfo(String SSID, String Password,
WifiCipherType Type) {
WifiConfiguration config = new WifiConfiguration();
config.allowedAuthAlgorithms.clear();
config.allowedGroupCiphers.clear();
config.allowedKeyManagement.clear();
config.allowedPairwiseCiphers.clear();
config.allowedProtocols.clear();
config.priority = 40;
config.SSID = "\"" + SSID + "\"";
if (mBssid != null && !mBssid.equals("")) {
config.BSSID = mBssid;
mBssid = "";
}
...
3 结论
在网上搜索了很多文章,都说解决不了这个问题,差点就放弃了。最终,在老大的鼓励(bian chi)下尝试自己读文档改源码,发现改config的BSSID是有效的。
实践是检验真理的唯一标准。
参考
wi-fi-Android如何连接指定BSSID的AP(access point, wifi接入点)?——CSDN问答频道
https://ask.csdn.net/questions/200703
关于Android Wifi NetworkId的一些理解。 - CSDN博客
https://blog.csdn.net/mengmengkenanjun/article/details/54287774
Android -- Wifi连接流程分析 - CSDN博客
https://blog.csdn.net/csdn_of_coder/article/details/51922801
服务集 - 维基百科,自由的百科全书
https://zh.wikipedia.org/wiki/%E6%9C%8D%E5%8A%A1%E9%9B%86_(%E6%97%A0%E7%BA%BF%E5%B1%80%E5%9F%9F%E7%BD%91)
SSID、BSSID、ESSID区别? - 知乎
https://www.zhihu.com/question/24362037
【WLAN从入门到精通-基础篇】第6期——WLAN常用概念-无线局域网-华为企业互动社区
https://forum.huawei.com/enterprise/zh/thread-318127-1-1.html
Android获取本机WiFi MAC Address之坑
http://ysuiboli.github.io/android/wifi/mac/2017/05/05/Android%E8%8E%B7%E5%8F%96%E6%9C%AC%E6%9C%BAWiFi-MAC-Address%E4%B9%8B%E5%9D%91.html
WifiConfiguration | Android Developers
https://developer.android.com/reference/android/net/wifi/WifiConfiguration