Android 12系统源码_多屏幕(二)模拟辅助设备功能开关实现原理
前言
上一篇我们通过为Android系统开启模拟辅助设备功能开关,最终实现了将一个Activity显示到多个屏幕的效果。
本篇文章我们具体来分析一下当我们开启模拟辅助设备功能开关的时候,Android系统做了什么哪些操作。
一、模拟辅助设备功能开关应用位置
Android12系统中,车机系统有一个专门的开发者选项页面,其完整名称如下:
com.android.car.developeroptions/com.android.car.developeroptions.CarDevelopmentSettingsDashboardActivity
可以发现此Activity对应的包名为com.android.car.developeroptions,输入adb命令:
adb shell pm path com.android.car.developeroptions
返回的结果是:
/system_ext/priv-app/CarDeveloperOptions/CarDeveloperOptions.apk
可以发现是一个名为CarDeveloperOptions的车机系统应用,直接在aosp源码中进行搜索,搜索结果如下:
可以发现这个系统应用位于/packages/services/Car/packages/CarDeveloperOptions目录。
二、模拟辅助设备功能开关相关源码
2.1 系统开发者选项对应的页面声明
CarDeveloperOptions系统应用的AndroidManifest.xml文件中对开发者选项页面的声明如下。
services/Car/packages/CarDeveloperOptions/AndroidManifest.xml
<manifest xmlns:android="http://schemas.android.com/apk/res/android"xmlns:androidprv="http://schemas.android.com/apk/prv/res/android"xmlns:tools="http://schemas.android.com/tools"coreApp="true"package="com.android.car.developeroptions"android:sharedUserId="android.uid.system">...代码省略...<activityandroid:name=".CarDevelopmentSettingsDashboardActivity"android:enabled="false"android:exported="true"android:icon="@drawable/ic_settings_development"android:label="@string/development_settings_title"android:taskAffinity=""android:theme="@style/Theme.CarDeveloperOptions"><intent-filter android:priority="1"><action android:name="android.settings.APPLICATION_DEVELOPMENT_SETTINGS"/><action android:name="com.android.settings.APPLICATION_DEVELOPMENT_SETTINGS"/><category android:name="android.intent.category.DEFAULT"/></intent-filter><meta-data android:name="com.android.settings.summary"android:resource="@string/summary_empty"/><meta-data android:name="com.android.settings.FRAGMENT_CLASS"android:value="com.android.car.developeroptions.CarDevelopmentSettingsDashboardFragment"/><meta-data android:name="com.android.settings.PRIMARY_PROFILE_CONTROLLED"android:value="true"/></activity>...代码省略...
</manifest>
2.2 系统开发者选项对应的Activity
1、CarDevelopmentSettingsDashboardActivity的系统源码非常简洁。
services/Car/packages/CarDeveloperOptions/src/com/android/car/developeroptions/CarDevelopmentSettingsDashboardActivity.java
public class CarDevelopmentSettingsDashboardActivity extends SettingsActivity {private static final String CAR_DEVELOPMENT_SETTINGS_FRAGMENT ="com.android.car.developeroptions.CarDevelopmentSettingsDashboardFragment";@Overrideprotected boolean isValidFragment(String fragmentName) {return CAR_DEVELOPMENT_SETTINGS_FRAGMENT.equals(fragmentName);}@Overrideprotected boolean isToolbarEnabled() {// Disable the default Settings toolbar in favor of a chassis toolbar.return false;}
}
此类中的源码非常简单,最关键的就是CAR_DEVELOPMENT_SETTINGS_FRAGMENT 这个字段,该字段指向了一个Fragment,该Fragment才是开发者选项页面的真正载体,
2、想要明白CarDevelopmentSettingsDashboardActivity页面的具体加载流程,我们有必要看下其父类SettingsActivity 。
packages/apps/Settings/src/com/android/settings/SettingsActivity.java
public class SettingsActivity extends SettingsBaseActivityimplements PreferenceManager.OnPreferenceTreeClickListener,PreferenceFragmentCompat.OnPreferenceStartFragmentCallback,ButtonBarHandler, FragmentManager.OnBackStackChangedListener {@Overrideprotected void onCreate(Bundle savedState) {...代码省略...//加载布局文件setContentView(R.layout.settings_main_prefs);...代码省略...//获取页面参数final String initialFragmentName = getInitialFragmentName(intent);...代码省略...//加载设置模块具体页面对应的fragmentlaunchSettingFragment(initialFragmentName, intent);...代码省略...}/*** 将initialFragmentName指向的fragment加载到当前Activity中*/void launchSettingFragment(String initialFragmentName, Intent intent) {if (initialFragmentName != null) {setTitleFromIntent(intent);Bundle initialArguments = intent.getBundleExtra(EXTRA_SHOW_FRAGMENT_ARGUMENTS);switchToFragment(initialFragmentName, initialArguments, true,mInitialTitleResId, mInitialTitle);} else {// Show search icon as up affordance if we are displaying the main DashboardmInitialTitleResId = R.string.dashboard_title;switchToFragment(TopLevelSettings.class.getName(), null /* args */, false,mInitialTitleResId, mInitialTitle);}}/*** 将fragmentName指向的fragment加载到当前Activity中*/private void switchToFragment(String fragmentName, Bundle args, boolean validate,int titleResId, CharSequence title) {Log.d(LOG_TAG, "Switching to fragment " + fragmentName);if (validate && !isValidFragment(fragmentName)) {throw new IllegalArgumentException("Invalid fragment for this activity: "+ fragmentName);}Fragment f = Utils.getTargetFragment(this, fragmentName, args);if (f == null) {return;}FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();transaction.replace(R.id.main_content, f);if (titleResId > 0) {transaction.setBreadCrumbTitle(titleResId);} else if (title != null) {transaction.setBreadCrumbTitle(title);}transaction.commitAllowingStateLoss();getSupportFragmentManager().executePendingTransactions();Log.d(LOG_TAG, "Executed frag manager pendingTransactions");}
}
packages/apps/Settings/res/layout/settings_main_prefs.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical"android:layout_height="match_parent"android:layout_width="match_parent"><com.android.settings.widget.SettingsMainSwitchBarandroid:id="@+id/switch_bar"android:visibility="gone"android:layout_width="match_parent"android:layout_height="wrap_content"/><FrameLayoutandroid:id="@+id/main_content"android:layout_width="match_parent"android:layout_height="0dp"android:layout_weight="1"/><RelativeLayout android:id="@+id/button_bar"android:layout_height="wrap_content"android:layout_width="match_parent"android:layout_weight="0"android:visibility="gone"><Button android:id="@+id/back_button"android:layout_width="150dip"android:layout_height="wrap_content"android:layout_margin="5dip"android:layout_alignParentStart="true"android:text="@*android:string/back_button_label"/><LinearLayoutandroid:orientation="horizontal"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_alignParentEnd="true"><Button android:id="@+id/skip_button"android:layout_width="150dip"android:layout_height="wrap_content"android:layout_margin="5dip"android:text="@*android:string/skip_button_label"android:visibility="gone"/><Button android:id="@+id/next_button"android:layout_width="150dip"android:layout_height="wrap_content"android:layout_margin="5dip"android:text="@*android:string/next_button_label"/></LinearLayout></RelativeLayout></LinearLayout>
上面我们列出了SettingsActivity和UI加载相关的源码,可以发现SettingsActivity先是加载了一个名为settings_main_prefs的布局文件,然后将initialFragmentName指向的fragment添加到了当前页面上。结合CarDevelopmentSettingsDashboardActivity的源码我们可以知道,开发者选项页面的真正载体是CarDevelopmentSettingsDashboardFragment。
packages/services/Car/packages/CarDeveloperOptions/src/com/android/car/developeroptions/CarDevelopmentSettingsDashboardFragment.java
public class CarDevelopmentSettingsDashboardFragment extends DevelopmentSettingsDashboardFragment {}
packages/apps/Settings/src/com/android/settings/development/DevelopmentSettingsDashboardFragment.java
public class DevelopmentSettingsDashboardFragment extends RestrictedDashboardFragmentimplements OnMainSwitchChangeListener, OemUnlockDialogHost, AdbDialogHost,AdbClearKeysDialogHost, LogPersistDialogHost,BluetoothA2dpHwOffloadRebootDialog.OnA2dpHwDialogConfirmedListener,AbstractBluetoothPreferenceController.Callback {@Overrideprotected int getPreferenceScreenResId() {return Utils.isMonkeyRunning() ? R.xml.placeholder_prefs : R.xml.development_settings;//页面对应的布局文件}private static List<AbstractPreferenceController> buildPreferenceControllers(Context context,Activity activity, Lifecycle lifecycle, DevelopmentSettingsDashboardFragment fragment,BluetoothA2dpConfigStore bluetoothA2dpConfigStore) {final List<AbstractPreferenceController> controllers = new ArrayList<>();...代码省略...controllers.add(new SecondaryDisplayPreferenceController(context));//模拟辅助设备功能组件控制器controllers.add(new GpuViewUpdatesPreferenceController(context));controllers.add(new HardwareLayersUpdatesPreferenceController(context));controllers.add(new DebugGpuOverdrawPreferenceController(context));controllers.add(new DebugNonRectClipOperationsPreferenceController(context));controllers.add(new ForceDarkPreferenceController(context));controllers.add(new EnableBlursPreferenceController(context));controllers.add(new ForceMSAAPreferenceController(context));controllers.add(new HardwareOverlaysPreferenceController(context));controllers.add(new SimulateColorSpacePreferenceController(context));controllers.add(new UsbAudioRoutingPreferenceController(context));controllers.add(new StrictModePreferenceController(context));controllers.add(new ProfileGpuRenderingPreferenceController(context));controllers.add(new KeepActivitiesPreferenceController(context));controllers.add(new BackgroundProcessLimitPreferenceController(context));controllers.add(new CachedAppsFreezerPreferenceController(context));controllers.add(new ShowFirstCrashDialogPreferenceController(context));controllers.add(new AppsNotRespondingPreferenceController(context));controllers.add(new NotificationChannelWarningsPreferenceController(context));controllers.add(new AllowAppsOnExternalPreferenceController(context));controllers.add(new ResizableActivityPreferenceController(context));controllers.add(new FreeformWindowsPreferenceController(context));controllers.add(new DesktopModePreferenceController(context));controllers.add(new NonResizableMultiWindowPreferenceController(context));controllers.add(new ShortcutManagerThrottlingPreferenceController(context));controllers.add(new EnableGnssRawMeasFullTrackingPreferenceController(context));controllers.add(new DefaultLaunchPreferenceController(context, "running_apps"));controllers.add(new DefaultLaunchPreferenceController(context, "demo_mode"));controllers.add(new DefaultLaunchPreferenceController(context, "quick_settings_tiles"));controllers.add(new DefaultLaunchPreferenceController(context, "feature_flags_dashboard"));controllers.add(new DefaultUsbConfigurationPreferenceController(context));controllers.add(new DefaultLaunchPreferenceController(context, "density"));controllers.add(new DefaultLaunchPreferenceController(context, "background_check"));controllers.add(new DefaultLaunchPreferenceController(context, "inactive_apps"));controllers.add(new AutofillLoggingLevelPreferenceController(context, lifecycle));controllers.add(new AutofillResetOptionsPreferenceController(context));controllers.add(new BluetoothCodecDialogPreferenceController(context, lifecycle,bluetoothA2dpConfigStore, fragment));controllers.add(new BluetoothSampleRateDialogPreferenceController(context, lifecycle,bluetoothA2dpConfigStore));controllers.add(new BluetoothBitPerSampleDialogPreferenceController(context, lifecycle,bluetoothA2dpConfigStore));controllers.add(new BluetoothQualityDialogPreferenceController(context, lifecycle,bluetoothA2dpConfigStore));controllers.add(new BluetoothChannelModeDialogPreferenceController(context, lifecycle,bluetoothA2dpConfigStore));controllers.add(new BluetoothHDAudioPreferenceController(context, lifecycle,bluetoothA2dpConfigStore, fragment));controllers.add(new SharedDataPreferenceController(context));controllers.add(new OverlaySettingsPreferenceController(context));return controllers;}
}
2.3 模拟辅助显示设备功能开关
1、由于CarDevelopmentSettingsDashboardFragment构建页面也和其他Settings模块的页面一样,大量使用了Preference这套组件来构建页面,如果对于Preference完全不了解,可以参考一下Android 12系统源码_Settings(一)认识Preference这篇文章。
由于Preference构建视图和常见的Android构建视图的方案有很大差异,要想使用Android那套UI架构来分析Settings模块的源码基本不可行,这里我们直接在aosp中搜索“模拟辅助显示设备”这几个字,搜索结果如下所示。
可以发现“模拟辅助显示设备”这个字符串对应的资源名称为overlay_display_devices_title。
2、继续在aosp中进行类型为.xml,名称为overlay_display_devices_title的资源的搜索,会发现development_settings.xml这个文件有引用。
packages/apps/Settings/res/xml/development_settings.xml
<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android"xmlns:settings="http://schemas.android.com/apk/res-auto"android:key="development_prefs_screen"android:title="@string/development_settings_title">...代码省略...<PreferenceCategoryandroid:key="debug_drawing_category"android:title="@string/debug_drawing_category"android:order="600">...代码省略...<!--模拟辅助显示设备功能开关--><ListPreferenceandroid:key="overlay_display_devices"android:title="@string/overlay_display_devices_title"android:entries="@array/overlay_display_devices_entries"android:entryValues="@array/overlay_display_devices_values" />...代码省略...</PreferenceCategory>...代码省略...
</PreferenceScreen>
base/packages/SettingsLib/res/values-zh-rCN/arrays.xml
<!-- 模拟辅助设备的条目标题 --><string-array name="overlay_display_devices_entries"><item msgid="4497393944195787240">"无"</item><item msgid="8461943978957133391">"480p"</item><item msgid="6923083594932909205">"480p(安全)"</item><item msgid="1226941831391497335">"720p"</item><item msgid="7051983425968643928">"720p(安全)"</item><item msgid="7765795608738980305">"1080p"</item><item msgid="8084293856795803592">"1080p(安全)"</item><item msgid="938784192903353277">"4K"</item><item msgid="8612549335720461635">"4K(安全)"</item><item msgid="7322156123728520872">"4K(画质提升)"</item><item msgid="7735692090314849188">"4K(画质提升、安全)"</item><item msgid="7346816300608639624">"720p,1080p(双屏)"</item></string-array>
base/packages/SettingsLib/res/values/arrays.xml
<!-- 模拟辅助设备的条目属性值 --><string-array name="overlay_display_devices_values" translatable="false" ><item></item><item>720x480/142</item><item>720x480/142,secure</item><item>1280x720/213</item><item>1280x720/213,secure</item><item>1920x1080/320</item><item>1920x1080/320,secure</item><item>3840x2160/320</item><item>3840x2160/320,secure</item><item>1920x1080/320|3840x2160/640</item><item>1920x1080/320|3840x2160/640,secure</item><item>1280x720/213;1920x1080/320</item></string-array>
结合布局文件可知,key值为overlay_display_devices的ListPreference组件就是我们要找的模拟辅助显示设备功能开关组件,其功能开关子条目标题和属性值刚好对应了
3、进一步在aosp中进行类型为.java,名称为overlay_display_devices的资源的搜索,会发现SecondaryDisplayPreferenceController.java这个类有引用,前面承载开发者设置页面内容的DevelopmentSettingsDashboardFragment里面就有引用到这个类。
/packages/apps/Settings/src/com/android/settings/development/SecondaryDisplayPreferenceController.java
public class SecondaryDisplayPreferenceController extends DeveloperOptionsPreferenceControllerimplements Preference.OnPreferenceChangeListener, PreferenceControllerMixin {private static final String OVERLAY_DISPLAY_DEVICES_KEY = "overlay_display_devices";private final String[] mListValues;private final String[] mListSummaries;public SecondaryDisplayPreferenceController(Context context) {super(context);mListValues = context.getResources().getStringArray(R.array.overlay_display_devices_values);mListSummaries = context.getResources().getStringArray(R.array.overlay_display_devices_entries);}@Overridepublic String getPreferenceKey() {return OVERLAY_DISPLAY_DEVICES_KEY;//preference组件的唯一key值}@Overridepublic boolean onPreferenceChange(Preference preference, Object newValue) {//用户选择了条目内容,对开关属性进行更新和数据保存writeSecondaryDisplayDevicesOption(newValue.toString());return true;}@Overridepublic void updateState(Preference preference) {//初始化模拟辅助设备功能更开关的属性值updateSecondaryDisplayDevicesOptions();}@Overrideprotected void onDeveloperOptionsSwitchDisabled() {super.onDeveloperOptionsSwitchDisabled();writeSecondaryDisplayDevicesOption(null);}private void updateSecondaryDisplayDevicesOptions() {//从global中获取当前模拟辅助设备功能开关的属性值final String value = Settings.Global.getString(mContext.getContentResolver(),Settings.Global.OVERLAY_DISPLAY_DEVICES);//获取当前选中的条目序列号int index = 0; // defaultfor (int i = 0; i < mListValues.length; i++) {if (TextUtils.equals(value, mListValues[i])) {index = i;break;}}final ListPreference listPreference = (ListPreference) mPreference;//设置模拟辅助设备功能开关菜单条目列表中当前选中的条目listPreference.setValue(mListValues[index]);listPreference.setSummary(mListSummaries[index]);}private void writeSecondaryDisplayDevicesOption(String newValue) {//更新模拟辅助设备功能开关的属性值到global里面Settings.Global.putString(mContext.getContentResolver(),Settings.Global.OVERLAY_DISPLAY_DEVICES, newValue);updateSecondaryDisplayDevicesOptions();}
}
public final class Settings {@UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553)@TestApi@Readablepublic static final String OVERLAY_DISPLAY_DEVICES = "overlay_display_devices";//模拟辅助设备功能开关属性对应的字段
}
用户对模拟辅助设备功能开关的操作最终会触发SecondaryDisplayPreferenceController的onPreferenceChange方法回调,该方法调用writeSecondaryDisplayDevicesOption将当前用户的选择是开关属性值以key值为overlay_display_devices保存到了global里面,这就意味着我们通过模拟辅助设备功能开关,最终就只是将一串字符串存储到了key为overlay_display_devices的值的global内容。
三、模拟辅助设备功能开关监听者
1、OverlayDisplayAdapter类中有对global的overlay_display_devices字段的变化做监听,这样该字段发生变化的时候可以收到回调。
frameworks/base/services/core/java/com/android/server/display/OverlayDisplayAdapter.java
final class OverlayDisplayAdapter extends DisplayAdapter {private final Handler mUiHandler;// Called with SyncRoot lock held.public OverlayDisplayAdapter(DisplayManagerService.SyncRoot syncRoot,Context context, Handler handler, Listener listener, Handler uiHandler) {super(syncRoot, context, handler, listener, TAG);mUiHandler = uiHandler;}@Overridepublic void registerLocked() {super.registerLocked();getHandler().post(new Runnable() {@Overridepublic void run() {//注册监听overlay_display_devices字段的内容变化getContext().getContentResolver().registerContentObserver(Settings.Global.getUriFor(Settings.Global.OVERLAY_DISPLAY_DEVICES),true, new ContentObserver(getHandler()) {@Overridepublic void onChange(boolean selfChange) { //触发回调updateOverlayDisplayDevices();}});updateOverlayDisplayDevices();}});}private void updateOverlayDisplayDevices() {synchronized (getSyncRoot()) {//继续调用updateOverlayDisplayDevicesLocked方法updateOverlayDisplayDevicesLocked();}}}
2、global的overlay_display_devices字段内容发生变化的时候,会回调OverlayDisplayAdapter的updateOverlayDisplayDevices方法。
该方法上锁之后继续调用updateOverlayDisplayDevicesLocked方法。
final class OverlayDisplayAdapter extends DisplayAdapter {private final ArrayList<OverlayDisplayHandle> mOverlays =new ArrayList<OverlayDisplayHandle>();private String mCurrentOverlaySetting = "";//当前的模拟辅助设备属性值private void updateOverlayDisplayDevicesLocked() {String value = Settings.Global.getString(getContext().getContentResolver(),Settings.Global.OVERLAY_DISPLAY_DEVICES);if (value == null) {value = "";}if (value.equals(mCurrentOverlaySetting)) {return;}mCurrentOverlaySetting = value;if (!mOverlays.isEmpty()) {Slog.i(TAG, "Dismissing all overlay display devices.");for (OverlayDisplayHandle overlay : mOverlays) {overlay.dismissLocked();}mOverlays.clear();}int count = 0;for (String part : value.split(DISPLAY_SPLITTER)) {Matcher displayMatcher = DISPLAY_PATTERN.matcher(part);if (displayMatcher.matches()) {if (count >= 4) {Slog.w(TAG, "Too many overlay display devices specified: " + value);break;}String modeString = displayMatcher.group(1);String flagString = displayMatcher.group(2);ArrayList<OverlayMode> modes = new ArrayList<>();for (String mode : modeString.split(MODE_SPLITTER)) {Matcher modeMatcher = MODE_PATTERN.matcher(mode);if (modeMatcher.matches()) {try {int width = Integer.parseInt(modeMatcher.group(1), 10);int height = Integer.parseInt(modeMatcher.group(2), 10);int densityDpi = Integer.parseInt(modeMatcher.group(3), 10);if (width >= MIN_WIDTH && width <= MAX_WIDTH&& height >= MIN_HEIGHT && height <= MAX_HEIGHT&& densityDpi >= DisplayMetrics.DENSITY_LOW&& densityDpi <= DisplayMetrics.DENSITY_XXXHIGH) {modes.add(new OverlayMode(width, height, densityDpi));continue;} else {Slog.w(TAG, "Ignoring out-of-range overlay display mode: " + mode);}} catch (NumberFormatException ex) {}} else if (mode.isEmpty()) {continue;}}if (!modes.isEmpty()) {int number = ++count;String name = getContext().getResources().getString(com.android.internal.R.string.display_manager_overlay_display_name,number);int gravity = chooseOverlayGravity(number);OverlayFlags flags = OverlayFlags.parseFlags(flagString);Slog.i(TAG, "Showing overlay display device #" + number+ ": name=" + name + ", modes=" + Arrays.toString(modes.toArray())+ ", flags=" + flags);mOverlays.add(new OverlayDisplayHandle(name, modes, gravity, flags, number));continue;}}Slog.w(TAG, "Malformed overlay display devices setting: " + value);}}}
相关文章:

Android 12系统源码_多屏幕(二)模拟辅助设备功能开关实现原理
前言 上一篇我们通过为Android系统开启模拟辅助设备功能开关,最终实现了将一个Activity显示到多个屏幕的效果。 本篇文章我们具体来分析一下当我们开启模拟辅助设备功能开关的时候,Android系统做了什么哪些操作。 一、模拟辅助设备功能开关应用位置 …...

【Go语言初探】(二)、项目文件结构和GOPATH设置
一、go语言项目文件结构 由go/bin、go/src和go/pkg三个子文件夹组成,见下图: 实际项目: 二、gopath路径变量设置 在项目中创建main.go文件后,IDE会提示设置GOPATH路径: 点击“configure GOPATH”,设置GOP…...

三种简单排序:插入排序、冒泡排序与选择排序 【算法 05】
三种简单排序:插入排序、冒泡排序与选择排序 在编程中,排序算法是基础且重要的知识点。虽然在实际开发中,我们可能会直接使用标准库中的排序函数(如C的std::sort),但了解并实现这些基础排序算法对于理解算法…...

Python -- GUI图形界面编程—GUI编程实例 博主也在持续学习中[ 持续更新中!!! 欢迎白嫖 也求粉啊啊啊~ ]
本文介绍了GUI的图形界面编程(相关视频是哔站上的应该搜这个题目就能找到),文章还是很基础的,反正我是小白从0开始,主要的结构tinkter库、重要组件简介(这个不用死记硬背 用的时候再说)、Label&…...
Vue2和Vue3中的diff算法
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、diff算法是什么?二、vue2中的diff算法三、vue3中的diff算法总结 前言 一、diff算法是什么? diff算法很早就存在了,一开…...

springboot使用aop或Jackson进行数据脱敏
1.aop 启动类加EnableAspectJAutoProxy 自定义注解,在实体类中使用表示被脱敏字段 建立aop切面类 可能这里gpt会建议你用Pointcut("execution(public * com.xx.aop..*.get*(..))")这种方式拦截,这种我试了,拦截不住。猜测在mvc返…...
【Solidity】基础介绍
数据类型 值类型 值类型的变量在赋值或作为函数参数传递时会被复制。 布尔类型:bool整数类型: 无符号:uint8、uint16、…、uint256 (uint256 可简写为 uint)有符号:int8、int16、…、int256 (int256可简写为 int) 地址类型&…...
【SpringBoot3】双向实时通讯 websocket
文章目录 一、Websocket使用步骤二、示例1:继承抽象类 AbstractWebSocketHandler后端代码前端代码 三、示例2:使用注解ServerEndpoint后端代码前端代码 四、前端代码封装 一、Websocket使用步骤 在Spring Boot中使用WebSocket是一个常见的需求ÿ…...

搭建内网开发环境(一)|基于docker快速部署开发环境
引言 最近因需要搭建一套简易版的纯内网的开发环境,服务器采用 centos8.0,容器化技术采用 docker 使用 docker-compose 进行容器编排。 该系列教程分为两大类: 软件安装和使用,这类是开发环境常用的软件部署和使用,涉…...

MATLAB R2023b配置Fortran编译器
MATLAB R2023b配置Fortran编译器 引言1. 安装Visual Studio 20192. 安装Intel API20243. 配置xml文件文件4. 设置环境变量5. MATLAB编译Fortran 引言 当我们需要用到MATLAB编译Fortran代码后进行调用计算时,整个配置流程较繁琐。下面以MATLAB R2023b为例࿰…...

2024新型数字政府综合解决方案(七)
新型数字政府综合解决方案通过集成人工智能、大数据、区块链和云计算技术,创建了一个高度智能化和互联互通的政府服务平台,旨在全面提升行政效率、服务质量和透明度。该平台实现了跨部门的数据整合与实时共享,利用人工智能进行智能决策支持和…...
搭建高可用k8s集群
高可用 Kubernetes V1.28.10 安装 文章目录 1. 环境介绍2. 准备工作2.1 修改主机名称2.2 修改hosts文件2.3 关闭防火墙和SLinux2.4 配置SSH免密访问2.4.1 主机名称: k8s-master-01 操作 2.5 配置yum源2.6 禁用Swarp分区2.7 同步时间2.8 配置内核转发及网桥过滤2.9 安装 IPVS 3…...

完美解决html2canvas + jsPDF导出pdf分页内容截断问题
代码地址:https://github.com/HFQ12333/export-pdf.git html2canvas jspdf方案是前端实现页面打印的一种常用方案,但是在实践过程中,遇到的最大问题就是分页截断的问题:当页面元素超过一页A4纸的时候,连续的页面就会…...
14 地址映射
14 地址映射 1、地址划分2、相关函数2.1 ioremap/iounmap2.2 mmap地址映射 3、总结 1、地址划分 明确:在linux系统中,不管是应用程序还是驱动程序,都不允许直接访问外设的物理地址,要想访问必须将物理地址映射到用户虚拟地址或者内核虚拟地址࿰…...

Java Resilience4j-RateLimiter学习
一. 介绍 Resilience4j-RateLimiter 是 Resilience4j 中的一个限流模块,我们对 Resilience4j 的 CircuitBreaker、Retry 已经有了一定的了解,现在来学习 RateLimiter 限流器; 引入依赖; <dependency><groupId>io.g…...

Nginx--地址重写Rewrite
一、什么是Rewrite Rewrite对称URL Rewrite,即URL重写,就是把传入Web的请求重定向到其他URL的过程 URL Rewrite最常见的应用是URL伪静态化,是将动态页面显示为静态页面方式的一种技术。比如http://www.123.com/news/index.php?id123 使用U…...

webflux源码解析(1)-主流程
目录 1.关键实例的创建1.1 实例创建1.2 初始化 2.处理请求的关键流程2.1 从ReactorHttpHandlerAdapter开始2.1 DispatcherHandler的初始化2.2查找mapping handler2.3 处理请求(执行handler)2.4 返回结果处理 3.webflux的配置装配参考: WebFlux是Spring 5.0框架推出的…...
ipad作为扩展屏的最简单方式
将iPad用作扩展屏幕有几种简单而有效的方法。以下是几种常见的方式: 1. Sidecar(苹果官方功能) 适用设备:iPad和Mac(macOS Catalina及以上版本)。功能:Sidecar 是苹果官方的功能,可…...
【卡码网Python基础课 17.判断集合成员】
目录 题目描述与分析一、集合二、集合的常用方法三、代码编写 题目描述与分析 题目描述: 请你编写一个程序,判断给定的整数 n 是否存在于给定的集合中。 输入描述: 有多组测试数据,第一行有一个整数 k,代表有 k 组测…...

生物研究新范式!AI语言模型在生物研究中的应用
–https://doi.org/10.1038/s41592-024-02354-y 留意更多内容,欢迎关注微信公众号:组学之心 Language models for biological research: a primer 研究团队及研究单位 James Zou–Department of Biomedical Data Science, Stanford University, Stan…...
论文解读:交大港大上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化学习框架(二)
HoST框架核心实现方法详解 - 论文深度解读(第二部分) 《Learning Humanoid Standing-up Control across Diverse Postures》 系列文章: 论文深度解读 + 算法与代码分析(二) 作者机构: 上海AI Lab, 上海交通大学, 香港大学, 浙江大学, 香港中文大学 论文主题: 人形机器人…...
【Linux】C语言执行shell指令
在C语言中执行Shell指令 在C语言中,有几种方法可以执行Shell指令: 1. 使用system()函数 这是最简单的方法,包含在stdlib.h头文件中: #include <stdlib.h>int main() {system("ls -l"); // 执行ls -l命令retu…...
蓝桥杯 2024 15届国赛 A组 儿童节快乐
P10576 [蓝桥杯 2024 国 A] 儿童节快乐 题目描述 五彩斑斓的气球在蓝天下悠然飘荡,轻快的音乐在耳边持续回荡,小朋友们手牵着手一同畅快欢笑。在这样一片安乐祥和的氛围下,六一来了。 今天是六一儿童节,小蓝老师为了让大家在节…...
React Native在HarmonyOS 5.0阅读类应用开发中的实践
一、技术选型背景 随着HarmonyOS 5.0对Web兼容层的增强,React Native作为跨平台框架可通过重新编译ArkTS组件实现85%以上的代码复用率。阅读类应用具有UI复杂度低、数据流清晰的特点。 二、核心实现方案 1. 环境配置 (1)使用React Native…...

全球首个30米分辨率湿地数据集(2000—2022)
数据简介 今天我们分享的数据是全球30米分辨率湿地数据集,包含8种湿地亚类,该数据以0.5X0.5的瓦片存储,我们整理了所有属于中国的瓦片名称与其对应省份,方便大家研究使用。 该数据集作为全球首个30米分辨率、覆盖2000–2022年时间…...

SpringCloudGateway 自定义局部过滤器
场景: 将所有请求转化为同一路径请求(方便穿网配置)在请求头内标识原来路径,然后在将请求分发给不同服务 AllToOneGatewayFilterFactory import lombok.Getter; import lombok.Setter; import lombok.extern.slf4j.Slf4j; impor…...

Map相关知识
数据结构 二叉树 二叉树,顾名思义,每个节点最多有两个“叉”,也就是两个子节点,分别是左子 节点和右子节点。不过,二叉树并不要求每个节点都有两个子节点,有的节点只 有左子节点,有的节点只有…...
CMake控制VS2022项目文件分组
我们可以通过 CMake 控制源文件的组织结构,使它们在 VS 解决方案资源管理器中以“组”(Filter)的形式进行分类展示。 🎯 目标 通过 CMake 脚本将 .cpp、.h 等源文件分组显示在 Visual Studio 2022 的解决方案资源管理器中。 ✅ 支持的方法汇总(共4种) 方法描述是否推荐…...

【数据分析】R版IntelliGenes用于生物标志物发现的可解释机器学习
禁止商业或二改转载,仅供自学使用,侵权必究,如需截取部分内容请后台联系作者! 文章目录 介绍流程步骤1. 输入数据2. 特征选择3. 模型训练4. I-Genes 评分计算5. 输出结果 IntelliGenesR 安装包1. 特征选择2. 模型训练和评估3. I-Genes 评分计…...

深度学习习题2
1.如果增加神经网络的宽度,精确度会增加到一个特定阈值后,便开始降低。造成这一现象的可能原因是什么? A、即使增加卷积核的数量,只有少部分的核会被用作预测 B、当卷积核数量增加时,神经网络的预测能力会降低 C、当卷…...