Android热点相关知识整理

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可用。经过搜索,发现可以通过获取手机名称来获取热点名称。


Android热点相关知识整理_第1张图片
图1.jpg
Android热点相关知识整理_第2张图片
图2.jpg

在系统设置,关于手机界面,有设备名称,在名称修改界面可以看到一行字:“使用蓝牙互联、WiFi直连、热点共享和USB连接时,其他设备将看到此名称”
也就是说,设备名称和热点名称是保持一致的。
于是,寻找获取手机名称的方法:

BluetoothAdapter myDevice = BluetoothAdapter.getDefaultAdapter();
String deviceName = myDevice.getName();

但是,这个方法也有问题,就是它只能获取手机默认的名称,如果用户手动修改了设备名称,这个方法返回的还是默认的名称。
所以,没招了。。。谁有更好的方法,欢迎留言。

4. Android开启热点本机IP

Android手机开启热点后,本机的IP地址是固定的192.168.43.1

你可能感兴趣的:(Android热点相关知识整理)