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

Android 12 SystemUI下拉状态栏禁止QuickQSPanel展开

1.概述

遇到需求,QuickQSPanel首次下拉后展示快捷功能模块以后就是显示QuickQSPanel,而不展开QSPanel,接下来要从下滑手势下拉出状态栏分析功能实现。也就是直接是展开状态。

2、涉及核心类

frameworks\base\packages\SystemUI\src\com\android\systemui\statusbar\phone

Android12或者Android13frameworks\base\packages\SystemUI\src\com\android\systemui\statusbar\phone\NotificationPanelViewController.java
frameworks/base/packages/SystemUI/res/layout/status_bar_expanded.xmlAndroid10frameworks\base\packages\SystemUI\src\com\android\systemui\statusbar\phone\NotificationPanelViewController.java
frameworks/base/packages/SystemUI/res/layout/status_bar_expanded.xml

3、SystemUI下拉状态栏禁止QuickQSPanel展开的核心功能分析和实现

在系统SystemUI中,它主要负责反馈系统及应用状态并与用户保持大量的交互,systemui中主要核心布局控件为以下部分
在SystemUI中,QSPanel 创建是从 StatusBar#makeStatusBarView 开始的
接下来分析的核心功能和布局如下:
StatusBar:通知消息提示和状态展现
NavigationBar:返回,HOME,Recent
KeyGuard:锁屏模块可以看做单独的应用,提供基本的手机个人隐私保护
Recents:近期应用管理,以堆叠栈的形式展现。
Notification Panel:展示系统或应用通知内容。提供快速系统设置开关。
VolumeUI:来用展示或控制音量的变化:媒体音量、铃声音量与闹钟音量
截屏界面:长按电源键+音量下键后截屏,用以展示截取的屏幕照片/内容
PowerUI:主要处理和Power相关的事件,比如省电模式切换、电池电量变化和开关屏事件等。
RingtonePlayer:铃声播放
StackDivider:控制管理分屏
PipUI:提供对于画中画模式的管理

3.1 status_bar_expanded.xml下拉状态栏布局分析

