【Android 源码分析】Activity生命周期之onStop-2
忽然有一天,我想要做一件事:去代码中去验证那些曾经被“灌输”的理论。
– 服装学院的IT男
本篇已收录于Activity短暂的一生系列
欢迎一起学习讨论Android应用开发或者WMS
V:WJB6995
Q:707409815
正文
生命周期系列:
-
Activity生命周期之onPause
-
onCreate,onStart,onResume-1
-
onCreate,onStart,onResume-2
-
Activity生命周期之onStop-1
-
Activity生命周期之onStop-2
-
Activity生命周期之onDestory
当前为“onStop”第二篇,整个流程分为3步:
-
- addToStopping 流程,将应用添加进需要stop的列表 – system_service进程处理
-
- stopIfPossible 流程,也就是开始执行stop – system_service进程处理
-
- 应用端处理stop流程 – 应用进程处理
前面个都是system_service进程处理的,第三步是在 SourceActivity 的应用进程处理。
1. 第二步:stopIfPossible 开始执行stop
根据上一小节后的2个线索,得到以下信息:
-
- “wm_stop_activity”的打印在 ActivityRecord::stopIfPossible 方法,并且打印log后就构建了 StopActivityItem
-
- 在 ActivityTaskSupervisor::activityIdleInternal 方法内有堆栈打印,日志如下:
ActivityTaskManager: activityIdleInternal:
Callers=com.android.server.wm.ActivityClientController.activityIdle:142android.app.IActivityClientController$Stub.onTransact:550com.android.server.wm.ActivityClientController.onTransact:125android.os.Binder.execTransactInternal:1285
根据这2点信息其实就已经可以获得到这么一个完整的调用链了:
ActivityClientController::activityIdleActivityTaskSupervisor::activityIdleInternalActivityTaskSupervisor::processStoppingAndFinishingActivitiesActivityRecord::stopIfPossibleActivityRecord::setState -- 设置为 STOPPINGStopActivityItem.obtain -- 构建触发StopActivityItem
后面的逻辑晚点再看,先分析 ActivityClientController::activityIdle 是怎么调用过来的。
1.1 ActivityClientController::activityIdle 的触发逻辑
TargetActivity 执行 onResume 的逻辑是在应用端是由 ActivityThread::handleResumeActivity方法开始的,而这个方法最后添加了一个 Idler 到 IdleHandler中,这个 Idler 就是触发
ActivityClientController::activityIdle 的地方。
整个AOSP中调用 ActivityClientController::activityIdle 的地方只有2个,根据log定位是在 ActivityThread 下的内部类 Idler 中。
# ActivityThreadpublic void handleResumeActivity(ActivityClientRecord r, boolean finalStateRequest,boolean isForward, String reason) {......// 触发 TargetActivity的 onResumeif (!performResumeActivity(r, finalStateRequest, reason)) {return;}......// 拿到activityfinal Activity a = r.activity;......//setView 逻辑r.nextIdle = mNewActivities;mNewActivities = r;//触发SourceActivity 的onStop if (localLOGV) Slog.v(TAG, "Scheduling idle handler for " + r);Looper.myQueue().addIdleHandler(new Idler());}
根据调用顺序:先触发 TargetActivity的 onResume,再触发 SourceActivity 的 onStop 。
这一点和实际的log打印也是对上了,接下来就看看 Idler() 是怎么执行的了。
addIdleHandler 方法会把一个 IdleHandler 添加到先洗队列的 mIdleHandlers 列表,这里是 Handler 知识点下关于 IdleHandler 的知识,不知道的可以搜一下相关的文章看看。
IdleHandler 的特效就是会在 CPU 空闲的时候才会执行相关任务。
1.1.1 Idler
Idler 是 ActivityThread 的内部类
# ActivityThreadprivate class Idler implements MessageQueue.IdleHandler {@Overridepublic final boolean queueIdle() {ActivityClientRecord a = mNewActivities;......if (a != null) {mNewActivities = null;final ActivityClient ac = ActivityClient.getInstance();ActivityClientRecord prev;do {// 日志if (localLOGV) Slog.v(TAG, "Reporting idle of " + a +" finished=" +(a.activity != null && a.activity.mFinished));if (a.activity != null && !a.activity.mFinished) {// 重点* 执行 ActivityClientController::activityIdleac.activityIdle(a.token, a.createdConfig, stopProfiling);a.createdConfig = null;}prev = a;a = a.nextIdle;prev.nextIdle = null;} while (a != null);}if (stopProfiling) {mProfiler.stopProfiling();}return false;}}
这里的重点就是在主线程空闲的时候,就会执行 queueIdle() 方法,进而触发 SourceActivity 的 StopActivityItem 事务构建。
在日志中搜索添加 Idler 的日志和 queueIdle 方法执行的日志信息如下:
Line 12732: 04-01 20:50:30.891 24294 24294 V ActivityThread: Scheduling idle handler for ActivityRecord{b7ece46 token=android.os.BinderProxy@1fee9be {com.google.android.dialer/com.google.android.dialer.extensions.GoogleDialtactsActivity}}Line 19074: 04-01 20:50:31.170 24294 24294 V ActivityThread: Reporting idle of ActivityRecord{b7ece46 token=android.os.BinderProxy@1fee9be {com.google.android.dialer/com.google.android.dialer.extensions.GoogleDialtactsActivity}} finished=false
可以看到时间上确实还是有很大间隔的。
1.2 ActivityClientController::activityIdle 的处理
上面看到了是如何调用过来的,现在看一下后续的调用逻辑
# ActivityClientController@Overridepublic void activityIdle(IBinder token, Configuration config, boolean stopProfiling) {final long origId = Binder.clearCallingIdentity();try {synchronized (mGlobalLock) {// TraceTrace.traceBegin(TRACE_TAG_WINDOW_MANAGER, "activityIdle");final ActivityRecord r = ActivityRecord.forTokenLocked(token);if (r == null) {return;}// 重点逻辑mTaskSupervisor.activityIdleInternal(r, false /* fromTimeout */,false /* processPausingActivities */, config);......}} finally {Trace.traceEnd(TRACE_TAG_WINDOW_MANAGER);Binder.restoreCallingIdentity(origId);}}
直接调用了 ActivityTaskSupervisor::activityIdleInternal 方法,这个方法在签名 SourceActivity 的 addToStopping 流程已经看过一次,现在条件不同执行的逻辑也不一样,再看一下这个方法。
# ActivityTaskSupervisorvoid activityIdleInternal(ActivityRecord r, boolean fromTimeout,boolean processPausingActivities, Configuration config) {// 重点* 1 logif (DEBUG_ALL) Slog.v(TAG, "Activity idle: " + r);// 重点* 2. ActivityRecord 不为null才执行内部逻辑if (r != null) {// 重点* 3. logif (DEBUG_IDLE) Slog.d(TAG_IDLE, "activityIdleInternal: Callers="+ Debug.getCallers(4));......}......// Atomically retrieve all of the other things to do. 原子检索所有其他要做的事情// 重点* 4. 看方法名是处理 停止和finish的Activity辑在这里processStoppingAndFinishingActivities(r, processPausingActivities, "idle");if (DEBUG_IDLE) {// 重点* 5. log Slogf.i(TAG, "activityIdleInternal(): r=%s, mStartingUsers=%s", r, mStartingUsers);}......}
这一次因为参数 r 不为null,所以还会打印“重点* 3”处的log,然后进入processStoppingAndFinishingActivities 方法。
1.2.1 再看 processStoppingAndFinishingActivities 逻辑(真正执行)
# ActivityTaskSupervisor/*** Processes the activities to be stopped or destroyed. This should be called when the resumed* 处理要停止或销毁的Activity。这应该在resumed时调用* activities are idle or drawn.*/private void processStoppingAndFinishingActivities(ActivityRecord launchedActivity,boolean processPausingActivities, String reason) {// 准备要执行 Stop 的Activity 集合 ArrayList<ActivityRecord> readyToStopActivities = null;// 重点 * 1. 遍历mStoppingActivitiesfor (int i = mStoppingActivities.size() - 1; i >= 0; --i) {// 获取到ActivityRecord (当前分析场景就1个)final ActivityRecord s = mStoppingActivities.get(i);final boolean animating = s.isAnimating(TRANSITION | PARENTS,ANIMATION_TYPE_APP_TRANSITION | ANIMATION_TYPE_RECENTS)|| s.inTransition();// 日志ProtoLog.v(WM_DEBUG_STATES, "Stopping %s: nowVisible=%b animating=%b "+ "finishing=%s", s, s.nowVisible, animating, s.finishing);// 条件满足才执行if (!animating || mService.mShuttingDown) {......// 打印 准备stop的logProtoLog.v(WM_DEBUG_STATES, "Ready to stop: %s", s);if (readyToStopActivities == null) {readyToStopActivities = new ArrayList<>();}// 重点 * 2. 添加进集合readyToStopActivities.add(s);// 从集合中移除mStoppingActivities.remove(i);}}// 重点 * 3. 遍历readyToStopActivitiesfinal int numReadyStops = readyToStopActivities == null ? 0 : readyToStopActivities.size();for (int i = 0; i < numReadyStops; i++) {final ActivityRecord r = readyToStopActivities.get(i);// 检查该ActivityRecord对象是否在历史记录中。 if (r.isInHistory()) {// 如果该ActivityRecord对象正在结束(可能是用户或系统触发的结束操作)。if (r.finishing) {// TODO(b/137329632): Wait for idle of the right activity, not just any.r.destroyIfPossible(reason);} else {// 重点* 4. 如果ActivityRecord对象不在结束状态,则尝试停止它r.stopIfPossible();}}}......}
这个时候再看这个方法就能理解方法上面的注释了:This should be called when the resumed 。
-
- 遍历 mStoppingActivities 集合,这个集合里是有当前场景的唯一元素是 SourceActivity (当前是QuickstepLauncher)。
这一次的打印如下:
WindowManager: Stopping ActivityRecord{f40797d u0 com.android.launcher3/.uioverrides.QuickstepLauncher} t20}: nowVisible=false animating=false finishing=false mShuttingDown=false
-
- 由于“animating=false”所以会执行 if 内部语句,把 SourceActivity 添加到 readyToStopActivities 集合中。
并打印ProtoLog
WindowManager: Ready to stop: ActivityRecord{f40797d u0 com.android.launcher3/.uioverrides.QuickstepLauncher} t20}
-
- 遍历 readyToStopActivities 下的元素,然后执行 ActivityRecord::stopIfPossible 方法
processStoppingAndFinishingActivities 方法是处理 Activity 的核心方法,他的触发点当前文章就分析到了2个,另外肯定还有其他的调用逻辑
比如我将 ActivityThread::handleResumeActivity 方法下最后一行将 Idler 添加进 IdleHandler 的代码删了,本以为就不会执行 SourceActivity 的 onStop 了,没想到还会通过发送一个 PROCESS_STOPPING_AND_FINISHING_MSG 消息来触发 processStoppingAndFinishingActivities 方法的执行。
毕竟 Activity 生命周期是基础代码,google这快代码的健壮性肯定是考虑周全的。
1.3 真正触发StopActivityItem构建的地方 ActivityRecord::stopIfPossible
# ActivityRecordvoid stopIfPossible() {// logif (DEBUG_SWITCH) Slog.d(TAG_SWITCH, "Stopping: " + this);final Task rootTask = getRootTask();......try {stopped = false;// 关键log (这里还是即将 stop)ProtoLog.v(WM_DEBUG_STATES, "Moving to STOPPING: %s (stop requested)", this);......// 重点* 1. 设置Activity的状态为STOPPING,原因为"stopIfPossible"setState(STOPPING, "stopIfPossible");if (DEBUG_VISIBILITY) {// logSlog.v(TAG_VISIBILITY, "Stopping:" + this);}// events日志 :wm_stop_activity: [0,198530468,com.example.myapplication/.MainActivity]EventLogTags.writeWmStopActivity(mUserId, System.identityHashCode(this), shortComponentName);// 重点* 2. 构建执行 Stop 事务mAtmService.getLifecycleManager().scheduleTransaction(app.getThread(), token,StopActivityItem.obtain(configChangeFlags));mAtmService.mH.postDelayed(mStopTimeoutRunnable, STOP_TIMEOUT);}}
就做了2件事:
-
- 设置状态为 STOPPING
-
- 构建执行 StopActivityItem
不过这里有很多log,跟踪问题的时候可以关注一下。
2. SystemService 端小结
当前冷启动场景下,SourceActivity 的onStop处理,SystemService 端其实就做了2件事。
-
- 把 SourceActivity 添加到 ActivityTaskSupervisor 类下的 mStoppingActivities 集合中,这一步我成为“addToStopping 流程”,对应的events日志为:“wm_add_to_stopping”
这一步的执行也毕竟早,在 SourceActivity 执行完 Pause 流程就触发了。
- 把 SourceActivity 添加到 ActivityTaskSupervisor 类下的 mStoppingActivities 集合中,这一步我成为“addToStopping 流程”,对应的events日志为:“wm_add_to_stopping”
-
- 执行 ActivityTaskSupervisor::processStoppingAndFinishingActivities 方法,将 mStoppingActivities 集合下的 ActivityRecord 取出,执行 ActivityRecord::stopIfPossible 方法,在这里构建执行 StopActivityItem 。
当前分析的是常见流程的调用逻辑,还有一些其他逻辑也会触发onStop,但是目前看来都会执行 ActivityTaskSupervisor::processStoppingAndFinishingActivities 方法,所以以后遇到相关问题可以关注一下该方法的执行。
到这里 SystemService 的处理就结束了,后续流程就在应用端了。
那后续就是应用进程执行 Stop 的流程了。
3.应用端 onStop 处理
应用端的处理就相对简单了,先看一下 StopActivityItem 的定义
# StopActivityItem@Overridepublic void execute(ClientTransactionHandler client, ActivityClientRecord r,PendingTransactionActions pendingActions) {Trace.traceBegin(TRACE_TAG_ACTIVITY_MANAGER, "activityStop");// 触发应用进程stopclient.handleStopActivity(r, mConfigChanges, pendingActions,true /* finalStateRequest */, "STOP_ACTIVITY_ITEM");Trace.traceEnd(TRACE_TAG_ACTIVITY_MANAGER);}
具体的实现肯定是在ActivityThread中了,整理的调用链如下:
StopActivityItem::executeClientTransactionHandler::handleStopActivityActivityThread::handleStopActivityActivityThread::performStopActivityInnerActivityThread::callActivityOnStopActivityThread::callActivityOnSaveInstanceState -- OnSaveInstanceState 逻辑Activity::performStopInstrumentation.callActivityOnStopActivity::onStop -- 生命周期ActivityThread::updateVisibility -- 处理View不可见
开始撸代码
# ActivityThread@Overridepublic void handleStopActivity(ActivityClientRecord r, int configChanges,PendingTransactionActions pendingActions, boolean finalStateRequest, String reason) {r.activity.mConfigChangeFlags |= configChanges;final StopInfo stopInfo = new StopInfo();// 重点* 1. onStop 流程performStopActivityInner(r, stopInfo, true /* saveState */, finalStateRequest,reason);if (localLOGV) Slog.v(TAG, "Finishing stop of " + r + ": win=" + r.window);// 重点* 2. 更新Activity的可见性状态为不可见updateVisibility(r, false);......}
这里分为2步逻辑,先执行 onStop ,再处理 View 可见性。
3.1 onStop 执行逻辑
# ActivityThreadprivate void performStopActivityInner(ActivityClientRecord r, StopInfo info,boolean saveState, boolean finalStateRequest, String reason) {......// One must first be paused before stopped... // 1. 必须先执行 pause,才可以 stop。performPauseActivityIfNeeded(r, reason);......// 2. 主流程callActivityOnStop(r, saveState, reason);}
这个方法做了2件事:
-
- 必须先执行 pause,才可以 stop,所以需要执行一下 performPauseActivityIfNeeded 方法,不过正常逻辑执行到这,已经是pause状态了,所以这个方法内部进去就会return。
# ActivityThreadprivate void performPauseActivityIfNeeded(ActivityClientRecord r, String reason) {if (r.paused) {// You are already paused silly...return;}......}
-
- 继续执行主流程,触发Activity的 onStop 回调
# ActivityThreadprivate void callActivityOnStop(ActivityClientRecord r, boolean saveState, String reason) {// Before P onSaveInstanceState was called before onStop, starting with P it's// called after. Before Honeycomb state was always saved before onPause.final boolean shouldSaveState = saveState && !r.activity.mFinished && r.state == null&& !r.isPreHoneycomb();final boolean isPreP = r.isPreP();if (shouldSaveState && isPreP) {// 重点* 1. callActivityOnSaveInstanceStatecallActivityOnSaveInstanceState(r);}try {// 重点* 2. onStop流程r.activity.performStop(r.mPreserveWindow, reason);} ......if (shouldSaveState && !isPreP) {// 重点* 1. callActivityOnSaveInstanceStatecallActivityOnSaveInstanceState(r);}}
-
- 可以看到2次执行了 onSaveInstanceState 回调,这个可以在【Android 11源码分析】看相关的
-
- 是本次分析的主流程,已经执行到Activity的方法了,里生命周期onStop不远了
# Activityprivate Instrumentation mInstrumentation;final void performStop(boolean preserveWindow, String reason) {......// Fragment的处理mFragments.dispatchStop();mCalled = false;// 回调onStop流程mInstrumentation.callActivityOnStop(this);......}
Instrumentation 这个类中处理了所有的Activity生命周期回调。
# Instrumentationpublic void callActivityOnStop(Activity activity) {// onStopactivity.onStop();}
3.2 Stop 处理View的可见性
# ActivityThreadprivate void updateVisibility(ActivityClientRecord r, boolean show) {// 拿到对应Activity的 DecorViewView v = r.activity.mDecor;if (v != null) {if (show) {// 如果Activity当前在服务器端不可见, 就需要处理成可见if (!r.activity.mVisibleFromServer) {r.activity.mVisibleFromServer = true;mNumVisibleActivities++;if (r.activity.mVisibleFromClient) {// 设置可见r.activity.makeVisible();}}} else {// 如果Activity当前在服务器端可见, 就需要处理成不可见if (r.activity.mVisibleFromServer) {r.activity.mVisibleFromServer = false;mNumVisibleActivities--;// 影藏Viewv.setVisibility(View.INVISIBLE);}}}}
所以经常会有人执行了onStop 才是真正的Activity不可见,原因就是onStop的逻辑会把 View设置为不可见,间接的Activity也就不可见了。
4. 总结
当前只是以桌面冷启动应用的场景来分析桌面的 onStop 流程,整个流程分析完对 onStop 流程也有了一个比较完整的了解,但是当前分析的调用流程并不能代表所有场景。具体情况还是需要具体分析,比如应用内启动 Activity 的调用链肯定和当前是有别的,但是无论怎么样,2个主流程还是要执行的。
相关文章:
【Android 源码分析】Activity生命周期之onStop-2
忽然有一天,我想要做一件事:去代码中去验证那些曾经被“灌输”的理论。 – 服装…...
SpringCloudStream+RocketMQ多topic
之前写过两篇关于SpringCloudStream文章 spring-cloud-stream版本升级,告别旧注解EnableBinding,拥抱函数式编程_spring-cloud-stream output注解没有了-CSDN博客 SpringCloudStreamRocketMQ事务消息配置_spring-cloud-starter-stream-rocketmq-CSDN博…...
随记 前端框架React的初步认识
区分语言 像js 就是构建的最基本的 React框架和Vue框架,用自己的话来说的话,就是对js进行了一层封装,使使用js更加的方便 但是,React框架和Vue框架又不能这么简单的理解,因为这些框架里还封装了一些其他的东西。 向…...

