1. Android开启热点的API
Android 8.0及以上,Android官方提供了开启热点的API
wifiManager.startLocalOnlyHotspot(new WifiManager.LocalOnlyHotspotCallback(){
@Override
public void onStarted(WifiManager.LocalOnlyHotspotReservation reservation) {
super.onStarted(reservation);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
Log.i(TAG, "Wifi Hotspot is on now, " + reservation.getWifiConfiguration().SSID);
Log.i(TAG, "Wifi Hotspot is on now, " + reservation.getWifiConfiguration().preSharedKey);
}
}
@Override
public void onStopped() {
super.onStopped();
Log.i(TAG, "onStopped: ");
}
@Override
public void onFailed(int reason) {
super.onFailed(reason);
Log.i(TAG, "onFailed: ");
}
}, new Handler());
05-21 20:37:18.517 18966-18966/com.gnet.rxjavademo I/hotspot: Wifi Hotspot is on now, AndroidShare_6819
05-21 20:37:18.517 18966-18966/com.gnet.rxjavademo I/hotspot: Wifi Hotspot is on now, a329a61ae014
该API生成的热点名称和密码都是随机的字符串,也就是一次性的,回调中可以获取到SSID和preSharedKey(就是密码)。而且要注意以下两点:
- 1、需要网络权限和定位权限(这点有点想不通)
int granted = ContextCompat.checkSelfPermission(this, Manifest.permission_group.LOCATION);
if (granted == PackageManager.PERMISSION_DENIED) {
ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.ACCESS_COARSE_LOCATION}, 0x11);
}
- 2、这个热点是不能访问外网的,只能是局域网内使用,所以没啥太大的用处。
- 3、保持这个热点的引用是有生命周期,我自己的例子中是局部变量,wifiManager是Actvity的一个私有属性,当拥有该wifiManager的Activity被Destroy时,当前创建的这个本地热点也会自动关闭。因此,需要将这个wifiMananger放在生命周期更长的类中。
综上,在Android 8.0上,没有开启可访问互联网的手机热点的方法。
2. 获取热点开关状态:
Android的API是隐藏的,不能直接用,这里只能用反射的方式来获取热点开启状态
/**
* 判断热点是否开启
* @param context
* @return
*/
public static boolean isWifiApOpen(Context context) {
try {
WifiManager manager = (WifiManager) context.getSystemService(Context.WIFI_SERVICE);
//通过放射获取 getWifiApState()方法
Method method = manager.getClass().getDeclaredMethod("getWifiApState");
//调用getWifiApState() ,获取返回值
int state = (int) method.invoke(manager);
//通过放射获取 WIFI_AP的开启状态属性
Field field = manager.getClass().getDeclaredField("WIFI_AP_STATE_ENABLED");
//获取属性值
int value = (int) field.get(manager);
//判断是否开启
if (state == value) {
return true;
} else {
return false;
}
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
return false;
}
3. 获取当前热点的名称:
获取热点名称的API也是别隐藏了,通过反射的方式获取。但是,Android 8.0以上,即使通过反射的方式也是获取不到的,代码运行时会报SecurityException,后面会说Android 8.0上的处理方法。
public static String getWiFiApSSID(Context context) {
String ssid = "";
// Android 8 及以上无法获取到ssid
if (Build.VERSION.SDK_INT < 26) {
try {
WifiManager manager = (WifiManager) context.getApplicationContext().getSystemService(Context.WIFI_SERVICE);
//拿到getWifiApConfiguration()方法
Method method = manager.getClass().getDeclaredMethod("getWifiApConfiguration");
//调用getWifiApConfiguration()方法,获取到 热点的WifiConfiguration
WifiConfiguration configuration = (WifiConfiguration) method.invoke(manager);
ssid = configuration.SSID;
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
} else {
/*BluetoothAdapter myDevice = BluetoothAdapter.getDefaultAdapter();
ssid = myDevice.getName();*/
return "";
}
return ssid;
}
针对Android 8.0以上,没有现成的API可用。经过搜索,发现可以通过获取手机名称来获取热点名称。
在系统设置,关于手机界面,有设备名称,在名称修改界面可以看到一行字:“使用蓝牙互联、WiFi直连、热点共享和USB连接时,其他设备将看到此名称”
也就是说,设备名称和热点名称是保持一致的。
于是,寻找获取手机名称的方法:
BluetoothAdapter myDevice = BluetoothAdapter.getDefaultAdapter();
String deviceName = myDevice.getName();
但是,这个方法也有问题,就是它只能获取手机默认的名称,如果用户手动修改了设备名称,这个方法返回的还是默认的名称。
所以,没招了。。。谁有更好的方法,欢迎留言。
4. Android开启热点本机IP
Android手机开启热点后,本机的IP地址是固定的192.168.43.1