<?xml version="1.0" encoding="utf-8"?>
<!--
**
** Copyright 2006, The Android Open Source Project
**
** Licensed under the Apache License, Version 2.0 (the "License");
** you may not use this file except in compliance with the License.
** You may obtain a copy of the License at
**
**     http://www.apache.org/licenses/LICENSE-2.0
**
** Unless required by applicable law or agreed to in writing, software
** distributed under the License is distributed on an "AS IS" BASIS,
** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
** See the License for the specific language governing permissions and
** limitations under the License.
*/
--><com.android.systemui.statusbar.phone.NotificationPanelViewxmlns:android="http://schemas.android.com/apk/res/android"xmlns:systemui="http://schemas.android.com/apk/res-auto"android:id="@+id/notification_panel"android:layout_width="match_parent"android:layout_height="match_parent"android:background="@android:color/transparent"><FrameLayoutandroid:id="@+id/big_clock_container"android:layout_width="match_parent"android:layout_height="match_parent"android:visibility="gone" /><ViewStubandroid:id="@+id/keyguard_qs_user_switch_stub"android:layout="@layout/keyguard_qs_user_switch"android:layout_height="match_parent"android:layout_width="match_parent" /><includelayout="@layout/keyguard_bottom_area"android:visibility="gone" /><ViewStubandroid:id="@+id/keyguard_user_switcher_stub"android:layout="@layout/keyguard_user_switcher"android:layout_height="match_parent"android:layout_width="match_parent" /><include layout="@layout/status_bar_expanded_plugin_frame"/><include layout="@layout/dock_info_bottom_area_overlay" /><com.android.keyguard.LockIconViewandroid:id="@+id/lock_icon_view"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_gravity="center"><!-- Background protection --><ImageViewandroid:id="@+id/lock_icon_bg"android:layout_width="match_parent"android:layout_height="match_parent"android:background="@drawable/fingerprint_bg"android:visibility="invisible"/><ImageViewandroid:id="@+id/lock_icon"android:layout_width="match_parent"android:layout_height="match_parent"android:padding="@dimen/lock_icon_padding"android:layout_gravity="center"android:scaleType="centerCrop"/></com.android.keyguard.LockIconView><com.android.systemui.statusbar.phone.NotificationsQuickSettingsContainerandroid:layout_width="match_parent"android:layout_height="match_parent"android:layout_gravity="@integer/notification_panel_layout_gravity"android:id="@+id/notification_container_parent"android:clipToPadding="false"android:clipChildren="false"><includelayout="@layout/keyguard_status_view"android:visibility="gone"/><include layout="@layout/dock_info_overlay" /><FrameLayoutandroid:id="@+id/qs_frame"android:layout="@layout/qs_panel"android:layout_width="@dimen/qs_panel_width"android:layout_height="0dp"android:clipToPadding="false"android:clipChildren="false"systemui:viewType="com.android.systemui.plugins.qs.QS"systemui:layout_constraintStart_toStartOf="parent"systemui:layout_constraintEnd_toEndOf="parent"systemui:layout_constraintTop_toTopOf="parent"systemui:layout_constraintBottom_toBottomOf="parent"/><androidx.constraintlayout.widget.Guidelineandroid:id="@+id/qs_edge_guideline"android:layout_width="wrap_content"android:layout_height="wrap_content"systemui:layout_constraintGuide_percent="0.5"android:orientation="vertical"/><com.android.systemui.statusbar.notification.stack.NotificationStackScrollLayoutandroid:id="@+id/notification_stack_scroller"android:layout_marginTop="@dimen/notification_panel_margin_top"android:layout_width="@dimen/notification_panel_width"android:layout_height="match_parent"android:layout_marginBottom="@dimen/close_handle_underlap"systemui:layout_constraintStart_toStartOf="parent"systemui:layout_constraintEnd_toEndOf="parent"/><include layout="@layout/ambient_indication"android:id="@+id/ambient_indication_container" /><include layout="@layout/photo_preview_overlay" /><includelayout="@layout/keyguard_status_bar"android:visibility="invisible" /><Buttonandroid:id="@+id/report_rejected_touch"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginTop="@dimen/status_bar_header_height_keyguard"android:text="@string/report_rejected_touch"android:visibility="gone" /><com.android.systemui.statusbar.phone.TapAgainViewandroid:id="@+id/shade_falsing_tap_again"android:layout_width="wrap_content"android:layout_height="wrap_content"systemui:layout_constraintLeft_toLeftOf="parent"systemui:layout_constraintRight_toRightOf="parent"systemui:layout_constraintBottom_toBottomOf="parent"android:layout_marginBottom="20dp"android:paddingHorizontal="16dp"android:minHeight="44dp"android:elevation="4dp"android:background="@drawable/rounded_bg_full"android:gravity="center"android:text="@string/tap_again"android:visibility="gone"/></com.android.systemui.statusbar.phone.NotificationsQuickSettingsContainer><FrameLayoutandroid:id="@+id/preview_container"android:layout_width="match_parent"android:layout_height="match_parent"></FrameLayout>
</com.android.systemui.statusbar.phone.NotificationPanelView>

在实现SystemUI下拉状态栏禁止QuickQSPanel展开的核心功能中,通过上述的分析得知,
从SystemUI中状态栏下拉的布局status_bar_expanded.xml的布局文件可以看到主要的下拉事件
是在NotificationPanelViewController.java中负责的,所以具体的下滑事件是在NotificationPanelViewController.java中处理的
所以接下来分析NotificationPanelViewController.java有关下拉状态栏的下滑展开事件

3.2 NotificationPanelViewController.java的下拉状态栏下滑事件分析