数据结构 ——— 单链表oj题:链表分割(带哨兵位单向不循环链表实现)
目录 题目要求 手搓简易单链表 代码实现 题目要求 现有一链表的头指针 ListNode* head ,给一定值 x ,编写一段代码将所有小于 x 的节点排在其余节点之前,且不能改变原来的数据顺序,返回重新排列后的链表的头节点 举例说明&a…...

华为 HCIP-Datacom H12-821 题库 (32)
🐣博客最下方微信公众号回复题库,领取题库和教学资源 🐤诚挚欢迎IT交流有兴趣的公众号回复交流群 🦘公众号会持续更新网络小知识😼 1.当一个运行 MSTP 协议的交换设备端口收到一个配置BPDU 时,会与设备保存的全局配…...
[C++][第三方库][brpc]详细讲解
目录 1.介绍2.安装3.类与接口介绍1.日志输出类与接口2.ProtoBuf类与接口3.服务端类与接口4.客户端类与接口 4.使用0.一般流程1.Server2.客户端 -- 同步调用3.客户端 -- 异步调用 1.介绍 brpc是用C编写的工业级RPC框架,常用于搜索、存储、机器学习、广告、推荐等高性…...
Python-Learning
补充不熟悉的python知识 1 **是表示平方 注释是用来阐述代码要做什么,以及是如何做的 先编写行之有效的代码,再决定是对其做进一步改进,还是转而去编写新代码 列表常用是append,但也有pop,这个pop是输出一个值&…...

