首页 > 其他分享 >Android 监听网络状态变化(无切换中间态版)

Android 监听网络状态变化(无切换中间态版)

时间:2024-07-01 11:58:37浏览次数:19  
标签:11 10 wifi 态版 qingshan 2023 Android 13917 监听

需求:

  • 获取当前的网络状态与类型(WIFI、数据流量)
  • 监听网络状态真正变化
  • 监听网络类型发生变化

业务场景:

  • 用户打开 App 时、使用过程中,出现无网络时,显示 Toast 提示。但当 wifi、数据流量 互相切换的过程中不要有提示。
  • 下载功能支持检测到用户连接上 wifi 时开启静默下载,当换成 数据流量 时停止静默下载。

需求分析:

获取当前网络状态与类型

即提供两个方法,用于获取当前的网络状态、网络类型。

监听网络状态真正变化

网络状态真正变化,首先明确网络状态只有【有网】与【无网】,所以当 WIFI 和 数据流量 同时开启的情况下,仅关闭一项,不会提示网络状态的变更。

监听网络类型发生变化

网络类型发生变化,是指当前使用的网络类型发生变化。举个例子,WIFI、数据流量同时开启,理论上当前网络类型是 WIFI,所以当仅关闭 数据流量 时,不会提示网络类型变更,但仅关闭 WIFI 时,会提示网络类型变更为 数据流量。
这里还需要注意一点,有一个“系统默认网络”的概念:系统通常首选不按流量计费的网络而非按流量计费的网络,首选网速较快的网络而非网速较慢的网络。

技术实现:

常见监听网络状态有三种方式:

  • 监听广播
  • ConnectivityManager#registerDefaultNetworkCallback()
  • ConnectivityManager#registerNetworkCallback()

下面逐一实现看效果:

监听网络状态变化广播


kotlin

复制代码

class NetConnectReceiver: BroadcastReceiver() { override fun onReceive(context: Context?, intent: Intent?) { Log.e("qingshan", "网络状态改变") context?.getSystemService(ConnectivityManager::class.java)?.allNetworkInfo?.filter { it.typeName == "MOBILE" || it.typeName == "WIFI" }?.forEach {networkInfo -> Log.e("qingshan", "${networkInfo?.typeName}, isConnect= ${networkInfo?.isConnected}") } } } //动态注册广播监听 class CustomApplication: Application() { override fun onCreate() { super.onCreate() //这里模拟工具类场景:全局一个监听,然后在工具类中分发。 registerReceiver(NetConnectReceiver(), IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION)) } }

效果:

  • 数据流量、wifi 全关,冷启动。收到两次广播,均表示【数据流量、wifi 】无连接。

kotlin

复制代码

2023-11-10 14:21:58.081 13917-13917 qingshan E 网络状态改变 2023-11-10 14:21:58.082 13917-13917 qingshan E MOBILE, isConnect= false 2023-11-10 14:21:58.082 13917-13917 qingshan E WIFI, isConnect= false 2023-11-10 14:21:58.083 13917-13917 qingshan E 网络状态改变 2023-11-10 14:21:58.083 13917-13917 qingshan E MOBILE, isConnect= false 2023-11-10 14:21:58.083 13917-13917 qingshan E WIFI, isConnect= false

  • 数据流量、wifi 全开,冷启动。收到连续两次广播,均表示当前【WIFI】连接。

kotlin

复制代码

2023-11-10 14:13:46.002 13917-13917 qingshan E 网络状态改变 2023-11-10 14:13:46.002 13917-13917 qingshan E MOBILE, isConnect= false 2023-11-10 14:13:46.002 13917-13917 qingshan E WIFI, isConnect= true 2023-11-10 14:13:46.003 13917-13917 qingshan E 网络状态改变 2023-11-10 14:13:46.003 13917-13917 qingshan E MOBILE, isConnect= false 2023-11-10 14:13:46.003 13917-13917 qingshan E WIFI, isConnect= true

  • 数据流量、wifi 全开,仅关闭流量。无广播。
  • 数据流量、wifi 全开,仅关闭wifi(用于模拟断开wifi 或 wifi网络不佳时系统自动启用数据流量)。收到多次广播,有时表示先【数据流量、wifi 】无连接,然后又出现【数据流量】连接,有时均表示当前【数据流量】连接。

