Android中WIFI开发总结
Posted Jason_Lee155
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android中WIFI开发总结相关的知识,希望对你有一定的参考价值。
WIFI就是一种无线联网技术,常见的是使用无线路由器。那么在这个无线路由器的信号覆盖的范围内都可以采用WIFI连接的方式进行联网。如果无线路由器连接了一个ADSL线路或其他的联网线路,则又被称为“热点”。
ADSL(Asymmetric Digital Subscriber Line )线路,是指非对称数字用户线路。利用现有的一对电话铜线,为用户提供上行、下行非对称的传输速率(即带宽)。在现有的双绞线和普通电话线上,根据当地线路状况提供最高3.5Mbps的上行速度和最高24Mbps的下行速度。上行是指用户到电信服务提供商方向,例如上传文件。下行是指从电信服务提供商到用户的方向,例如下载文件。
在android中对Wifi操作,android本身提供了一些有用的包,在android.net.wifi包下面,主要包括以下几个类和接口:
1.ScanResult
主要用来描述已经检测出的接入点,包括接入点的地址,接入点的名称,身份认证,频率,信号强度等信息。其实就是通过wifi 硬件的扫描来获取一些周边的wifi 热点的信息。
2.WifiConfiguration
Wifi网络的配置,包括安全设置等,在我们连通一个wifi 接入点的时候,需要获取到的一些信息。主要包含四个属性:
- BSSID:BSS是一种特殊的Ad-hoc LAN(一种支持点对点访问的无线网络应用模式)的应用,一个无线网络至少由一个连接到有线网络的AP和若干无线工作站组成,这种配置称为一个基本服务装置。一群计算机设定相同的 BSS名称,即可自成一个group,而此BSS名称,即所谓BSSID。通常,手机WLAN中,bssid其实就是无线路由的MAC地址。
- networkid:网络ID。
- PreSharedKey:无线网络的安全认证模式。
- SSID:SSID(Service Set Identif)用于标识无线局域网,SSID不同的无线网络是无法进行互访的。
3.WifiInfo
wifi无线连接的描述,包括(接入点,网络连接状态,隐藏的接入点,IP地址,连接速度,MAC地址,网络ID,信号强度等信息)。这里简单介绍一下WifiManager中常用的方法:
getSSID() | 获得SSID(热点名称) |
getBSSID() | 获取BSSID |
getDetailedStateOf() | 获取客户端的连通性 |
getHiddenSSID() | 获得SSID 是否被隐藏 |
getIpAddress() | 获取IP 地址 |
getLinkSpeed() | 获得连接的速度 |
getMacAddress() | 获得Mac 地址 |
getRssi() | 获得802.11n 网络的信号 |
4.WifiManager
wifi连接统一管理类,获取WIFI网卡的状态(WIFI网卡的状态是由一系列的×××常量来表示的):
WIFI_STATE_DISABLING = 0 | WIFI网卡正在关闭 |
WIFI_STATE_DISABLED = 1 | WIFI网卡不可用 |
WIFI_STATE_ENABLING = 2 | WIFI网正在打开 (WIFI启动需要一段时间) |
WIFI_STATE_ENABLED = 3 | WIFI网卡可用 |
WIFI_STATE_UNKNOWN = 4 | 未知网卡状态 |
WIFI_AP_STATE_DISABLING = 10 | WIFI热点正在关闭 |
WIFI_AP_STATE_DISABLED = 11 | WIFI热点不可用 |
WIFI_AP_STATE_ENABLING = 12 | WIFI热点正在打开 |
WIFI_AP_STATE_ENABLED = 13 | WIFI热点可用 |
5. 在AndroidManifest.xml进行对WIFI操作的权限设置
<!-- 以下是使用wifi访问网络所需的权限 -->
<uses-permission android:name="android.permission.CHANGE_NETWORK_STATE"></uses-permission>
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE"></uses-permission>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"></uses-permission>
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"></uses-permission>
下面是wifi开发的一些步骤:
1.首先是如何打开wifi开关
打开之前先获取WifiManager 对象,通过该对象的isWifiEnabled():boolean 方法来获取当前wifi的开启情况,如果未打开,则执行打开wifi开关操作:
WifiManager mWifiManager = (WifiManager) context.getSystemService(Context.WIFI_SERVICE);
if (!mWifiManager.isWifiEnabled()) {//wifi未打开 执行打开操作
mWifiManager.setWifiEnabled(true);//同样的执行关闭操作的话: mWifiManager.setWifiEnabled(false);
}
2. WIFI打开之后自然是执行扫描操作,搜索周边范围内的热点信息
/**
* 扫描热点,扫描时耗时操作,如果界面中需要展示进度条的话,建议将扫描操作放在子线程中操作
*/
mWifiManager.startScan();
// 得到扫描结果
List<ScanResult> mWifiList = mWifiManager.getScanResults();
// 得到配置好的网络连接,列表中可能出现重复的热点,并且可能是ssid为空的热点,根据需求情况 自行过滤
mWifiConfiguration = mWifiManager.getConfiguredNetworks();
// 查看扫描结果
public StringBuilder lookUpScan() {
StringBuilder stringBuilder = new StringBuilder();
for (int i = 0; i < mWifiList.size(); i++) {
stringBuilder
.append("Index_" + new Integer(i + 1).toString() + ":");
// 将ScanResult信息转换成一个字符串包
// 其中把包括:BSSID、SSID、capabilities、frequency、level
stringBuilder.append((mWifiList.get(i)).toString());
stringBuilder.append("/n");
}
return stringBuilder;
}
3. 获取正在连接中的SSID
mWifiInfo.getSSID()
/**
* 获得当前连接的热点 用上面方法 可能获得的结果为: "0x" 或 "<unknown ssid>"
* 猜测是因为wifiInfo的问题,因此每次去getSSID()的时候需要获得最新的wifiInfo对象
*/
4. 如果需要的话 对热点信号强度进行排序
//将搜索到的wifi根据信号从强到弱进行排序
private List<ScanResult> sortByLevel(List<ScanResult> list) {
ScanResult temp = null;
for (int i = 0; i < list.size(); i++)
for (int j = 0; j < list.size(); j++) {
if (list.get(i).level > list.get(j).level) //level属性即为强度
{
temp = list.get(i);
list.set(i, list.get(j));
list.set(j, temp);
}
}
return list;
}
5.连接到热点
/**
* 众所周知 热点的加密分为三种情况:1没有密码 2用wep加密 3用wpa加密
*/
public static WifiConfiguration CreateWifiInfo(String SSID, String Password, int Type) {
WifiConfiguration config = new WifiConfiguration();
config.allowedAuthAlgorithms.clear();
config.allowedGroupCiphers.clear();
config.allowedKeyManagement.clear();
config.allowedPairwiseCiphers.clear();
config.allowedProtocols.clear();
config.SSID = "\\"" + SSID + "\\"";
WifiConfiguration tempConfig = IsExsits(SSID);
if (tempConfig != null) {
mWifiManager.removeNetwork(tempConfig.networkId);
}
if (Type == 1) //WIFICIPHER_NOPASS
{
/* config.wepKeys[0] = "";//连接无密码热点时加上这两句会出错
config.wepTxKeyIndex = 0;*/
config.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.NONE);
}
if (Type == 2) //WIFICIPHER_WEP
{
config.hiddenSSID = true;
config.wepKeys[0] = "\\"" + Password + "\\"";
config.allowedAuthAlgorithms.set(WifiConfiguration.AuthAlgorithm.SHARED);
config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.CCMP);
config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.TKIP);
config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.WEP40);
config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.WEP104);
config.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.NONE);
config.wepTxKeyIndex = 0;
}
if (Type == 3) //WIFICIPHER_WPA
{
config.preSharedKey = "\\"" + Password + "\\"";
config.hiddenSSID = true;
config.allowedAuthAlgorithms.set(WifiConfiguration.AuthAlgorithm.OPEN);
config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.TKIP);
config.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.WPA_PSK);
config.allowedPairwiseCiphers.set(WifiConfiguration.PairwiseCipher.TKIP);
//config.allowedProtocols.set(WifiConfiguration.Protocol.WPA);
config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.CCMP);
config.allowedPairwiseCiphers.set(WifiConfiguration.PairwiseCipher.CCMP);
config.status = WifiConfiguration.Status.ENABLED;
}
return config;
}
// 添加一个网络并连接
public static boolean addNetwork(WifiConfiguration wcg) {
int wcgID = mWifiManager.addNetwork(wcg);
boolean b = mWifiManager.enableNetwork(wcgID, true);
System.out.println("a--" + wcgID);
System.out.println("b--" + b);
if (b) {
linkingID = wcgID;
}
return b;
}
//连接方式
WifiConfiguration wifiConfiguration = CreateWifiInfo(SSID, Password, Type);
boolean flag = addNetwork(wifiConfiguration);//连接网络
/**
*flag 返回true 并不能代表热点连接成功,但是返回false一定代表连接不成功
*当密码位数不对时也会直接返回false,因此不能用该参数来判别是否连接成功
*这也是我在项目中碰到的一个难题
*/
项目中涉及到wifi切换连接,这个时候就需要监听热点切换情况,这是个难点。不同设备的热点连接速度也不一致,我的做法是启动一个定时任务,当wifi进行切换时先保存oldCurrentSSID,gonaLinkSSID来保存目的连接SSID,然后定时的去读取当前连接中的ssid:mWifiInfo.getSSID(),来实时的监听当前热点情况,如果当前热点与oldCurrentSSID不等,并且等于gonaLinkSSID,即代表热点切换成功。这边需要考虑的是这个定时的长度,因为wifi没有正在连接的状态,所以这边无法准确知道何时连接完毕,所以这里只能给个大概的时间,同时在该时间内如果满足连接成功状态,即可提前取消定时刷新任务。同时需要结合广播(当网络状态变化时系统会发出一条广播)来得到最准确的值。
6. 断开指定连接
// 断开指定ID的网络,这边的id在添加连接时获取 int wcgID = mWifiManager.addNetwork(wcg);
public static void disconnectWifi(int netId) {
if(0 == linkingID ){
return;
}
mWifiManager.disableNetwork(linkingID);
mWifiManager.disconnect();
LogUti.i("info", "断开连接 id" + linkingID);
}
//有些情况下我们可能并不方便拿到那个id,这个时候可以投机取巧的地方是随便连接一个不能连接成功的ssid,当ssid切换的时候 wifi会先自动断开当前连接的热点才去连接目标ssid
最后,成功连接到wifi后,通过本机IP地址便可以进行相关网络开发,在下一篇中我会讲解一个WIFI热点搜索、创建、连接的Demo,希望能帮助到大家!
以上是关于Android中WIFI开发总结的主要内容,如果未能解决你的问题,请参考以下文章
Android开发—智能家居系列:用手机对WIFI模块进行配置
我的Android进阶之旅NDK开发之在C++代码中使用Android Log打印日志,打印出C++的函数耗时以及代码片段耗时详情
Android 逆向Android 进程注入工具开发 ( Visual Studio 开发 Android NDK 应用 | Visual Studio 中 SDK 和 NDK 安装位置 )(代码片段