如何让 Android 的前端页面像 iOS 一样“优雅”?
作者:方英杰(崇之) 最近在调研前端页面适配 Android 端异形屏的方案,调研过程中发现了一些比较有意思的点,本文主要是做一个总结。 一、提出问题 首先,我们需要知道 Android 上的前端适配面临着什么问题。 问题其实很…...

10.3学习
1.循环依赖 循环依赖其实就是循环引用,也就是两个或者两个以上的 Bean 互相持有对方,最终形成闭环。比如A 依赖于B,B又依赖于A Spring中循环依赖场景有: prototype 原型 bean循环依赖 构造器的循环依赖(构造器注入)…...

Shell文本处理(三)
Shell文本处理三:字符串处理 1、字符串截取(切片)2、字符串替换3、字符串删除4、去除空格5、大小写转换6、字符串分割7、去除中文在Shell中,字符串没有单独的数据类型,一切都是变量。但这并不意味着我们不能像在Java、Python等其他编程语言中那样处理字符串 1、字符串截取…...
5个python多线程简单示例
示例 1: 基本线程创建 # 示例 1: 基本线程创建 import threading import timedef print_numbers():for i in range(5):time.sleep(1)print(i)# 创建线程 thread threading.Thread(targetprint_numbers)# 启动线程 thread.start()# 等待线程完成(可选) …...
Streamlit:用Python快速构建交互式Web应用
在传统的Web开发中,开发者常常需要编写大量的前端和后端代码,才能实现一个简单的交互式Web应用。Streamlit 通过简化这一过程,使得你只需要用Python编写代码,就能快速创建具有丰富交互功能的Web应用。本文将介绍如何使用Streamlit…...
深入浅出Vue.js组件开发:从基础到高级技巧
解锁Python编程的无限可能:《奇妙的Python》带你漫游代码世界 Vue.js 是一个轻量级且功能强大的 JavaScript 框架,专注于构建用户界面。它的核心优势之一是组件系统,它允许开发者通过模块化、可复用的方式构建复杂的应用程序。在这篇文章中,我们将详细探讨如何开发 Vue.js…...