kotlin

复制代码

2023-11-10 14:18:46.622 13917-13917 qingshan E 网络状态改变 2023-11-10 14:18:46.624 13917-13917 qingshan E MOBILE, isConnect= true 2023-11-10 14:18:46.624 13917-13917 qingshan E WIFI, isConnect= false 2023-11-10 14:18:46.624 13917-13917 qingshan E 网络状态改变 2023-11-10 14:18:46.624 13917-13917 qingshan E MOBILE, isConnect= true 2023-11-10 14:18:46.624 13917-13917 qingshan E WIFI, isConnect= false 2023-11-10 14:18:46.665 13917-13917 qingshan E 网络状态改变 2023-11-10 14:18:46.666 13917-13917 qingshan E MOBILE, isConnect= true 2023-11-10 14:18:46.666 13917-13917 qingshan E WIFI, isConnect= false 2023-11-10 14:18:46.666 13917-13917 qingshan E 网络状态改变 2023-11-10 14:18:46.666 13917-13917 qingshan E MOBILE, isConnect= true 2023-11-10 14:18:46.666 13917-13917 qingshan E WIFI, isConnect= false

有时日志如下:


kotlin

复制代码

2023-11-10 14:25:03.460 13917-13917 qingshan E 网络状态改变 2023-11-10 14:25:03.461 13917-13917 qingshan E MOBILE, isConnect= false 2023-11-10 14:25:03.461 13917-13917 qingshan E WIFI, isConnect= false 2023-11-10 14:25:03.461 13917-13917 qingshan E 网络状态改变 2023-11-10 14:25:03.462 13917-13917 qingshan E MOBILE, isConnect= false 2023-11-10 14:25:03.462 13917-13917 qingshan E WIFI, isConnect= false 2023-11-10 14:25:03.527 13917-13917 qingshan E 网络状态改变 2023-11-10 14:25:03.528 13917-13917 qingshan E MOBILE, isConnect= true 2023-11-10 14:25:03.528 13917-13917 qingshan E WIFI, isConnect= false 2023-11-10 14:25:03.528 13917-13917 qingshan E 网络状态改变 2023-11-10 14:25:03.528 13917-13917 qingshan E MOBILE, isConnect= true 2023-11-10 14:25:03.528 13917-13917 qingshan E WIFI, isConnect= false

  • 数据流量、wifi 全关。收到两次广播,均表示【数据流量、wifi 】无连接。

注意:先关 wifi 再关数据时,先打印上面(“数据流量、wifi 全开,仅关闭wifi”场景)日志,再打印下面日志。


kotlin

复制代码

2023-11-10 14:21:58.081 13917-13917 qingshan E 网络状态改变 2023-11-10 14:21:58.082 13917-13917 qingshan E MOBILE, isConnect= false 2023-11-10 14:21:58.082 13917-13917 qingshan E WIFI, isConnect= false 2023-11-10 14:21:58.083 13917-13917 qingshan E 网络状态改变 2023-11-10 14:21:58.083 13917-13917 qingshan E MOBILE, isConnect= false 2023-11-10 14:21:58.083 13917-13917 qingshan E WIFI, isConnect= false

  • 仅开数据流量,再开 wifi(用于模拟使用过程中自动连接上wifi)。收到多次广播,均表示【wifi】连接。

kotlin

复制代码

2023-11-10 14:31:21.285 13917-13917 qingshan E 网络状态改变 2023-11-10 14:31:21.286 13917-13917 qingshan E MOBILE, isConnect= false 2023-11-10 14:31:21.286 13917-13917 qingshan E WIFI, isConnect= true 2023-11-10 14:31:21.287 13917-13917 qingshan E 网络状态改变 2023-11-10 14:31:21.287 13917-13917 qingshan E MOBILE, isConnect= false 2023-11-10 14:31:21.287 13917-13917 qingshan E WIFI, isConnect= true 2023-11-10 14:31:21.317 13917-13917 qingshan E 网络状态改变 2023-11-10 14:31:21.319 13917-13917 qingshan E MOBILE, isConnect= false 2023-11-10 14:31:21.319 13917-13917 qingshan E WIFI, isConnect= true 2023-11-10 14:31:21.319 13917-13917 qingshan E 网络状态改变 2023-11-10 14:31:21.320 13917-13917 qingshan E MOBILE, isConnect= false 2023-11-10 14:31:21.320 13917-13917 qingshan E WIFI, isConnect= true

