App可以通过了解所连接的网络类型来获益,例如启用某些功能需要5G提供的带宽和低延迟。如果只有2G或3G网络可用,加载时间会比较慢,因此我们可以对加载时间有一定的预期。
在这里,我们可以利用TelephonyManager类来获取各种关于移动网络状态的信息,其中包括网络类型!不过,使用TelephonyManager相当复杂,因为不同的Android版本有不同的情况需要考虑。
下面我提供了一个示例应用程序,它可以检测我们所连接的移动网络类型,不仅仅是5G / 4G / 3G / 2G,还可以获取到具体的子类型。该应用使用了TelephonyManager
,并结合了Jetpack Compose、ViewModel和Kotlin Flow的编写方式。
https://github.com/tdcolvin/NetworkTypeDetector
TelephonyManager
注册以接收网络信息更新获取TelephonyManager
的方法如下:
val telephonyManager =
context.getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManager
...当上下文是一个Context实例时。请注意,一些手机具有多个SIM卡;如果您想查询特定的SIM卡,请调用TelephonyManager
实例上的.createForSubscriptionId(simCardNumber)
。
使用这个实例,我们现在可以获取网络信息更新。所使用的过程取决于Android版本,即用户的Android版本,而不是您应用的目标API级别。
Android 12及更高版本是最简单的情况,因为有一个专用的监听器,并且不需要权限。
要注册接收网络类型信息,我们使用registerTelephonyCallback(Executor, TelephonyCallback)
方法,如下所示:
// The thread Executor used to run the listener. This governs how threads are created and
// reused. Here we use a single thread.
val exec = Executors.newSingleThreadExecutor()// Create the callback object
val callback = object : TelephonyCallback(), TelephonyCallback.DisplayInfoListener {
override fun onDisplayInfoChanged(telephonyDisplayInfo: TelephonyDisplayInfo) {
//TODO: This is next
}
}
// Finally, register the callback so it can start receiving results.
telephonyManager.registerTelephonyCallback(exec, callback)
注销监听器,方法如下:
telephonyManager.unregisterTelephonyCallback(callback)
"""
注册电话管理器回调的原始方法是使用 listen 方法。该方法接受各种类型的监听器;我们需要的是实现 onDisplayInfoChanged
接口的监听器。
有趣的是,这个方法在一个 Android 版本中就出现并消失了:
这需要 READ_PHONE_STATE
权限。我们将在 UI 代码中稍后处理。现在我们将继续,假定我们已经拥有该权限。
// (At the top of the file)
@file:Suppress("DEPRECATION") //Suppressed as required to support old version// SDK 30 uses TelephonyManager.listen() to listen for TelephonyDisplayInfo changes.
// It requires READ_PHONE_STATE permission.
@Suppress("OVERRIDE_DEPRECATION") //Suppressed as required to support old version
// This is the object that will receive the results
val callback = object : PhoneStateListener(exec) {
override fun onDisplayInfoChanged(telephonyDisplayInfo: TelephonyDisplayInfo) {
//TODO: This is next
}
}
// Start listening for results
telephonyManager.listen(callback, PhoneStateListener.LISTEN_DISPLAY_INFO_CHANGED)
注销监听使用下面代码:
telephonyManager.listen(callback, 0)
Android 10及以下没有任何监听网络类型变化的方法。要支持较旧的版本,您需要实施一个循环,每隔几秒主动检查。
检查的代码如下:
val networkType = telephonyManager.dataNetworkType
这需要 READ_PHONE_STATE
权限。
请注意,Android 10及以下版本不能支持5G,因为5G仅在Android 11及以上版本中可用。
在上述 Android 11 和 ≥12 的代码中,会收到一个带有TelephonyDisplayInfo
对象的回调。该对象包含一个 networkType
和一个 overrideNetworkType
。而在 Android ≤10 的代码中,只会收到一个 networkType
。
无论哪种情况,networkType
可以是以下之一:
val baseTypeString = when(networkType) {
TelephonyManager.NETWORK_TYPE_CDMA -> "CDMA"
TelephonyManager.NETWORK_TYPE_1xRTT -> "1xRTT"
TelephonyManager.NETWORK_TYPE_EDGE -> "EDGE"
TelephonyManager.NETWORK_TYPE_EHRPD -> "eHRPD"
TelephonyManager.NETWORK_TYPE_EVDO_0 -> "EVDO rev 0"
TelephonyManager.NETWORK_TYPE_EVDO_A -> "EVDO rev A"
TelephonyManager.NETWORK_TYPE_EVDO_B -> "EVDO rev B"
TelephonyManager.NETWORK_TYPE_GPRS -> "GPRS"
TelephonyManager.NETWORK_TYPE_GSM -> "GSM"
TelephonyManager.NETWORK_TYPE_HSDPA -> "HSDPA"
TelephonyManager.NETWORK_TYPE_HSPA -> "HSPA"
TelephonyManager.NETWORK_TYPE_HSPAP -> "HSPA+"
TelephonyManager.NETWORK_TYPE_HSUPA -> "HSUPA"
TelephonyManager.NETWORK_TYPE_IDEN -> "iDen"
TelephonyManager.NETWORK_TYPE_IWLAN -> "IWLAN"
TelephonyManager.NETWORK_TYPE_LTE -> "LTE"
TelephonyManager.NETWORK_TYPE_NR -> "NR (new radio) 5G"
TelephonyManager.NETWORK_TYPE_TD_SCDMA -> "TD_SCDMA"
TelephonyManager.NETWORK_TYPE_UMTS -> "UMTS"
else -> "[Unknown]"
}
如果可用,overrideNetworkType
会为某些类型的4G和5G连接提供更多信息。以下是选项:
val overrideString = when(overrideNetworkType) {
TelephonyDisplayInfo.OVERRIDE_NETWORK_TYPE_NR_NSA -> "5G non-standalone"
TelephonyDisplayInfo.OVERRIDE_NETWORK_TYPE_NR_ADVANCED -> "5G standalone (advanced)"
TelephonyDisplayInfo.OVERRIDE_NETWORK_TYPE_LTE_ADVANCED_PRO -> "LTE Advanced Pro (5Ge)"
TelephonyDisplayInfo.OVERRIDE_NETWORK_TYPE_LTE_CA -> "LTE (carrier aggregation)"
else -> null
}
可能的null处理:
val netTypeString = overrideString ?: baseTypeString
我在ViewModel中使用了一个Kotlin callbackFlow
来设置上述监听器。如果您以前没有遇到过callbackFlow
,那就太棒了:它是一个流,可用于在外部API上创建一个监听器,当有人注册时,自动移除该监听器。
我使用.stateIn(viewModelScope, SharingStarted.WhileSubscribed(5000), null)
将callbackFlow
从冷可观察对象转换为共享热可观察对象。这样,如果有多个消费者注册,就不会创建多个DisplayInfoListeners
或PhoneStateListeners
。WhileSubscribed(5000)
部分确保可观察对象在所有消费者消失后仍保持存在一段时间,以防它们即将重新出现。(例如,在屏幕旋转的情况下会发生这种情况)。
在Composable中,我使用collectAsStateWithLifecycle()
来确保监听器仅在应用程序位于前台时处于活动状态。
https://github.com/tdcolvin/NetworkTypeDetector