Python并发编程挑战与解决方案
Python并发编程挑战与解决方案 并发编程是现代软件开发中的一项核心能力,它允许多个任务同时运行,提高程序的性能和响应速度。Python因其易用性和灵活性而广受欢迎,但其全局解释器锁(GIL)以及其他特性给并发编程带来了…...

LeetCode从入门到超凡(五)深入浅出---位运算
引言 大家好,我是GISer Liu😁,一名热爱AI技术的GIS开发者。本系列文章是我跟随DataWhale 2024年9月学习赛的LeetCode学习总结文档;本文主要讲解 位运算算法。💕💕😊 一、 位运算简介 1.什么是位…...

一些 Go Web 开发笔记
原文:Julia Evans - 2024.09.27 在过去的几周里,我花了很多时间在用 Go 开发一个网站,虽然不知道它最终会不会发布,但在这个过程中我学到了一些东西,想记录下来。以下是我的一些收获: Go 1.22 现在有了更…...

[Go语言快速上手]初识Go语言
目录 一、什么是Go语言 二、第一段Go程序 1、Go语言结构 注意 2、Go基础语法 关键字 运算符优先级 三、Go语言数据类型 示例 小结 一、什么是Go语言 Go语言,通常被称为Golang,是一种静态类型、编译型的计算机编程语言。它由Google的Robert Gr…...