ConnectivityManager#registerDefaultNetworkCallback()

要求 android sdk >=24 用于监听“系统默认网络”发生变更。 请勿在回调中调用 ConnectivityManager 的同步方法来查找新可用网络的属性,因为这会受到竞态条件的影响。例如:在 onLost() 回调中调用 connectivityManager.getNetworkCapabilities()。


kotlin

复制代码

class CustomApplication: Application() { override fun onCreate() { super.onCreate() //这里模拟工具类场景:全局一个监听,然后在工具类中分发。 getSystemService(ConnectivityManager::class.java).apply { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { registerDefaultNetworkCallback(object : NetworkCallback() { override fun onAvailable(network: Network) { super.onAvailable(network) Log.e("qingshan", "def -> onAvailable") } override fun onLost(network: Network) { super.onLost(network) Log.e("qingshan", "def -> onLost") } override fun onCapabilitiesChanged( network: Network, networkCapabilities: NetworkCapabilities ) { super.onCapabilitiesChanged(network, networkCapabilities) Log.e("qingshan", "def -> 可正常访问网络 = ${networkCapabilities.hasCapability(NetworkCapabilities.NET_CAPABILITY_VALIDATED)} " + "& 数据连接 = ${networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR)} " + "& wifi连接= ${networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_WIFI)}") } }) } } } }

效果

  • 数据流量、wifi 全关,冷启动。无回调触发。
  • 数据流量、wifi 全开,冷启动。触发 onAvailable() 和 onCapabilitiesChanged() 回调。表示当前【WIFI】连接。

kotlin

复制代码

2023-11-10 15:22:31.427 8342-12273 qingshan E def -> onAvailable 2023-11-10 15:22:31.427 8342-12273 qingshan E def -> 可正常访问网络 = true & 数据连接 = false & wifi连接= true

  • 数据流量、wifi 全开,仅关闭流量。无回调触发。
  • 数据流量、wifi 全开,仅关闭wifi(用于模拟断开wifi 或 wifi网络不佳时系统自动启用数据流量)。触发一次 onLost() 和 onAvailable() 回调,触发多次 onCapabilitiesChanged() 回调。表示当前【数据流量】连接。

kotlin

复制代码

2023-11-10 15:30:26.103 8342-12273 qingshan E def -> onLost 2023-11-10 15:30:26.150 8342-12273 qingshan E def -> onAvailable 2023-11-10 15:30:26.151 8342-12273 qingshan E def -> 可正常访问网络 = true & 数据连接 = true & wifi连接= false

  • 数据流量、wifi 全关。数据流量和 wifi 谁先关闭触发的回调不同,wifi 先关会触发两次 onLost() 回调。有时表示【数据流量、wifi】无连接,有时先表示【wifi】无连接,切换为【数据流量】连接,再表示【数据流量、wifi】无连接。

注意:先关 wifi 再关数据时,先打印上面(“数据流量、wifi 全开,仅关闭wifi”场景)日志,再打印下面日志。


kotlin

复制代码

2023-11-10 15:35:42.923 8342-12273 qingshan E def -> onLost

