当前位置: 首页 > news >正文

设置首选网络类型以及调用Android框架层的隐藏API

在Android SDK中提供的framework.jar是阉割版本的,比如有些类标记为hide,这些类不会被打包到这个jar中,而有些只是类中的某个方法或或属性被标记为hide,则这些类或属性会被打包到framework.jar,但是我们无法调用,如果手动输入这些方法或属性,会提示不存在,在这种情况下,我们可以使用反射的方式去调用这些隐藏的API,但是比较麻烦,而且代码可读性没这么高了。

解决方案是,我们可以把没被阉割的framework.jar搞到我们的项目中去使用,这样就不需要使用反射了,代码可读性就会提高。

举个例子,在一台Android 7.1.1的Android手机中,在移动网络设置中有一个 “首选网络类型” 的设置,可以设置4G、3G、2G,截图如下:
在这里插入图片描述
在这里插入图片描述
通过如下代码可以获取当前手机使用的网络是什么:

fun getCurrentNetworkType(context: Context): String {val telephonyManager = context.getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManagerreturn when (telephonyManager.dataNetworkType) {TelephonyManager.NETWORK_TYPE_LTE -> "4G"TelephonyManager.NETWORK_TYPE_NR -> "5G"TelephonyManager.NETWORK_TYPE_HSPA -> "3G"...还有其它很多的类型else -> "Unknown"}
}

在新一点的版本手机中,还会有5G的选项。在旧的一些版本的手机中,可能并不显示多少多少G,而是显示具体的网络类型,比如:

  • LTE/GSM/CDMA
  • LTE only
  • GSM only

如上示例,LTE其实就是4G网络(包含移动、联通、电信),GSM是2G网络(包含移动、联通),CDMA是电信的2G网络,如果选择第一行,就表示优先用LTE,如果没有LTE,它就会用GSMCDMA。而LTE only就只用LTE网络,如果没有LTE网络就会断网。

这个首选网络类型的设置,一般都会有一个类型最全的,即包含移动/联通/电信,且包含5G/4G/3G/2G的选项,且这个选项一般排在最前面,这样的选项用英文描述为“Global”,有全面的/全球的意思,意思就是你用这个选项,你插什么卡都能用,比如联通/电信/移动,而且不管你是4G、3G还是2G都能用。

在Android中有一个Api可以设置网络首选项为 “Global” :

val telephonyManager = getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManager
telephonyManager.setPreferredNetworkTypeToGlobal()

在较新版本的手机中,可能没有设置首选网络类型选项,只有选择关闭5G或是启用5G,其实它底层就是给你设置一个带或不带5G的首选网络类型而已。而有的手机甚至连关闭5G的功能开关都没有,也没有首选网络类型的设置界面,这很不方便,比如有时候测试,我就希望使用4G网络,但是手机上没有设置可以去进行修改,怎么办?那我就可以用代码调用系统API来进行修改,查看setPreferredNetworkTypeToGlobal()的源码,如下:

public boolean setPreferredNetworkTypeToGlobal() {return setPreferredNetworkTypeToGlobal(getSubId());
}public boolean setPreferredNetworkTypeToGlobal(int subId) {return setPreferredNetworkType(subId, RILConstants.NETWORK_MODE_NR_LTE_CDMA_EVDO_GSM_WCDMA);
}

可以看到,它实际上是调用了setPreferredNetworkType方法,然后设置了一个RILConstants.NETWORK_MODE_NR_LTE_CDMA_EVDO_GSM_WCDMA类型的网络,它是包含5G的,NR就是5G类型,如果我想关闭5G,我只需要设置一个没5G的常量即可,比如:RILConstants.NETWORK_MODE_LTE_CDMA_EVDO_GSM_WCDMA。但是当我在代码调用的时候发现没办法调用相关API,如下:
在这里插入图片描述
如上图所示,红色就表示这些API是不存在的,打开TelephonyManager类的源码后发现函数是在的,getSubId()为私有的,如下:

private int getSubId() {if (SubscriptionManager.isUsableSubIdValue(mSubId)) {return mSubId;}return SubscriptionManager.getDefaultSubscriptionId();
}

setPreferredNetworkType只是标记了hide,如下:

/*** Set the preferred network type. * @hide* @deprecated Use {@link #setAllowedNetworkTypesForReason} instead.*/
@Deprecated
@UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553)
public boolean setPreferredNetworkType(int subId, @PrefNetworkMode int networkType) {try {ITelephony telephony = getITelephony();if (telephony != null) {return telephony.setAllowedNetworkTypesForReason(subId,TelephonyManager.ALLOWED_NETWORK_TYPES_REASON_USER,RadioAccessFamily.getRafFromNetworkType(networkType));}} catch (RemoteException ex) {Rlog.e(TAG, "setPreferredNetworkType RemoteException", ex);}return false;
}