NotificationPanelViewController是一个以面板形式展示通知内容的主要视图控制器。它可以显示系统通知,
例如来自操作系统的警告、提醒或更新通知,以及来自其他应用程序的通知,如社交媒体应用或
即时消息应用发送的消息通知。通过notificationPanelViewController,用户可以方便地查看和管理收到的通知。

    public class NotificationPanelViewController extends PanelViewController { {@Overrideprotected TouchHandler createTouchHandler() {return new TouchHandler() {@Overridepublic boolean onInterceptTouchEvent(MotionEvent event) {if (mBlockTouches || mQsFullyExpanded && mQs.disallowPanelTouches()) {return false;}initDownStates(event);// Do not let touches go to shade or QS if the bouncer is visible,// but still let user swipe down to expand the panel, dismissing the bouncer.if (mStatusBar.isBouncerShowing()) {return true;}if (mBar.panelEnabled() && mHeadsUpTouchHelper.onInterceptTouchEvent(event)) {mMetricsLogger.count(COUNTER_PANEL_OPEN, 1);mMetricsLogger.count(COUNTER_PANEL_OPEN_PEEK, 1);return true;}if (!shouldQuickSettingsIntercept(mDownX, mDownY, 0)&& mPulseExpansionHandler.onInterceptTouchEvent(event)) {return true;}if (!isFullyCollapsed() && onQsIntercept(event)) {return true;}return super.onInterceptTouchEvent(event);}

在实现SystemUI下拉状态栏禁止QuickQSPanel展开的核心功能中,通过上述的分析得知,
在NotificationPanelViewController中的onInterceptTouchEvent(MotionEvent event) 手势下滑上滑事件的
处理,而具体处理下拉状态栏下滑事件的是在onQsIntercept(MotionEvent event)中继续处理的
接下来看下onQsIntercept(MotionEvent event)处理手势事件

 private boolean onQsIntercept(MotionEvent event) {int pointerIndex = event.findPointerIndex(mTrackingPointer);if (pointerIndex < 0) {pointerIndex = 0;mTrackingPointer = event.getPointerId(pointerIndex);}final float x = event.getX(pointerIndex);final float y = event.getY(pointerIndex);switch (event.getActionMasked()) {case MotionEvent.ACTION_DOWN:mIntercepting = true;mInitialTouchY = y;mInitialTouchX = x;initVelocityTracker();trackMovement(event);if (shouldQuickSettingsIntercept(mInitialTouchX, mInitialTouchY, 0)) {getParent().requestDisallowInterceptTouchEvent(true);}if (mQsExpansionAnimator != null) {onQsExpansionStarted();mInitialHeightOnTouch = mQsExpansionHeight;mQsTracking = true;mIntercepting = false;mNotificationStackScroller.cancelLongPress();}break;case MotionEvent.ACTION_POINTER_UP:final int upPointer = event.getPointerId(event.getActionIndex());if (mTrackingPointer == upPointer) {// gesture is ongoing, find a new pointer to trackfinal int newIndex = event.getPointerId(0) != upPointer ? 0 : 1;mTrackingPointer = event.getPointerId(newIndex);mInitialTouchX = event.getX(newIndex);mInitialTouchY = event.getY(newIndex);}break;case MotionEvent.ACTION_MOVE:final float h = y - mInitialTouchY;trackMovement(event);if (mQsTracking) {// Already tracking because onOverscrolled was called. We need to update here// so we don't stop for a frame until the next touch event gets handled in// onTouchEvent.setQsExpansion(h + mInitialHeightOnTouch);trackMovement(event);mIntercepting = false;return true;}if (Math.abs(h) > mTouchSlop && Math.abs(h) > Math.abs(x - mInitialTouchX)&& shouldQuickSettingsIntercept(mInitialTouchX, mInitialTouchY, h)) {mQsTracking = true;onQsExpansionStarted();notifyExpandingFinished();mInitialHeightOnTouch = mQsExpansionHeight;mInitialTouchY = y;mInitialTouchX = x;mIntercepting = false;mNotificationStackScroller.cancelLongPress();return true;}break;case MotionEvent.ACTION_CANCEL:case MotionEvent.ACTION_UP:trackMovement(event);if (mQsTracking) {flingQsWithCurrentVelocity(y,event.getActionMasked() == MotionEvent.ACTION_CANCEL);mQsTracking = false;}mIntercepting = false;break;}return false;}

在实现SystemUI下拉状态栏禁止QuickQSPanel展开的核心功能中,通过上述的分析得知,
在NotificationPanelViewController中onQsIntercept(MotionEvent event)的是具体处理下拉事件的方法
在case MotionEvent.ACTION_MOVE:中可以根据手势滑动的距离来设置下拉状态栏中的
QSPanel展开和收缩,通过setQsExpansion(h + mInitialHeightOnTouch);来根据手势滑动的
距离来设置QSPanel是否需要展开,所以具体看下setQsExpansion的相关代码

 private void setQsExpansion(float height) {// add core startif (mQs != null) return;//add core endheight = Math.min(Math.max(height, mQsMinExpansionHeight), mQsMaxExpansionHeight);mQsFullyExpanded = height == mQsMaxExpansionHeight && mQsMaxExpansionHeight != 0;if (height > mQsMinExpansionHeight && !mQsExpanded && !mStackScrollerOverscrolling) {setQsExpanded(true);} else if (height <= mQsMinExpansionHeight && mQsExpanded) {setQsExpanded(false);}mQsExpansionHeight = height;updateQsExpansion();requestScrollerTopPaddingUpdate(false /* animate */);updateHeaderKeyguardAlpha();if (mBarState == StatusBarState.SHADE_LOCKED|| mBarState == StatusBarState.KEYGUARD) {updateKeyguardBottomAreaAlpha();updateBigClockAlpha();}if (mBarState == StatusBarState.SHADE && mQsExpanded&& !mStackScrollerOverscrolling && mQsScrimEnabled) {mQsNavbarScrim.setAlpha(getQsExpansionFraction());}if (mAccessibilityManager.isEnabled()) {setAccessibilityPaneTitle(determineAccessibilityPaneTitle());}if (!mFalsingManager.isUnlockingDisabled() && mQsFullyExpanded&& mFalsingManager.shouldEnforceBouncer()) {mStatusBar.executeRunnableDismissingKeyguard(null, null /* cancelAction */,false /* dismissShade */, true /* afterKeyguardGone */, false /* deferred */);}if (mExpansionListener != null) {mExpansionListener.onQsExpansionChanged(mQsMaxExpansionHeight != 0? mQsExpansionHeight / mQsMaxExpansionHeight : 0);}if (DEBUG) {invalidate();}}

4、修改

在实现SystemUI下拉状态栏禁止QuickQSPanel展开的核心功能中,通过上述的分析得知,
在NotificationPanelViewController.java类中setQsExpansion(float height) 方法来设置下拉状态栏
QSPanel高度的方法中,根据高度判断是否需要展开收缩QSPanel,所以通过增加判断

if (mQs != null)return;

来去掉下滑展开QSPanel,就可以做到只显示QuickQSPanel,而
不用展开QSPanel,实现单次下拉状态栏,定制UI 就可以在QuickQSPanel中定制就可以了

 private void setQsExpansion(float height) {// mQs  不等于空直接复用if (mQs != null) return;// mQs 等于空新建height = Math.min(Math.max(height, mQsMinExpansionHeight), mQsMaxExpansionHeight);mQsFullyExpanded = height == mQsMaxExpansionHeight && mQsMaxExpansionHeight != 0;if (height > mQsMinExpansionHeight && !mQsExpanded && !mStackScrollerOverscrolling&& !mDozing) {setQsExpanded(true);} else if (height <= mQsMinExpansionHeight && mQsExpanded) {setQsExpanded(false);}mQsExpansionHeight = height;updateQsExpansion();requestScrollerTopPaddingUpdate(false /* animate */);updateHeaderKeyguardAlpha();if (mBarState == StatusBarState.SHADE_LOCKED || mBarState == KEYGUARD) {updateKeyguardBottomAreaAlpha();positionClockAndNotifications();updateBigClockAlpha();}if (mAccessibilityManager.isEnabled()) {mView.setAccessibilityPaneTitle(determineAccessibilityPaneTitle());}if (!mFalsingManager.isUnlockingDisabled() && mQsFullyExpanded&& mFalsingCollector.shouldEnforceBouncer()) {mStatusBar.executeRunnableDismissingKeyguard(null, null /* cancelAction */,false /* dismissShade */, true /* afterKeyguardGone */, false /* deferred */);}for (int i = 0; i < mExpansionListeners.size(); i++) {mExpansionListeners.get(i).onQsExpansionChanged(mQsMaxExpansionHeight != 0 ? mQsExpansionHeight / mQsMaxExpansionHeight : 0);}if (DEBUG) {mView.invalidate();}}

相关文章:

Android 12 SystemUI下拉状态栏禁止QuickQSPanel展开

1.概述 遇到需求&#xff0c;QuickQSPanel首次下拉后展示快捷功能模块以后就是显示QuickQSPanel&#xff0c;而不展开QSPanel&#xff0c;接下来要从下滑手势下拉出状态栏分析功能实现。也就是直接是展开状态。 2、涉及核心类 frameworks\base\packages\SystemUI\src\com\and…...

二分思想与相关问题(下)

接下来详细讲解几道比较难的例题&#xff0c;仔细体会二分和其他概念混合在一起的趣味。 下面这道题涉及了“碎片拼接”的概念&#xff0c;很妙&#xff0c;也很难想。 P r o b l e m 5 Problem5 Problem5 同时运行N台电脑的最长时间 LeetCode2141 你有 n 台电脑。给你整数 n…...

【算法专题】搜索算法

二叉树剪枝 LCR 047. 二叉树剪枝 - 力扣&#xff08;LeetCode&#xff09; 本题要求我们将全部为0的二叉树去掉&#xff0c;也就是剪枝&#xff0c;当我们举一个具体的例子进行模拟时&#xff0c;会发现&#xff0c;只关注于对其中一个子树的根节点进行剪枝&#xff0c;由于我…...

B2064 斐波那契数列

题目描述 斐波那契数列是指这样的数列&#xff1a;数列的第一个和第二个数都为 11&#xff0c;接下来每个数都等于前面 22 个数之和。 给出一个正整数 aa&#xff0c;要求斐波那契数列中第 aa 个数是多少。 输入格式 第 11 行是测试数据的组数 nn&#xff0c;后面跟着 nn 行…...

Spark的介绍

一、分布式的思想 不管是数据也好&#xff0c;计算也好&#xff0c;都没有最大的电脑&#xff0c;而是多个小电脑组合而成。 存储&#xff1a;将3T的文件拆分成若干个小文件&#xff0c;例如每500M一个小文件&#xff0c;将这些小文件存储在不同的机器上 。 -- HDFS 计算&#…...

SpringBoot项目是如何启动

启动步骤 概念 运行main方法&#xff0c;初始化SpringApplication 从spring.factories读取listener ApplicationContentInitializer运行run方法读取环境变量&#xff0c;配置信息创建SpringApplication上下文预初始化上下文&#xff0c;将启动类作为配置类进行读取调用 refres…...

科技之光,照亮未来之路“2024南京国际人工智能展会”

全球科技产业的版图正以前所未有的速度重构&#xff0c;而位于中国东部沿海经济带的江浙沪地区&#xff0c;作为科技创新与产业升级的高地&#xff0c;始终站在这一浪潮的最前沿。2024年&#xff0c;这一区域的科技盛宴——“2024南京人工智能展会”即将在南京国际博览中心盛大…...

在深度学习计算机视觉的语义分割中,Boundary和Edge的区别是?

在深度学习中的计算机视觉任务中&#xff0c;语义分割中的 Boundary 和 Edge 其实有一些相似之处&#xff0c;但它们的定义和使用场景略有不同。下面是两者的区别&#xff1a; 1. Boundary&#xff08;边界&#xff09; 定义&#xff1a;Boundary 是指一个对象或区域的边界&a…...

【JAVA入门】Day41 - 字节缓冲流和字符缓冲流

【JAVA入门】Day41 - 字节缓冲流和字符缓冲流 文章目录 【JAVA入门】Day41 - 字节缓冲流和字符缓冲流一、缓冲流的体系结构二、字节缓冲流2.1 字节缓冲流提高效率的底层原理 三、字符缓冲流 在IO流体系中&#xff0c;FileInputStream&#xff0c;FileOutputStream&#xff0c;F…...

collocate join,bucket join,broadcast join,shuffle join对比分析

在分布式计算和大数据处理中,尤其是在使用像 Apache Spark、Hive 等大数据处理框架时,Join 操作是非常常见的。根据数据分布方式和执行机制,Join 操作可以分为不同的类型,如 Collocate Join、Bucket Join、Broadcast Join 和 Shuffle Join。以下是它们的详细对比分析: 1.…...

微信自动通过好友和自动拉人进群,微加机器人这个功能太好用了

又发现一个好用的功能&#xff0c;之前就想找一个这种工具&#xff0c;现在发现可以利用微加机器人的两个功能来实现&#xff0c;分别是加好友和关键词拉群 首先 微加机器人的专业版 > 功能 > 加好友设置 可以设置一个关键词通过,这样别人加好友的时候只需要输入制定内…...

R语言统计分析——功效分析3(相关、线性模型)

参考资料&#xff1a;R语言实战【第2版】 1、相关性 pwr.r.test()函数可以对相关性分析进行功效分析。格式如下&#xff1a; pwr.r.test(n, r, sig.level, power, alternative) 其中&#xff0c;n是观测数目&#xff0c;r是效应值&#xff08;通过线性相关系数衡量&#xff0…...

Django创建模型

1、根据创建好应用模块 python manage.py startapp tests 2、在models文件里创建模型 from django.db import modelsfrom book.models import User# Create your models here. class Tests(models.Model):STATUS_CHOICES ((0, 启用),(1, 停用),# 更多状态...)add_time mode…...

盘点2024年大家都在用的短视频剪辑工具

你现在休息的时间是不是都靠短视频来消遣&#xff1f;看着看着你就会发现短视频制作好像我也可以了吧&#xff1f;这次我就介绍一些简单好操作的短视频剪辑工具。 1.FOXIT视频剪辑 连接直达>>https://www.pdf365.cn/foxitclip/ 短视频剪辑其实也不难&#xff0c;只需…...

“左侧文字横向”的QTabWidget

左侧用 QToolButton 组&#xff0c; 右侧用 QStackedWidget&#xff0c;信号槽绑定切换页面 可定制化高 QButtonGroup* btnGp new QButtonGroup(this);btnGp->addButton(ui->btn1, 0);btnGp->addButton(ui->btn2, 1);btnGp->addButton(ui->btn3, 2);connect…...

python学习之字符串操作

str python # 定义一个字符串变量 print(id(str))print(str) # 打印整个字符串 print(str[0:-1]) # 打印字符串第一个到倒数第二个字符&#xff08;不包含倒数第一个字符&#xff09; print(str[0]) # 打印字符串的第一个字符 print(str[2:5]) # 打印字符串第三到第…...

第7篇:【系统分析师】计算机网络

考点汇总 考点详情 1网络模型和协议&#xff1a;OSI/RM七层模型&#xff0c;网络标准和协议&#xff0c;TCP/IP协议族&#xff0c;端口 七层&#xff1a;应用层&#xff0c;表示层&#xff0c;会话层&#xff0c;传输层&#xff0c;网络层&#xff0c;数据链路层&#xff0c;…...

无人机培训机构组装调试技术详解

一、基础知识学习 在进入无人机组装调试领域之前&#xff0c;扎实的基础知识是不可或缺的。学员需掌握以下内容&#xff1a; 1. 无人机基本原理&#xff1a;了解无人机的飞行原理&#xff0c;包括升力、推力、重力和阻力等基本物理概念&#xff0c;以及无人机的飞行控制系统&…...

‌汽车的舒适进入功能是什么意思?

移动管家汽车的舒适进入系统是指无钥匙进入功能&#xff0c;它允许驾驶者在距离车辆一定范围内自动感应解锁车辆&#xff0c;并具备无钥匙启动功能‌。舒适进入系统的核心优势包括&#xff1a; ‌智能化操作‌&#xff1a;无需传统钥匙&#xff0c;通过智能感应实现车门解锁和…...

杂七杂八-系统环境安装

杂七杂八-系统&环境安装 1. 系统安装2. 环境安装 仅个人笔记使用&#xff0c;后续会根据自己遇到问题记录&#xff0c;感谢点赞关注 1. 系统安装 Windows安装linux子系统WSL2&#xff1a;使用windows系统跑linux程序(大模型)WSL VSCode&#xff1a;VSCode连接WSL实现高效…...

网络六边形受到攻击

大家读完觉得有帮助记得关注和点赞&#xff01;&#xff01;&#xff01; 抽象 现代智能交通系统 &#xff08;ITS&#xff09; 的一个关键要求是能够以安全、可靠和匿名的方式从互联车辆和移动设备收集地理参考数据。Nexagon 协议建立在 IETF 定位器/ID 分离协议 &#xff08;…...

生成xcframework

打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式&#xff0c;可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...

零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?

一、核心优势&#xff1a;专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发&#xff0c;是一款收费低廉但功能全面的Windows NAS工具&#xff0c;主打“无学习成本部署” 。与其他NAS软件相比&#xff0c;其优势在于&#xff1a; 无需硬件改造&#xff1a;将任意W…...

【HarmonyOS 5.0】DevEco Testing:鸿蒙应用质量保障的终极武器

——全方位测试解决方案与代码实战 一、工具定位与核心能力 DevEco Testing是HarmonyOS官方推出的​​一体化测试平台​​&#xff0c;覆盖应用全生命周期测试需求&#xff0c;主要提供五大核心能力&#xff1a; ​​测试类型​​​​检测目标​​​​关键指标​​功能体验基…...

PPT|230页| 制造集团企业供应链端到端的数字化解决方案:从需求到结算的全链路业务闭环构建

制造业采购供应链管理是企业运营的核心环节&#xff0c;供应链协同管理在供应链上下游企业之间建立紧密的合作关系&#xff0c;通过信息共享、资源整合、业务协同等方式&#xff0c;实现供应链的全面管理和优化&#xff0c;提高供应链的效率和透明度&#xff0c;降低供应链的成…...

java调用dll出现unsatisfiedLinkError以及JNA和JNI的区别

UnsatisfiedLinkError 在对接硬件设备中&#xff0c;我们会遇到使用 java 调用 dll文件 的情况&#xff0c;此时大概率出现UnsatisfiedLinkError链接错误&#xff0c;原因可能有如下几种 类名错误包名错误方法名参数错误使用 JNI 协议调用&#xff0c;结果 dll 未实现 JNI 协…...

现代密码学 | 椭圆曲线密码学—附py代码

Elliptic Curve Cryptography 椭圆曲线密码学&#xff08;ECC&#xff09;是一种基于有限域上椭圆曲线数学特性的公钥加密技术。其核心原理涉及椭圆曲线的代数性质、离散对数问题以及有限域上的运算。 椭圆曲线密码学是多种数字签名算法的基础&#xff0c;例如椭圆曲线数字签…...

数据库分批入库

今天在工作中&#xff0c;遇到一个问题&#xff0c;就是分批查询的时候&#xff0c;由于批次过大导致出现了一些问题&#xff0c;一下是问题描述和解决方案&#xff1a; 示例&#xff1a; // 假设已有数据列表 dataList 和 PreparedStatement pstmt int batchSize 1000; // …...

QT3D学习笔记——圆台、圆锥

类名作用Qt3DWindow3D渲染窗口容器QEntity场景中的实体&#xff08;对象或容器&#xff09;QCamera控制观察视角QPointLight点光源QConeMesh圆锥几何网格QTransform控制实体的位置/旋转/缩放QPhongMaterialPhong光照材质&#xff08;定义颜色、反光等&#xff09;QFirstPersonC…...

Docker拉取MySQL后数据库连接失败的解决方案

在使用Docker部署MySQL时&#xff0c;拉取并启动容器后&#xff0c;有时可能会遇到数据库连接失败的问题。这种问题可能由多种原因导致&#xff0c;包括配置错误、网络设置问题、权限问题等。本文将分析可能的原因&#xff0c;并提供解决方案。 一、确认MySQL容器的运行状态 …...