  • 仅开数据流量,再开 wifi(用于模拟使用过程中自动连接上wifi)。触发一次 onAvailable() 回调,触发多次 onCapabilitiesChanged() 回调。表示当前【wifi】连接,但有时会有不同的网络状态。

kotlin

复制代码

2023-11-10 15:33:05.596 8342-12273 qingshan E def -> onAvailable 2023-11-10 15:33:05.596 8342-12273 qingshan E def -> 可正常访问网络 = true & 数据连接 = false & wifi连接= true

有时日志如下:


kotlin

复制代码

2023-11-10 15:39:22.644 8342-12273 qingshan E def -> onAvailable 2023-11-10 15:39:22.645 8342-12273 qingshan E def -> 可正常访问网络 = false & 数据连接 = true & wifi连接= false 2023-11-10 15:39:22.842 8342-12273 qingshan E def -> 可正常访问网络 = true & 数据连接 = true & wifi连接= false

ConnectivityManager#registerNetworkCallback()

用于监听所有网络发生变更。 请勿在回调中调用 ConnectivityManager 的同步方法来查找新可用网络的属性,因为这会受到竞态条件的影响。例如:在 onLost() 回调中调用 connectivityManager.getNetworkCapabilities()。


kotlin

复制代码

class CustomApplication: Application() { override fun onCreate() { super.onCreate() //这里模拟工具类场景:全局一个监听,然后在工具类中分发。 getSystemService(ConnectivityManager::class.java).apply { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { registerNetworkCallback(NetworkRequest.Builder() .addTransportType(NetworkCapabilities.TRANSPORT_CELLULAR) .addTransportType(NetworkCapabilities.TRANSPORT_WIFI) .addCapability(NetworkCapabilities.NET_CAPABILITY_VALIDATED) .build(), object : NetworkCallback(){ override fun onAvailable(network: Network) { super.onAvailable(network) Log.e("qingshan", "all -> onAvailable") } override fun onLost(network: Network) { super.onLost(network) Log.e("qingshan", "all -> onLost") } override fun onCapabilitiesChanged( network: Network, networkCapabilities: NetworkCapabilities ) { super.onCapabilitiesChanged(network, networkCapabilities) Log.e("qingshan", "all -> 可正常访问网络 = ${networkCapabilities.hasCapability(NetworkCapabilities.NET_CAPABILITY_VALIDATED)} " + "& 数据连接 = ${networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR)} " + "& wifi连接= ${networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_WIFI)}") } }) } } } }

效果

  • 数据流量、wifi 全关,冷启动。无回调触发。
  • 数据流量、wifi 全开,冷启动。触发两次 onAvailable() 和 onCapabilitiesChanged() 回调,表示当前【数据流量、wifi】均连接。

kotlin

复制代码

2023-11-10 17:24:37.153 10713-6397 qingshan E all -> onAvailable 2023-11-10 17:24:37.153 10713-6397 qingshan E all -> 可正常访问网络 = true & 数据连接 = false & wifi连接= true 2023-11-10 17:24:37.155 10713-6397 qingshan E all -> onAvailable 2023-11-10 17:24:37.155 10713-6397 qingshan E all -> 可正常访问网络 = true & 数据连接 = true & wifi连接= false

  • 数据流量、wifi 全开,仅关闭流量。触发一次 onLost() 回调,表示当前有网络关闭,具体是什么不知道。

kotlin

复制代码

2023-11-10 17:26:28.243 10713-6397 qingshan E all -> onLost

  • 数据流量、wifi 全开,仅关闭wifi(用于模拟断开wifi 或 wifi网络不佳时系统自动启用数据流量)。触发一次 onLoast() 回调,表示当前有网络关闭。因为关闭的是“系统默认网络”,所以会触发多次 onCapabilitiesChanged() 回调,表示当前变成【数据流量】连接。

kotlin

复制代码

2023-11-10 17:30:43.302 10713-6397 qingshan E all -> onLost 2023-11-10 17:30:45.464 10713-6397 qingshan E all -> 可正常访问网络 = true & 数据连接 = true & wifi连接= false

  • 数据流量、wifi 全关。触发两次 onLost() 回调,但先 wifi 时会在两次 onLost() 中间触发一次 onCapabilitiesChanged() 回调。表示当前网络逐个不可用。

先关wifi,再关数据流量,日志如下:


kotlin

复制代码

2023-11-10 17:36:19.335 10713-6397 qingshan E all -> onLost 2023-11-10 17:36:19.475 10713-6397 qingshan E all -> 可正常访问网络 = true & 数据连接 = true & wifi连接= false 2023-11-10 17:36:20.802 10713-6397 qingshan E all -> onLost

先关数据流量,再关wifi,日志如下:


kotlin

复制代码

2023-11-10 17:37:20.487 10713-6397 qingshan E all -> onLost 2023-11-10 17:37:22.176 10713-6397 qingshan E all -> onLost

  • 仅开数据流量,再开 wifi(用于模拟使用过程中自动连接上wifi)。会触发一次 onAvailable(),触发多次 onCapabilitiesChanged() 回调。表示当前【wifi】已连接(但不意味着正在使用的是 wifi,具体正在使用的网络类型由系统决定,即系统默认网络)。

kotlin

复制代码

2023-11-10 17:33:42.284 10713-6397 qingshan E all -> onAvailable 2023-11-10 17:33:42.284 10713-6397 qingshan E all -> 可正常访问网络 = true & 数据连接 = false & wifi连接= true