这里还有一些信息,maxTargetSdk 表明这个API的最大目标SDK为Android R,即Android 11,所以更新版本的手机使用这个API可能就不管用了。另名deprecated 表明这个函数已经过时了,可以使用setAllowedNetworkTypesForReason代替。

标志为hide的函数是无法直接调用的,这说明Android官方不希望我们调用这个API,所以标志为隐藏。而且RILConstants这整个类都是隐藏的,连源代码都无法查看。

这些API是在 framework.jar中的,Android SDK中提供的framework.jar是阉割的版本,所以我们需要一个完整的版本。可以从手机中获取到完整的,从设备的 /system/framework/framework.jar 位置获取,不记得是否需要root了。拿到这个jar之后,发现它里面是4个dex文件:classes.dex、classes2.dex、classes3.dex和classes4.dex,所以需要转换为class文件,使用dex2jar工具命令:

d2j-dex2jar framework.jar -o framework-output.jar

然后在app目录下创建一个framework目录(目录名称随便都可以,放libs目录下都可以,但是libs目录中一般还有别的jar,且别的jar设置不一样,所以最好单独创建一个目录来放framework.jar,以便单独设置它)。把framework.jar放到这个目录中,然后右击这个jar文件,选择 “Add As Library…”,然后会弹出一个框,如下:
在这里插入图片描述
我们选择Classess即可,然后又弹一个框,如下:
在这里插入图片描述
选择你要使用的模块即可。查看配置的变化,其实这些图形化操作就是在模块的build.gradle.kts的依赖中添加了一行:

implementation(files("framework\\framework.jar"))

所以可以手动输入这一行,如果你记得住的话。记不住就用图形化操作来添加。这里还需要修改一下,改成如下:

compileOnly(files("framework\\framework.jar"))

因为这个framework.jar在手机中本身就有了,位置为:/system/framework/framework.jar,所以我们添加framework.jar只需要参与编译不让代码报错就行,不需要打包到apk中,这就是compileOnlyimplementation的区别。同步gradle之后,RILConstants就可以导入了,如下:

在这里插入图片描述
getSubId()是私有函数,没办法了,只能用反射了。而setPreferredNetworkTypepublic的,为什么还不能调用呢?这是因为Android SDK中本来就有framework.jar,且这个jar中有TelephonyManager.class,只是这个类中的setPreferredNetworkType函数是标记为隐藏而已。这说明Android Studio开发工具默认使用了SDK中的framework.jar中的TelephonyManager.class类了,那怎样设置Android Studio让其使用我们的framework.jar中的TelephonyManager.class类呢?这个我也去寻找过,但是没找到方法,那不管了,经实验,虽然显示红色,但是一样是可以正常运行的。

所以,导入framework.jar好像作用不大,如果是调用一些被隐藏的类,这没问题,但是调用类没隐藏,只是里面的某个函数或属性隐藏,那还是调用不了。如果你有很多要使用的类是整个类都被hide标志的,则这种情况使用framework.jar就比较好用了,因为这些被hide标志的类在SDK中是不存在的,所以在代码中调用时,它会使用我们framework.jar中的类,这会比使用反射简单的多。

在我使用反射调用getSubId()函数的时候,报异常如下:

NoSuchMethodException: android.telephony.TelephonyManager.getSubId []

当时是一头雾水,后面加了系统签名就OK了,而且我没有声明任何的权限。

虽然代码红色也能正常运行,但是还是看着不舒服,所以可以结合kotlin的扩展函数,然后配合反射来使用,这样即不失代码可读性,也不会显示为红色了,RILConstants类中的关于首选网络类型在TelephonyManager也有声明隐藏的属性去指向这些常量,所以都可以通过反射来调用,示例代码如下:

class MainActivity : AppCompatActivity() {private val binding by lazy { ActivityMainBinding.inflate(layoutInflater) }override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(binding.root)val tm = getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManagerval oldPreferredNetworkType = tm.getPreferredNetworkType(tm.getSubId())tm.setPreferredNetworkType(tm.getSubId(), tm.NETWORK_MODE_NR_LTE_TDSCDMA_CDMA_EVDO_GSM_WCDMA)// tm.setPreferredNetworkTypeToGlobal()val newPreferredNetworkType = tm.getPreferredNetworkType(tm.getSubId())Timber.i("旧首选网络类型: $oldPreferredNetworkType")Timber.i("新首选网络类型: $newPreferredNetworkType")printAllNetworkMode()}/** 值为33,最广的网络类型,包含5G、4G、3G、2G */val TelephonyManager.NETWORK_MODE_NR_LTE_TDSCDMA_CDMA_EVDO_GSM_WCDMA: Intget() {val field = TelephonyManager::class.java.getField("NETWORK_MODE_NR_LTE_TDSCDMA_CDMA_EVDO_GSM_WCDMA")return field.getInt(this)}@SuppressLint("SoonBlockedPrivateApi")fun TelephonyManager.getSubId(): Int {val getSubIdMethod = TelephonyManager::class.java.getDeclaredMethod("getSubId").also { it.isAccessible = true }val subId = getSubIdMethod.invoke(this) as Intreturn subId}fun TelephonyManager.getPreferredNetworkType(subId: Int): Int {val getPreferredNetworkTypeMethod = TelephonyManager::class.java.getMethod("getPreferredNetworkType", Int::class.java)val preferredNetworkType = getPreferredNetworkTypeMethod.invoke(this, subId) as Intreturn preferredNetworkType}fun TelephonyManager.setPreferredNetworkType(subId: Int, networkType: Int) {val setPreferredNetworkTypeMethod = TelephonyManager::class.java.getMethod("setPreferredNetworkType", Int::class.java, Int::class.java)setPreferredNetworkTypeMethod.invoke(this, subId, networkType)}fun printAllNetworkMode() {val map = TreeMap<Int, String>()TelephonyManager::class.java.fields.forEach { field ->val fieldName = field.nameif (fieldName.startsWith("NETWORK_MODE_")) {val fieldValue = field.getInt(null)map[fieldValue] = fieldName // 把value当key,是让其对值排序}}for ((key, value) in map) {Timber.i("$value=$key")}}}

注:运行这个代码需要系统权限,不需要声明任何权限,有系统权限就行。运行结果如下:

旧首选网络类型: 24
新首选网络类型: 33
NETWORK_MODE_WCDMA_PREF=0 // 首选 WCDMA(即3G 优先,3G不可用则使用2G)。
NETWORK_MODE_GSM_ONLY=1	  // 仅使用GSM(2G)
NETWORK_MODE_WCDMA_ONLY=2 // 仅使用 WCDMA(3G)。
NETWORK_MODE_GSM_UMTS=3	  // 允许GSM(2G) 和 UMTS(3G)。
NETWORK_MODE_CDMA_EVDO=4  // 允许CDMA和EVDO(电信的2G和3G)
NETWORK_MODE_CDMA_NO_EVDO=5 // 允许在CDMA2000 网络(2G)上工作,但不启用 EV-DO(3G)数据传输。
NETWORK_MODE_EVDO_NO_CDMA=6 // 允许3G,不允许2G
NETWORK_MODE_GLOBAL=7
NETWORK_MODE_LTE_CDMA_EVDO=8
NETWORK_MODE_LTE_GSM_WCDMA=9
NETWORK_MODE_LTE_CDMA_EVDO_GSM_WCDMA=10
NETWORK_MODE_LTE_ONLY=11
NETWORK_MODE_LTE_WCDMA=12
NETWORK_MODE_TDSCDMA_ONLY=13
NETWORK_MODE_TDSCDMA_WCDMA=14
NETWORK_MODE_LTE_TDSCDMA=15
NETWORK_MODE_TDSCDMA_GSM=16
NETWORK_MODE_LTE_TDSCDMA_GSM=17
NETWORK_MODE_TDSCDMA_GSM_WCDMA=18
NETWORK_MODE_LTE_TDSCDMA_WCDMA=19
NETWORK_MODE_LTE_TDSCDMA_GSM_WCDMA=20
NETWORK_MODE_TDSCDMA_CDMA_EVDO_GSM_WCDMA=21
NETWORK_MODE_LTE_TDSCDMA_CDMA_EVDO_GSM_WCDMA=22
NETWORK_MODE_NR_ONLY=23
NETWORK_MODE_NR_LTE=24
NETWORK_MODE_NR_LTE_CDMA_EVDO=25
NETWORK_MODE_NR_LTE_GSM_WCDMA=26
NETWORK_MODE_NR_LTE_CDMA_EVDO_GSM_WCDMA=27
NETWORK_MODE_NR_LTE_WCDMA=28
NETWORK_MODE_NR_LTE_TDSCDMA=29
NETWORK_MODE_NR_LTE_TDSCDMA_GSM=30
NETWORK_MODE_NR_LTE_TDSCDMA_WCDMA=31
NETWORK_MODE_NR_LTE_TDSCDMA_GSM_WCDMA=32
NETWORK_MODE_NR_LTE_TDSCDMA_CDMA_EVDO_GSM_WCDMA=33

对于UMTS,它是一种3G标准,而WCDMA、TD-SCDMA、HSPA等是这个标准的具体实现,其中:

  • WCDMA 是 UMTS 的主要无线电接入技术
  • TD-SCDMA 是中国提出的一种 3G 接入技术,作为 UMTS 的一种备选技术
  • HSPA 是 WCDMA 的增强技术,提供更高的数据速率,进一步提高网络性能。

所以这3种都是3G技术,联通主要使用 WCDMA 和 HSPA ,移动主要使用TD-SCDMA。

CDMA2000 在早期部分用作 2G 网络,但它实际上是一种 3G 网络技术,并提供了比 2G 更高的数据传输速率和更先进的通信功能。虽然 CDMA2000 是 3G 网络技术,但它对早期的 CDMA 2G 网络(例如 IS-95)是向下兼容的。这意味着,在没有高速数据服务的区域,设备仍然可以使用 CDMA2000 的低速数据服务(类似于 2G 服务)。

中国电信采用的是 CDMA2000 技术作为其 3G 网络标准,CDMA2000 网络的增强版本是 EV-DO (Evolution-Data Optimized)。在其 3G 网络中,主要使用的是 EV-DO 技术,类似于 HSPA+,它提供了更高的下载和上传速度。

所以NETWORK_MODE中的CDMA 指的是CDMA2000 ,它是3G网络,但是也向下兼容2G。对于电信的类型:

NETWORK_MODE_CDMA_EVDO=4  	// 允许2G和3G
NETWORK_MODE_CDMA_NO_EVDO=5 // 允许2G,不允许3G
NETWORK_MODE_EVDO_NO_CDMA=6 // 允许3G,不允许2G

注意:慎用setPreferredNetworkTypeToGlobal(),一开始我以为它会选最广的那个网络类型,而且看源代码它也是这么做的,但是在一台Android设备上运行后,再获取它设置的网络类型结果为10,对应为NETWORK_MODE_LTE_CDMA_EVDO_GSM_WCDMA,这并没有包含5G,所以还是自己选择一个具体的值去设置比较保险。

OK,有了这个代码,我们可以分别测试某一些网络了,比如只测2G或只测3G或4G、5G等,对应的网络类型值为:

NETWORK_MODE_GSM_ONLY=1 	// 2G(移动和联通)
NETWORK_MODE_WCDMA_ONLY=2 	// 3G(联通)
NETWORK_MODE_TDSCDMA_ONLY=13// 3G(移动)
NETWORK_MODE_CDMA_EVDO=4  	// 3G(电信)
NETWORK_MODE_LTE_ONLY=11 	// 仅4G
NETWORK_MODE_NR_ONLY=23 	// 仅5G
NETWORK_MODE_NR_LTE=24		// 5G优先,支持4G
NETWORK_MODE_GLOBAL=7 		// 包含所有的网络类型
NETWORK_MODE_NR_LTE_TDSCDMA_CDMA_EVDO_GSM_WCDMA=33 // 包含所有的网络类型

对于WCDMA,ChatGPT说:

  • 中国移动(China Mobile):主要采用 TD-SCDMA(中国自己的 3G 标准),但在某些地区和网络切换时,也会支持 WCDMA。
  • 中国联通(China Unicom):使用 WCDMA 技术作为其 3G 标准。
  • 中国电信(China Telecom):主要使用 CDMA2000 网络,但在一些地区也会支持 WCDMA。

所以,WCDMA不仅仅是联通,对别的运营商可能也是有用的。具体自己实验一下,我没有实验过。

现在好像2G、3G都慢慢在淘汰了,有些设备已经不支持这些类型了,所以平时我们测试时只关注4G、5G即可。而且我发现公司的一台Android设备默认网络类型就是NETWORK_MODE_NR_LTE,即只用4G和5G的类型。

完整示例如下:

界面如下:
在这里插入图片描述

import android.annotation.SuppressLint
import android.content.Context
import android.os.Bundle
import android.telephony.TelephonyManager
import android.view.View
import androidx.appcompat.app.AppCompatActivity
import cn.dazhou.setpreferrednetworktype.databinding.ActivityMainBinding
import timber.log.Timber
import java.util.TreeMap
import kotlin.concurrent.threadclass MainActivity : AppCompatActivity() {private val binding by lazy { ActivityMainBinding.inflate(layoutInflater) }private val tm by lazy { getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManager }override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(binding.root)updateCurrentNetworkType()// tm.setPreferredNetworkTypeToGlobal()printAllNetworkMode()binding.lteOnly.setOnClickListener { setPreferredNetworkType(tm.NETWORK_MODE_LTE_ONLY) }binding.nrOnly.setOnClickListener { setPreferredNetworkType(tm.NETWORK_MODE_NR_ONLY) }binding.lteNrOnly.setOnClickListener { setPreferredNetworkType(tm.NETWORK_MODE_NR_LTE) }binding.ltePref.setOnClickListener { setPreferredNetworkType(tm.NETWORK_MODE_LTE_TDSCDMA_CDMA_EVDO_GSM_WCDMA) }binding.nrPref.setOnClickListener { setPreferredNetworkType(tm.NETWORK_MODE_NR_LTE_TDSCDMA_CDMA_EVDO_GSM_WCDMA) }}fun setPreferredNetworkType(networkType: Int) {binding.loading.visibility = View.VISIBLEthread {// 当切换网络的时候,比如从仅5G优选切换到4G优先,此时是不允许使用5G了,则切换到4G就会花比较多的时间// 如果网络类型变了,但是网络没变,这样的切换就很快,比如从5G优先切换到仅5G就很快,或者从仅4G切换到4G优先也很快。tm.setPreferredNetworkType(tm.getSubId(), networkType)runOnUiThread {updateCurrentNetworkType()binding.loading.visibility = View.GONE}}}@SuppressLint("SetTextI18n")private fun updateCurrentNetworkType() {binding.textView.text = "当前首选网络类型:${getNetworkTypeName(tm.getPreferredNetworkType(tm.getSubId()))}"}/** 值为11,仅4G */val TelephonyManager.NETWORK_MODE_LTE_ONLY: Intget() {val field = TelephonyManager::class.java.getField("NETWORK_MODE_LTE_ONLY")return field.getInt(this)}/** 值为23,仅5G */val TelephonyManager.NETWORK_MODE_NR_ONLY: Intget() {val field = TelephonyManager::class.java.getField("NETWORK_MODE_NR_ONLY")return field.getInt(this)}/** 值为24,仅5G/4G */val TelephonyManager.NETWORK_MODE_NR_LTE: Intget() {val field = TelephonyManager::class.java.getField("NETWORK_MODE_NR_LTE")return field.getInt(this)}/** 值为7,最广的网络类型,包含5G、4G、3G、2G */val TelephonyManager.NETWORK_MODE_GLOBAL: Intget() {val field = TelephonyManager::class.java.getField("NETWORK_MODE_GLOBAL")return field.getInt(this)}/** 值为22,4G优先,包含4G、3G、2G */val TelephonyManager.NETWORK_MODE_LTE_TDSCDMA_CDMA_EVDO_GSM_WCDMA: Intget() {val field = TelephonyManager::class.java.getField("NETWORK_MODE_LTE_TDSCDMA_CDMA_EVDO_GSM_WCDMA")return field.getInt(this)}/** 值为33,5G优先,最广的网络类型,包含5G、4G、3G、2G */val TelephonyManager.NETWORK_MODE_NR_LTE_TDSCDMA_CDMA_EVDO_GSM_WCDMA: Intget() {val field = TelephonyManager::class.java.getField("NETWORK_MODE_NR_LTE_TDSCDMA_CDMA_EVDO_GSM_WCDMA")return field.getInt(this)}@SuppressLint("SoonBlockedPrivateApi")fun TelephonyManager.getSubId(): Int {val getSubIdMethod = TelephonyManager::class.java.getDeclaredMethod("getSubId").also { it.isAccessible = true }val subId = getSubIdMethod.invoke(this) as Intreturn subId}fun TelephonyManager.getPreferredNetworkType(subId: Int): Int {val getPreferredNetworkTypeMethod = TelephonyManager::class.java.getMethod("getPreferredNetworkType", Int::class.java)val preferredNetworkType = getPreferredNetworkTypeMethod.invoke(this, subId) as Intreturn preferredNetworkType}fun TelephonyManager.setPreferredNetworkType(subId: Int, networkType: Int) {val setPreferredNetworkTypeMethod = TelephonyManager::class.java.getMethod("setPreferredNetworkType", Int::class.java, Int::class.java)setPreferredNetworkTypeMethod.invoke(this, subId, networkType)}fun getAllNetworkMode(): TreeMap<Int, String> {if (map.isNotEmpty()) {return map}TelephonyManager::class.java.fields.forEach { field ->val fieldName = field.nameif (fieldName.startsWith("NETWORK_MODE_")) {val fieldValue = field.getInt(null)map[fieldValue] = fieldName // 把value当key,是让其对值排序}}return map}private val map = TreeMap<Int, String>()fun printAllNetworkMode() {val map = getAllNetworkMode()for ((key, value) in map) {Timber.i("$value=$key")}}fun getNetworkTypeName(networkType: Int): String {val map = getAllNetworkMode()return map[networkType] ?: "未知"}}

界面布局代码如下:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayoutxmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:id="@+id/main"android:layout_width="match_parent"android:layout_height="match_parent"tools:context=".MainActivity"><LinearLayoutandroid:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"android:gravity="center"><TextViewandroid:id="@+id/textView"android:layout_width="wrap_content"android:layout_height="wrap_content"android:textSize="16sp"android:text="当前首选网络类型:"/><LinearLayoutandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="vertical"><LinearLayoutandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginTop="16dp"android:orientation="horizontal"><Buttonandroid:layout_width="115dp"android:layout_height="wrap_content"android:text="仅4G"android:id="@+id/lteOnly"/><Buttonandroid:layout_width="115dp"android:layout_height="wrap_content"android:layout_marginStart="16dp"android:text="仅5G"android:id="@+id/nrOnly"/><Buttonandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginStart="16dp"android:text="仅5G/4G"android:id="@+id/lteNrOnly"/></LinearLayout><LinearLayoutandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginTop="8dp"android:orientation="horizontal"><Buttonandroid:layout_width="115dp"android:layout_height="wrap_content"android:text="4G优先"android:id="@+id/ltePref"/><Buttonandroid:layout_width="115dp"android:layout_height="wrap_content"android:layout_marginStart="16dp"android:text="5G优先"android:id="@+id/nrPref"/></LinearLayout></LinearLayout></LinearLayout><LinearLayoutandroid:id="@+id/loading"android:visibility="gone"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"android:gravity="center"android:background="#bb000000"><ProgressBarandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:indeterminateTint="@android:color/white"android:progressTint="@android:color/white"/><TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:textSize="24sp"android:textColor="@color/white"android:text="正在切换网络..."/></LinearLayout></FrameLayout>

相关文章:

设置首选网络类型以及调用Android框架层的隐藏API

在Android SDK中提供的framework.jar是阉割版本的&#xff0c;比如有些类标记为hide&#xff0c;这些类不会被打包到这个jar中&#xff0c;而有些只是类中的某个方法或或属性被标记为hide&#xff0c;则这些类或属性会被打包到framework.jar&#xff0c;但是我们无法调用&#…...

“Gold-YOLO:基于聚合与分发机制的高效目标检测新范式”

&#x1f3e1;作者主页&#xff1a;点击&#xff01; &#x1f916;编程探索专栏&#xff1a;点击&#xff01; ⏰️创作时间&#xff1a;2024年12月26日8点00分 神秘男子影, 秘而不宣藏。 泣意深不见, 男子自持重, 子夜独自沉。 论文源地址&#xff08;有视频&#xf…...

神经网络-AlexNet

AlexNet是在2012年的ImageNet竞赛后&#xff0c;整理发表的文章&#xff0c;也是对CNN网络的衍生。 网络结构 AlexNet网络结构如下图所示&#xff0c;网络分为了上下两部分&#xff0c;对应两个不同的GPU训练&#xff0c;可以更好的利用GPU算力。只有在特殊的网络层后&#x…...

Hutool 发送 HTTP 请求的几种常见写法

最简单的 GET 请求&#xff1a; String result HttpUtil.get("https://www.baidu.com");带参数的 GET 请求&#xff1a; // 方法1: 直接拼接URL参数 String result HttpUtil.get("https://www.baidu.com?name张三&age18");// 方法2: 使用 HashMap…...

【Linux】进度条

本文中&#xff0c;我们来写一个进度条。 本文大纲&#xff1a; 写一个命令行版的进度条。 1.回车换行 2.缓冲区问题&#xff08;本文不深究&#xff09; ​ 2.1测试代码 3.写一个什么样的进度条&#xff1f; ​ version1 ​ version2 回车换行 这俩不是一个概念&…...

【zookeeper核心源码解析】第四课:客户端与服务端读写的io核心流程

系列文章目录 【zookeeper核心源码解析】第一课&#xff1a;zk启动类核心流程序列图 【zookeeper核心源码解析】第二课&#xff1a;俯瞰QuorumPeer启动核心流程&#xff0c;实现选举关键流程 【zookeeper核心源码解析】第三课&#xff1a;leader与follower何时开始同步&#…...

强化学习蘑菇书笔记

绪论 强化学习就是一个智能体在一个不确定的环境中最大化它的奖励。智能体在一个环境中获取某个状态后&#xff0c;做一个动作&#xff0c;也称为决策&#xff0c;在环境中执行这个决策以后&#xff0c;会有一个奖励。尽可能多地获得更多的奖励。 强化学习概述 强化学习与监…...

《机器学习》——线性回归模型

文章目录 线性回归模型简介一元线性回归模型多元线性回归模型误差项分析一元线性模型实例完整代码 多元线性模型实例完整代码 线性回归模型简介 线性回归是利用数理统计中回归分析&#xff0c;来确定两种或两种以上变量间相互依赖的定量关系的一种统计分析方法。 相关关系&…...

Linux(Centos 7.6)网卡信息没有了问题处理

1.问题现象 虚拟机打开后&#xff0c;使用ifconfig查看IP信息&#xff0c;虚拟机默认的网卡名称是ens33&#xff0c;ifconfig没有看到相关问题&#xff0c;远程连接工具Xshell也不能正常访问该虚拟机。 [rootnode1 ~]# ifconfig lo: flags73<UP,LOOPBACK,RUNNING> mtu…...

WEB攻防-通用漏洞-文件上传-js验证-MIME验证-user.ini-语言特征

目录 定义 1.前端验证 2.MIME验证 3.htaccess文件和.user. ini 4.对内容进行了过滤&#xff0c;做了内容检测 5.[ ]符号过滤 6.内容检测php [] {} ; 7.()也被过滤了 8.反引号也被过滤 9.文件头检测 定义 文件上传漏洞是指攻击者上传了一个可执行文件&#xff08;如木马…...

mybatis-plus代码生成器

<!--mybatis-plus--><dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-boot-starter</artifactId><version>3.5.2</version></dependency><!--mybatis-plus-generator 生成器--><depende…...

12.24-12.28Mysql锁阅读笔记

1.Mysql的锁有哪些种类 全局锁&#xff0c; 通过flush tables with read lock 应用场景是全局备份&#xff0c;备份的时候如果有两个表&#xff0c;备份 先备份了用户表&#xff0c;然后用户了商品&#xff0c;再备份商品表 那么商品表库存减少了&#xff0c;然而用户表的育儿…...

支持最新 mysql9的workbench8.0.39 中文汉化教程来了

之前在 B 站上发布了 mysql8 workbench 汉化教程&#xff0c;一年多来帮助很多初学者解决了不熟悉英文的烦恼。 汉化视频可以访问&#xff1a; 2024最新版mysql8.0.39中文版mysql workbench汉化 中文升级 旧版汉化报错解决_哔哩哔哩_bilibili MySql Workbench汉化_哔哩哔哩_…...

golang连接jenkins构建build

1.安装jenkins依赖 go get github.com/bndr/gojenkins2.代码 import ("context""file/utils/logs""github.com/bndr/gojenkins""github.com/gin-gonic/gin""net/http""time" )// 接收单个静态文件上线参数 type…...

SCAU高程进阶题(自用)

18711 字符串去重 Description 一个完全由小写字母组成的长度为n的字符串&#xff0c;现在要求你去除所有重复的字母&#xff0c;并将剩下的字母按从小到大的次序输出。 如输入baaadccaab&#xff0c;输出abcd。 输入格式 第一行一个整数n&#xff0c;表示字符串长度(0<n&…...

基于STM32F103控制L298N驱动两相四线步进电机

文章目录 前言一、模块参数二、接口说明三、准备工作四、直流电机驱动引脚接线效果展示 五、两相四线步进电机驱动步进电机相关概念拍数驱动时序引脚接线效果展示 六、参考示例 前言 L298N 是一种常见的双 H 桥电机驱动模块&#xff0c;广泛用于驱动直流电机和步进电机。它基于…...

libreoffice在Windows和Linux环境的安装和结合Springboot使用教程

前言&#xff1a; 在公司做开发时&#xff0c;遇到一个需求&#xff0c;要求上传的文件有图片&#xff0c;也有word和pdf。预览信息时&#xff0c;既要求能水印展示出来&#xff0c;又要求能大图水印预览。思索许久&#xff0c;我决定采取全部打水印然后转成图片Base64&#x…...

前端开发 -- 自动回复机器人【附完整源码】

一&#xff1a;效果展示 本项目实现了一个简单的网页聊天界面&#xff0c;用户可以在输入框中输入消息&#xff0c;并点击发送按钮或按下回车键来发送消息。机器人会根据用户发送的消息内容&#xff0c;通过关键字匹配来生成自动回复。 二&#xff1a;源代码分享 <!DOCTYP…...

vue+echarts实现疫情折线图

效果&#xff1a; 代码&#xff1a; <<template><div><div id"left1" style "height:800px;width:100%"></div></div> </template><script> //疫情数据//export default {data() {return {data:{//疫情数据…...

服务器nfs文件共享

1. 配置 NFS 服务器(NFS Server) 在 Ubuntu/Debian 上: sudo apt update sudo apt install nfs-kernel-server在 CentOS/RHEL 上: sudo yum install nfs-utils1.2 创建共享目录 选择一个要共享的目录,并确保该目录的权限正确设置。例如,假设我们要共享 /srv/nfs 目录…...

基于Flask实现的医疗保险欺诈识别监测模型

基于Flask实现的医疗保险欺诈识别监测模型 项目截图 项目简介 社会医疗保险是国家通过立法形式强制实施&#xff0c;由雇主和个人按一定比例缴纳保险费&#xff0c;建立社会医疗保险基金&#xff0c;支付雇员医疗费用的一种医疗保险制度&#xff0c; 它是促进社会文明和进步的…...

涂鸦T5AI手搓语音、emoji、otto机器人从入门到实战

“&#x1f916;手搓TuyaAI语音指令 &#x1f60d;秒变表情包大师&#xff0c;让萌系Otto机器人&#x1f525;玩出智能新花样&#xff01;开整&#xff01;” &#x1f916; Otto机器人 → 直接点明主体 手搓TuyaAI语音 → 强调 自主编程/自定义 语音控制&#xff08;TuyaAI…...

NLP学习路线图(二十三):长短期记忆网络(LSTM)

在自然语言处理(NLP)领域,我们时刻面临着处理序列数据的核心挑战。无论是理解句子的结构、分析文本的情感,还是实现语言的翻译,都需要模型能够捕捉词语之间依时序产生的复杂依赖关系。传统的神经网络结构在处理这种序列依赖时显得力不从心,而循环神经网络(RNN) 曾被视为…...

Spring数据访问模块设计

前面我们已经完成了IoC和web模块的设计&#xff0c;聪明的码友立马就知道了&#xff0c;该到数据访问模块了&#xff0c;要不就这俩玩个6啊&#xff0c;查库势在必行&#xff0c;至此&#xff0c;它来了。 一、核心设计理念 1、痛点在哪 应用离不开数据&#xff08;数据库、No…...

Java多线程实现之Thread类深度解析

Java多线程实现之Thread类深度解析 一、多线程基础概念1.1 什么是线程1.2 多线程的优势1.3 Java多线程模型 二、Thread类的基本结构与构造函数2.1 Thread类的继承关系2.2 构造函数 三、创建和启动线程3.1 继承Thread类创建线程3.2 实现Runnable接口创建线程 四、Thread类的核心…...

今日学习:Spring线程池|并发修改异常|链路丢失|登录续期|VIP过期策略|数值类缓存

文章目录 优雅版线程池ThreadPoolTaskExecutor和ThreadPoolTaskExecutor的装饰器并发修改异常并发修改异常简介实现机制设计原因及意义 使用线程池造成的链路丢失问题线程池导致的链路丢失问题发生原因 常见解决方法更好的解决方法设计精妙之处 登录续期登录续期常见实现方式特…...

如何在网页里填写 PDF 表格?

有时候&#xff0c;你可能希望用户能在你的网站上填写 PDF 表单。然而&#xff0c;这件事并不简单&#xff0c;因为 PDF 并不是一种原生的网页格式。虽然浏览器可以显示 PDF 文件&#xff0c;但原生并不支持编辑或填写它们。更糟的是&#xff0c;如果你想收集表单数据&#xff…...

技术栈RabbitMq的介绍和使用

目录 1. 什么是消息队列&#xff1f;2. 消息队列的优点3. RabbitMQ 消息队列概述4. RabbitMQ 安装5. Exchange 四种类型5.1 direct 精准匹配5.2 fanout 广播5.3 topic 正则匹配 6. RabbitMQ 队列模式6.1 简单队列模式6.2 工作队列模式6.3 发布/订阅模式6.4 路由模式6.5 主题模式…...

vulnyx Blogger writeup

信息收集 arp-scan nmap 获取userFlag 上web看看 一个默认的页面&#xff0c;gobuster扫一下目录 可以看到扫出的目录中得到了一个有价值的目录/wordpress&#xff0c;说明目标所使用的cms是wordpress&#xff0c;访问http://192.168.43.213/wordpress/然后查看源码能看到 这…...

stm32wle5 lpuart DMA数据不接收

配置波特率9600时&#xff0c;需要使用外部低速晶振...