基于STM32的智能风扇控制系统设计
引言 本项目将基于STM32微控制器设计一个智能风扇控制系统,通过温度传感器实时检测环境温度,并根据预设的温度范围自动调节风扇的转速。该系统展示了STM32的PWM输出、传感器接口以及自动控制应用的实现。 环境准备 1. 硬件设备 STM32F103C8T6 开发板…...

OpenCV 形态学相关函数详解及用法示例
OpenCV形态学相关的运算包含腐蚀(MORPH_ERODE),膨胀(MORPH_DILATE),开运算(MORPH_OPEN),闭运算(MORPH_CLOSE),梯度运算(MORPH_GRADIENT),顶帽运算(MORPH_TOPHAT),黑帽运算(MORPH_BLACKHAT),击中…...

Kafka学习笔记(三)Kafka分区和副本机制、自定义分区、消费者指定分区
文章目录 前言7 分区和副本机制7.1 生产者分区写入策略7.1.1 轮询分区策略7.1.2 随机分区策略7.1.3 按key分区分配策略7.1.4 自定义分区策略7.1.4.1 实现Partitioner接口7.1.4.2 实现分区逻辑7.1.4.3 配置使用自定义分区器7.1.4.4 分区测试 7.2 消费者分区分配策略7.2.1 RangeA…...