  • 仅开wifi,再开数据流量。会触发一次 onAvailable(),触发多次 onCapabilitiesChanged() 回调。表示当前【数据流量】已连接(但不意味着正在使用的是数据流量,具体正在使用的网络类型由系统决定,即系统默认网络)。

kotlin

复制代码

2023-11-10 17:27:49.875 10713-6397 qingshan E all -> onAvailable 2023-11-10 17:27:49.876 10713-6397 qingshan E all -> 可正常访问网络 = true & 数据连接 = true & wifi连接= false 2023-11-10 17:27:49.878 10713-6397 qingshan E all -> 可正常访问网络 = true & 数据连接 = true & wifi连接= false

总结

根据上面的实际运行效果可以得知:

  • ConnectivityManager#registerNetworkCallback() 是监听所有网络变换的,监听范围广,但无法得知当前“系统默认网络”是什么,可以实现判断网络状态,但无法判断网络类型。
  • 广播监听 与 ConnectivityManager#registerDefaultNetworkCallback() 都是监听“系统默认网络”,所以可以实现网络状态与类型的判断,但都存在重复回调的情况,所以要做过滤处理,以及“系统默认网络”切换到普通网络时会有偶现短暂“无网络”状态,需要做延迟处理。
  • 广播监听所使用的方式有标记为“废弃”,同时 ConnectivityManager#registerDefaultNetworkCallback() 也有版本的限制,所以可以两者结合使用,优先使用 egisterDefaultNetworkCallback(),广播监听用于兜底。

所以,综合上述,得出如下工具类实现:


kotlin

复制代码

import android.content.BroadcastReceiver import android.content.Context import android.content.Intent import android.content.IntentFilter import android.net.ConnectivityManager import android.net.Network import android.net.NetworkCapabilities import android.os.Build import android.os.Handler import android.os.Looper sealed class ConnectType(val value: Int) { object Mobile : ConnectType(0) object Wifi : ConnectType(1) object None : ConnectType(-1) companion object { fun convert2Type(value: Int): ConnectType { return when (value) { Mobile.value -> Mobile Wifi.value -> Wifi else -> None } } } } object NetConnectManager { private var mConnectivityManager: ConnectivityManager? = null private val mainHandler = Handler(Looper.getMainLooper()) private val mNetTypeListener = mutableListOf<(type: ConnectType) -> Unit>() private val mNetStateListener = mutableListOf<(isAvailable: Boolean) -> Unit>() private var mCurrentConnectType: ConnectType? = null private var mIsNetAvailable: Boolean? = null /** * 初始化 */ fun init(context: Context) { mConnectivityManager = context.getSystemService(ConnectivityManager::class.java) if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { mConnectivityManager?.registerDefaultNetworkCallback(DefaultNetConnectCallback()) } else { context.registerReceiver( NetConnectReceiver(), IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION) ) } } /** * 注册网络类型监听 */ fun addNetTypeChangeListener(listener: (type: ConnectType) -> Unit) { mNetTypeListener.add(listener) } /** * 反注册网络类型监听 */ fun removeNetTypeChangeListener(listener: (type: ConnectType) -> Unit) { mNetTypeListener.remove(listener) } /** * 注册网络状态监听 */ fun addNetStatusChangeListener(listener: (isAvailable: Boolean) -> Unit) { mNetStateListener.add(listener) } /** * 反注册网络状态监听 */ fun removeNetStatusChangeListener(listener: (isAvailable: Boolean) -> Unit) { mNetStateListener.remove(listener) } /** * 获取当前网络类型 */ fun getConnectType(): ConnectType { if (mConnectivityManager == null) { throw UninitializedPropertyAccessException("请先调用init()初始化") } return mCurrentConnectType ?: mConnectivityManager?.getNetworkCapabilities( mConnectivityManager?.activeNetwork ).let { return if (it?.hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR) == true) { ConnectType.Mobile } else if (it?.hasTransport(NetworkCapabilities.TRANSPORT_WIFI) == true) { ConnectType.Wifi } else { ConnectType.None } } } /** * 获取当前是否网络已连接 */ fun isConnected(): Boolean { if (mConnectivityManager == null) { throw UninitializedPropertyAccessException("请先调用init()初始化") } return (mIsNetAvailable ?: mConnectivityManager?.getNetworkCapabilities(mConnectivityManager?.activeNetwork) ?.hasCapability(NetworkCapabilities.NET_CAPABILITY_VALIDATED)) == true } private class DefaultNetConnectCallback : ConnectivityManager.NetworkCallback() { override fun onLost(network: Network) { super.onLost(network) mCurrentConnectType = ConnectType.None mainHandler.postDelayed({ if (mCurrentConnectType == ConnectType.None && mIsNetAvailable == true) { mIsNetAvailable = false mNetStateListener.forEach { it.invoke(false) } mNetTypeListener.forEach { it(ConnectType.None) } } }, 500) } override fun onCapabilitiesChanged( network: Network, networkCapabilities: NetworkCapabilities ) { super.onCapabilitiesChanged(network, networkCapabilities) mainHandler.post { val isConnected = networkCapabilities.hasCapability(NetworkCapabilities.NET_CAPABILITY_VALIDATED) val isCellular = networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR) val isWifi = networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_WIFI) if (isConnected) { val newConnectType = if (isCellular) ConnectType.Mobile else if (isWifi) ConnectType.Wifi else ConnectType.None if (mIsNetAvailable == null || mIsNetAvailable == false) { mIsNetAvailable = true mNetStateListener.forEach { it(true) } } if (mCurrentConnectType != newConnectType) { mCurrentConnectType = newConnectType mNetTypeListener.forEach { it(newConnectType) } } } } } } private class NetConnectReceiver : BroadcastReceiver() { override fun onReceive(context: Context?, intent: Intent?) { val activityNetworkInfo = context?.getSystemService(ConnectivityManager::class.java)?.allNetworkInfo?.filter { (it.type == ConnectType.Mobile.value || it.type == ConnectType.Wifi.value) && it.isConnected }?.firstOrNull() if (activityNetworkInfo != null) { if (mIsNetAvailable == null || mIsNetAvailable == false) { mIsNetAvailable = true mNetStateListener.forEach { it(true) } } ConnectType.convert2Type(activityNetworkInfo.type).let { connectType -> if (connectType != mCurrentConnectType) { mCurrentConnectType = connectType mNetTypeListener.forEach { it(connectType) } } } return } mCurrentConnectType = ConnectType.None mainHandler.postDelayed({ if (mCurrentConnectType == ConnectType.None && mIsNetAvailable == true) { mIsNetAvailable = false mNetStateListener.forEach { it(false) } mNetTypeListener.forEach { it(ConnectType.None) } } }, 500) } } }

效果:

测试代码:


kotlin

复制代码

class MainActivity : AppCompatActivity(){ override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) Log.e("qingshan", "网络是否连接= ${NetConnectManager.isConnected()} & 网络类型= ${NetConnectManager.getConnectType()}") NetConnectManager.addNetTypeChangeListener { Log.e("qingshan", "网络类型[监听]= $it") } NetConnectManager.addNetStatusChangeListener { Log.e("qingshan", "网络是否已连接[监听]= $it") } } }

  • 数据流量、wifi 全关,冷启动。

kotlin

复制代码

qingshan E 网络是否连接= false & 网络类型= com.stefan.simpleskin.ConnectType$None@6f21c25

  • 数据流量、wifi 全开,冷启动。

kotlin

复制代码

qingshan E 网络是否连接= true & 网络类型= com.stefan.simpleskin.ConnectType$Wifi@34771fa qingshan E 网络类型[监听]= com.stefan.simpleskin.ConnectType$Wifi@34771fa

  • 数据流量、wifi 全开,仅关闭流量。 无监听回调。
  • 数据流量、wifi 全开,仅关闭wifi(用于模拟断开wifi 或 wifi网络不佳时系统自动启用数据流量)。

kotlin

复制代码

qingshan E 网络类型[监听]= com.stefan.simpleskin.ConnectType$Mobile@cf38ed1

  • 数据流量、wifi 全关。

kotlin

复制代码

qingshan E 网络是否已连接[监听]= false qingshan E 网络类型[监听]= com.stefan.simpleskin.ConnectType$None@fddad36

  • 仅开数据流量,再开 wifi(用于模拟使用过程中自动连接上wifi)。