第19节 Node.js Express 框架
Express 是一个为Node.js设计的web开发框架,它基于nodejs平台。 Express 简介 Express是一个简洁而灵活的node.js Web应用框架, 提供了一系列强大特性帮助你创建各种Web应用,和丰富的HTTP工具。 使用Express可以快速地搭建一个完整功能的网站。 Expre…...

地震勘探——干扰波识别、井中地震时距曲线特点
目录 干扰波识别反射波地震勘探的干扰波 井中地震时距曲线特点 干扰波识别 有效波:可以用来解决所提出的地质任务的波;干扰波:所有妨碍辨认、追踪有效波的其他波。 地震勘探中,有效波和干扰波是相对的。例如,在反射波…...

【JVM】- 内存结构
引言 JVM:Java Virtual Machine 定义:Java虚拟机,Java二进制字节码的运行环境好处: 一次编写,到处运行自动内存管理,垃圾回收的功能数组下标越界检查(会抛异常,不会覆盖到其他代码…...
使用van-uploader 的UI组件,结合vue2如何实现图片上传组件的封装
以下是基于 vant-ui(适配 Vue2 版本 )实现截图中照片上传预览、删除功能,并封装成可复用组件的完整代码,包含样式和逻辑实现,可直接在 Vue2 项目中使用: 1. 封装的图片上传组件 ImageUploader.vue <te…...
Rapidio门铃消息FIFO溢出机制
关于RapidIO门铃消息FIFO的溢出机制及其与中断抖动的关系,以下是深入解析: 门铃FIFO溢出的本质 在RapidIO系统中,门铃消息FIFO是硬件控制器内部的缓冲区,用于临时存储接收到的门铃消息(Doorbell Message)。…...

学校时钟系统,标准考场时钟系统,AI亮相2025高考,赛思时钟系统为教育公平筑起“精准防线”
2025年#高考 将在近日拉开帷幕,#AI 监考一度冲上热搜。当AI深度融入高考,#时间同步 不再是辅助功能,而是决定AI监考系统成败的“生命线”。 AI亮相2025高考,40种异常行为0.5秒精准识别 2025年高考即将拉开帷幕,江西、…...
LangChain知识库管理后端接口:数据库操作详解—— 构建本地知识库系统的基础《二》
这段 Python 代码是一个完整的 知识库数据库操作模块,用于对本地知识库系统中的知识库进行增删改查(CRUD)操作。它基于 SQLAlchemy ORM 框架 和一个自定义的装饰器 with_session 实现数据库会话管理。 📘 一、整体功能概述 该模块…...
JavaScript基础-API 和 Web API
在学习JavaScript的过程中,理解API(应用程序接口)和Web API的概念及其应用是非常重要的。这些工具极大地扩展了JavaScript的功能,使得开发者能够创建出功能丰富、交互性强的Web应用程序。本文将深入探讨JavaScript中的API与Web AP…...

Linux nano命令的基本使用
参考资料 GNU nanoを使いこなすnano基础 目录 一. 简介二. 文件打开2.1 普通方式打开文件2.2 只读方式打开文件 三. 文件查看3.1 打开文件时,显示行号3.2 翻页查看 四. 文件编辑4.1 Ctrl K 复制 和 Ctrl U 粘贴4.2 Alt/Esc U 撤回 五. 文件保存与退出5.1 Ctrl …...
NPOI操作EXCEL文件 ——CAD C# 二次开发
缺点:dll.版本容易加载错误。CAD加载插件时,没有加载所有类库。插件运行过程中用到某个类库,会从CAD的安装目录找,找不到就报错了。 【方案2】让CAD在加载过程中把类库加载到内存 【方案3】是发现缺少了哪个库,就用插件程序加载进…...