kotlin

复制代码

qingshan E 网络类型[监听]= com.stefan.simpleskin.ConnectType$Wifi@34771fa

  • 仅开wifi,再开数据流量。无监听回调。

标签:11,10,wifi,态版,qingshan,2023,Android,13917,监听
From: https://blog.csdn.net/qq_17105113/article/details/140097011

相关文章

  • Android应用启动流程一次看透
    1.1、冷启动和热启动冷启动:当启动应用时,后台没有该应用的进程,这时系统会重新创建一个新的进程分配给该应用,然后再根据启动的参数,启动对应的进程组件,这个启动方式就是冷启动。热启动:当启动应用时,后台已有该应用的进程(例:按back键、home键,应用虽然会退出,但是该应用的进程是依然......
  • Android系统之System Server大纲
    前言SystemServer是android基本服务的提供者,是android系统运行的最基本需求,所有server运行在一个叫system_process的进程中,system_process进程是androidjava虚拟机跑的第一个进程,从Zygote创建而来,是andorid系统最重要的java虚拟机。可以说,整个android系统的业务都是围绕syste......
  • Android Gradle 开发与应用 (三): 依赖管理与版本控制
    目录1.依赖管理的重要性1.1依赖的类型1.2Gradle中的依赖声明2.版本控制的策略2.1固定版本与动态版本2.2版本冲突的解决3.Gradle插件的使用3.1常用的Gradle插件3.2自定义插件4.多模块项目中的依赖管理4.1模块间依赖4.2公共依赖5.依赖版本管理的最......
  • 【转】Androidstudio报错Algorithm HmacPBESHA256 not available
     删除debug.keystone这个文件就可以了。 https://blog.csdn.net/O_PUTI/article/details/138227534 -----参考了更改GradleJDK等的办法都没有用,最终通过一个一个问题拍错解决。第一个问题:版本不一致 第二个问题秘钥获取不成功:删除这个文件 然后就编译成功了。......
  • 【RK3588】Android系统OTA升级详细流程(三)
    接着上一篇文章,做一下OTA包远程升级测试。1.搭建OTA升级服务器Ubuntu系统下搭建OTA服务器。我们将使用Tomcat来搭建web服务器,Tomcat是一个免费、开源的轻量级web应用服务器。1.1安装JDKTomcat依赖于Java运行环境,所以需要先安装Java环境。可自行前往JavaArchive|Oracle......
  • 适用于 Android 的 几种短信恢复应用程序
    Android设备上的短信丢失可能由于多种原因而丢失,例如意外删除、恢复出厂设置、系统崩溃或病毒攻击。是否有应用程序可以恢复Android上已删除的短信?幸运的是,有几款短信恢复应用程序可以扫描您的Android手机并从内存或SIM卡中检索已删除的短信。然而,并非所有短信恢复应用......
  • 测试Android-邮E行原生浏览器文件访问
    docx下载doc下载xls下载xlsx下载zip下载......
  • 在Linux中,如何使用 tcpdump 监听主机为 192.168.1.1,tcp 端⼝为 80 的数据,并将将输出结
    在Linux中,要使用tcpdump监听目标为主机IP为192.168.1.1,且TCP端口为80的数据包,并将输出结果保存到名为tcpdump.log的文件中,可以按照以下步骤操作:打开终端:首先,你需要在具有足够权限的用户账户下打开一个终端窗口。通常,root权限是必要的,因为普通用户可能无法监听网络接口的所有流......
  • Android Studio开发Android TV
    目录AndroidStudio开发AndroidTV步骤1:准备工作步骤2:创建新项目步骤3:选择Activity模板步骤4:配置Activity步骤5:开发和测试步骤6:优化体验结论AndroidStudio开发AndroidTVAndroidTV是基于Android操作系统的电视平台,可以为用户提供丰富的娱乐和应用体验。它适用于......
  • 【RK3588】Android系统OTA升级详细流程(一)
    OTA(overtheair)升级是Android系统提供的标准软件升级方式。它功能强大,提供了完全升级(完整包)、增量升级模式(差异包),既可以通过本地升级,也可以通过网络升级。本文我们将编译三个版本的Android镜像,分别为V1.0.0版本、V1.0.1版本以及V1.0.2版本,其中V1.0.0版本作为基础版本直接烧录......