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

Runloop

假设你的项目中有关tableView,然后还有一个定时器timer在执行,定时器代码如下:

    var num = 0override func viewDidLoad() {super.viewDidLoad()let timer = Timer(timeInterval: 1,target: self,selector: #selector(self.run),userInfo: nil,repeats: true)RunLoop.current.add(timer, forMode: .default)}@objc func run() {num += 1print(Thread.current ,num)Thread.sleep(forTimeInterval: 3)}

当你滚动tableView时,你会发现timer定时器会停止,当停止滚动tableView,timer定时器会继续进行。这是什么原因呢?

如果学过线程,你会知道:主线程的优先级是最高的,主线程也叫UI线程,也就是说UI事件必须在主线程中进行,因此在iOS中,UI事件会优先进行。

什么是Runloop?

Runloop是一种事件循环机制,用来循环的处理线程中的事件,当Runloop启动时,如果线程中有事件,它就会调用方法来处理该事件,如果没有事件,就会进入休眠,等该线程中有事件需要处理时才会被唤醒。

线程和Runloop的关系

线程和Runloop是一一对应的,它们的关系保存在一个全局的Dictionary中,其中线程是Key,Runloop是Value,对于主线程中的Runloop是在程序启动时自动创建并启动,而其它子线程的Runloop是懒加载的,也就是说,只有子线程有任务需要被执行时Runloop才会创建并启动。

Runloop的功能

Runloop的功能:

  • 线程保活,Runloop会在线程中没有任务时让线程进入休眠而不会退出,比如保活主线程,使得应用程序不会退出。
  • 负责监听事件,如网络事件,计时器事件,触摸事件
  • 渲染UI,一次Runloop循环,需要渲染屏幕上所有变化的像素点
  • 节省CPU开销,让程序该休息时休息

Runloop的基本组成

输入源

输入源(Input Source)用于处理异步事件,比如用户交互事件,网络事件等,Runloop运行时,会检查当前线程是否有输入源需要处理,如果有会立即处理。

输入源也可以分为两种source:source0和source1,其中source1基于port的系统内核事件,可以主动唤醒runloop,在底层是通过Dictionary的Value来存储的,其中Key是machport。而source0是不基于port的,是应用程序主动触发的事件(如:按钮点击触发事件),不会主动唤醒runloop。

如:当我们点击按钮时,触发的事件会被包装成Event,Event通过machport机制告诉source1,source1主动唤醒runloop,然后将事件Event分发给Source0,然后由Source0来处理。

定时器源

定时执行的任务,如NSTimer,Runloop会定期检测定时器,到达时间执行相应操作。

观察者

用于监听Runloop的状态,基本状态如下:

kCFRunloopEntry (runloop准备启动)
kCFRunloopBeforeTimers (通知观察者,runloop将要对Timer的一些相关事件进行处理了)
kCFRunloopBeforeSources (将要处理一些Sources事件)
kCFRunloopBeforeWaiting( 即将要发生用户态到内核态的切换 用户态 —> 内核态)没事做进入内核态避免资源浪费,即:即将进入休眠。
kCFRunloopAfterWaiting (内核态—转—>用户态)即将被唤醒
kCFRunloopExit (runloop退出通知)

Runloop的几种模式

Runloop主要分为以下几种模式:

NSDefaultRunloopMode:默认模式,处理大多数应用事件,比如Timer定时器事件,网络事件等。

NSRunloopCommonModes:常用模式,允许 RunLoop 同时处理多种事件类型。使用场景包括当用户滚动 UIScrollView 时仍能处理定时器事件。

UITrackingRunloopMode:界面追踪模式,处理UI事件,比如滚动tableView时,保证不受其它Mode影响。

UIInitializationRunLoopMode :在刚启动App时第进入的第一个Mode,启动完成后就不再使用

GSEventReceiveRunLoopMode :接受系统事件的内部Mode

这里可以发现,上面我们的timer添加在default模式下,而tableView的滚动是在Tracking模式下的,所以当tableView滚动时不受其它Mode影响,所以timer定时器事件停止执行,而停止滚动tableView,Runloop切换到default模式,正常处理Timer事件。

如果timer添加在common模式下,timer事件不会受tabelView滚动影响,因为tableView滚动时,在Tracking模式下,可以让标记为"common"的事件继续进行。

如果timer添加在Tracking模式下,当滚动tableView时,timer方法执行,当tableView滚动停止,方法不执行。

阻塞

在上面的讲解中,当定时器设置成common模式,UI事件(tableView滚动)进行时,timer事件也会进行,但它们只是看上去是同时进行,实际上无论Mode怎么变化,它始终都是在一个线程上循环往复。

在iOS开发中,有一个非常重要的原则:永远不能阻塞主线程。因此你的timer事件中不能出现耗时操作,比如改成这样,你会发现你的UI界面会卡顿:

@objc func run() {num += 1print(Thread.current ,num)Thread.sleep(forTimeInterval: 3)//让当前线程休眠
}

Runloop实现的功能

1.自动释放池

自动释放池(AutoreleasePool)的创建、释放、销毁时机如下:

  • kCFRunLoopEntry; // 进入runloop之前,创建一个自动释放池
  • kCFRunLoopBeforeWaiting; // 休眠之前,销毁自动释放池,创建一个新的自动释放池
  • kCFRunLoopExit; // 退出runloop之前,销毁自动释放池

2.硬件事件响应

当一个硬件事件(触摸/锁屏/摇晃等)发生后,首先由 IOKit.framework 生成一个 IOHIDEvent 事件并由 SpringBoard 接收。SpringBoard是iOS的系统进程,接收按键(锁屏/静音等)。

然后,SpringBoard 通过 mach_port 将事件传递给目标 App 的进程。此时App 进程中的 RunLoop 会监听来自系统的这些事件。当事件到达时,RunLoop 中注册的 Source1 事件源被触发,唤醒 RunLoop,并执行回调来处理这些事件。

事件进入应用进程后, _UIApplicationHandleEventQueue() 会把 IOHIDEvent 处理并包装成 UIEvent 进行处理或分发,其中包括识别 UIGesture/处理屏幕旋转/发送给 UIWindow 等。通常事件比如 UIButton 点击、touchesBegin/Move/End/Cancel 事件都是在这个回调中完成的。

3.手势识别

当上面的 _UIApplicationHandleEventQueue() 识别了一个手势时,其首先会调用 Cancel 将当前的 touchesBegin/Move/End 系列回调打断。随后系统将对应的 UIGestureRecognizer 标记为待处理。苹果注册了一个 Runloop的Observer 监测 BeforeWaiting (Loop即将进入休眠) 事件,这个Observer的回调函数是 _UIGestureRecognizerUpdateObserver(),其内部会获取所有刚被标记为待处理的 GestureRecognizer,并执行GestureRecognizer的回调。当有 UIGestureRecognizer 的变化(创建/销毁/状态改变)时,这个回调都会进行相应处理。

这里可以发现:手势识别是在Runloop即将进入休眠的时候进行处理的,这样保证了手势识别的优先级比较高。

4.界面更新

 当在操作 UI 时,比如改变了 Frame、更新了 UIView/CALayer 的层次时,或者手动调用了 UIView/CALayer 的 setNeedsLayout/setNeedsDisplay方法后,这个 UIView/CALayer 就被标记为待处理,并被提交到一个全局的容器去。苹果注册了一个 Runloop的Observer 监听BeforeWaiting(即将进入休眠) 和 Exit (即将退出Loop) 事件,回调去执行一个很长的函数:_ZN2CA11Transaction17observer_callbackEP19__CFRunLoopObservermPv()。这个函数里会遍历所有待处理的 UIView/CAlayer 以执行实际的绘制和调整,并更新 UI 界面。

界面更新是在Runloop进入休眠前进行的,这样可以避免用户交互时立即刷新页面。

5.定时器

• NSTimer 是基于 RunLoop 的定时器,底层实现为 CFRunLoopTimerRef。当你创建一个 NSTimer 并将其添加到 RunLoop 中时,RunLoop 会为定时器注册好未来的触发时间点。

• 定时器回调并不总是会在非常准确的时间点触发,因为 RunLoop 会对定时器进行一些优化。定时器有一个 tolerance 参数,允许指定触发时间的最大误差。这样可以在保证性能的前提下,避免系统资源的浪费。

如果在某个时间点执行了一个较长的任务,RunLoop 可能会错过该时间点,直接跳到下一个时间点,而不会延后执行。例如,如果 NSTimer 设置为每 10 秒触发一次,但是某次因为执行长时间任务导致错过了 10:00 的触发点,那么定时器会直接在下一个 10:10 的时间点触发。

监听Runloop

如果我们想知道Runloop何时开始,何时结束,可以通过Observer监听Runloop.

如果你只想监听Runloop的某一个状态(比如即将进入休眠),可以如下创建:

       // 获取当前的 Runlooplet runloop = CFRunLoopGetCurrent()// 需要监听 Runloop 的哪个状态let activities = CFRunLoopActivity.beforeWaiting.rawValue// 创建 Runloop 观察者let observer = CFRunLoopObserverCreateWithHandler(nil, activities, true, 0) { [weak self] (ob, ac) in。。。。//这里写监听到Runloop某种状态后执行的操作}}// 注册 Runloop 观察者CFRunLoopAddObserver(runloop, observer, .defaultMode)

如果要监听所有状态:

enum RunloopError: Error {case canNotCreate
}      do {let block = { (ob: CFRunLoopObserver?, ac: CFRunLoopActivity) inif ac == .entry {print("进入 Runloop")}else if ac == .beforeTimers {print("即将处理 Timer 事件")}else if ac == .beforeSources {print("即将处理 Source 事件")}else if ac == .beforeWaiting {print("Runloop 即将休眠")}else if ac == .afterWaiting {print("Runloop 被唤醒")}else if ac == .exit {print("退出 Runloop")}}let ob = try createRunloopObserver(block: block)CFRunLoopAddObserver(CFRunLoopGetCurrent(), ob, .defaultMode)}catch RunloopError.canNotCreate {print("runloop 观察者创建失败")}catch {}
}fileprivate func createRunloopObserver(block: @escaping (CFRunLoopObserver?, CFRunLoopActivity) -> Void) throws -> CFRunLoopObserver {
//创建监听所有状态的观察者let ob = CFRunLoopObserverCreateWithHandler(kCFAllocatorDefault, CFRunLoopActivity.allActivities.rawValue, true, 0, block)guard let observer = ob else {throw RunloopError.canNotCreate}return observer}
}

Runloop的应用

线程保活

当子线程执行完任务后,子线程会被销毁,而频繁的线程创建和销毁会导致资源的浪费,所有我们在然后在开启子线程后保证子线程永远活着,这时候就需要常驻线程,即:给线程开启一个Runloop。

优化卡顿

在日常开发中,如果我们要加载一个tableView,一般分两部分进行处理:

1.在子线程中处理cell中呈现的数据资源,其中包括网络请求、数据解析、构造模型等等。

2.模型数组准备完毕,在主线程中刷新tableView,使用模型数据填充cell。

这个思路看似很完美,其实还是会有问题,就是如果第二步,也就是使用模型数据填充cell时十分耗时,怎么办?

简单来说,就是:前面讲Runloop功能时,有一点是渲染UI,假设我们在滑动tableView,这时会触发屏幕上的UI变化,而UI变化会触发Cell的复用和渲染,但其实,Cell的渲染是一个是否耗时的操作,这样就会导致Runloop循环一次的时间变长,造成UI卡顿。这个问题该如何优化呢?

解决思路:将Cell的渲染操作剥离出来。

具体过程如下:

1.用一个block数组,存放渲染Cell的代码

2.在cellForRowAtIndexPath代理方法中直接返回cell。

3.监听Runloop,在即将进入休眠阶段取出block数组中的代码进行执行。

简单来说,就是利用runloop的休眠时间来处理Cell的渲染操作。

参考:

https://github.com/miaoqiu/RunLoop?tab=readme-ov-file#按照官方文档source的分类

https://github.com/tianziyao/Runloop

iOS--RunLoop原理_ios 定时器与runloop-CSDN博客

Runloop解析_objective-c runloop-CSDN博客

相关文章:

Runloop

假设你的项目中有关tableView,然后还有一个定时器timer在执行,定时器代码如下: var num 0override func viewDidLoad() {super.viewDidLoad()let timer Timer(timeInterval: 1,target: self,selector: #selector(self.run),userInfo: nil,r…...

代码随想录第40天

121. 买卖股票的最佳时机 class Solution:def maxProfit(self, prices: List[int]) -> int:cost, profit float(inf), 0for price in prices:cost min(cost, price)profit max(profit, price - cost)return profit122.买卖股票的最佳时机II class Solution:def maxPr…...

element plus table组件多选获取数据id

首先给table加上 selection-change"handleSelectionChange"事件 示例 <el-table selection-change"handleSelectionChange" stripe:data"data?.slice((currentPage3 - 1) * pageSize3, currentPage3 * pageSize3)" style"width: 100%…...

自动驾驶:百年演进

亲爱的小伙伴们&#x1f618;&#xff0c;在求知的漫漫旅途中&#xff0c;若你对深度学习的奥秘、JAVA 、PYTHON与SAP 的奇妙世界&#xff0c;亦或是读研论文的撰写攻略有所探寻&#x1f9d0;&#xff0c;那不妨给我一个小小的关注吧&#x1f970;。我会精心筹备&#xff0c;在…...

STM32 了解OLED

内容扩展 调试方式串口调试&#xff1a;通过串口调试&#xff0c;将调试信息发送到电脑端&#xff0c;电脑使用串口助手显示调试信息 显示屏调试&#xff1a;直接将显示屏连接到单片机&#xff0c;将调试信息打印到显示屏上 keil调试模式&#xff1a;借助Keil软件的调试模式&a…...

NanoLog起步笔记-7-log解压过程初探

nonolog起步笔记-6-log解压过程初探 再看解压过程建立调试工程修改makefile添加新的launch项 注&#xff1a;重新学习nanolog的README.mdPost-Execution Log Decompressor 下面我们尝试了解&#xff0c;解压的过程&#xff0c;是如何得到文件头部的meta信息的。 再看解压过程 …...

python连接redis详细步骤

1.下载并安装windows python Window 平台安装 Python: 以下为在 Window 平台上安装 Python 的简单步骤。 打开 WEB 浏览器访问 Python Releases for Windows | Python.org &#xff1a; 记得勾选 Add Python 3.6 to PATH。 在cmd 执行pip install redis 2.编辑python代码…...

使用redis 的stream 做消息中间件 多线程消费消息

1.redis stream 特点 1.支持消息持久化 2.消费者组模式 3.消息确认机制 4. 消息重试机制 5. 死信队列2. 消息生产者服务 2.1 如下代码Service Slf4j public class StreamMessageProducer {Autowiredprivate StringRedisTemplate redisTemplate;private static final String S…...

《操作系统 - 清华大学》6 -5:局部页面置换算法:最不常用置换算法 (LFU, Least Frequently Used)

文章目录 1. 最不常用算法的工作原理2.最不常用算法特征3. 示例 1. 最不常用算法的工作原理 最不常用算法&#xff1a;注意并不是表示算法本身不常用&#xff0c;而是采取最不常使用页面的策略&#xff0c;Least Frequently Used&#xff0c; LFU。LRU 是最久未被访问的页&…...

GWAS分析先做后学

大家好&#xff0c;我是邓飞。 GWAS分析是生物信息和统计学的交叉学科&#xff0c;上可以学习编程&#xff0c;下可以学习统计。对于Linux系统&#xff0c;R语言&#xff0c;作图&#xff0c;统计学&#xff0c;机器学习等方向&#xff0c;都是一个极好的入门项目。生物信息如…...

【threejs】创建FPS相机

原理说明 控制器是一个很麻烦的东西&#xff0c;需要创建更多的类来管理相机行为&#xff0c;并且可自定义性差&#xff0c;所以将部分控制器的功能绑定到相机上&#xff0c;可以解决这些问题&#xff0c;所以我以 FlyControls为例&#xff0c;将控制器功能绑定到相机上&#…...

路径规划之启发式算法之十一:布谷鸟搜索算法(Cuckoo Search,CS)

布谷鸟搜索算法&#xff08;Cuckoo Search&#xff0c;CS&#xff09;是一种新兴的自然启发式算法&#xff0c;由剑桥大学的杨新社教授和S.戴布&#xff08;Xin-She Yang和Suash Deb&#xff09;于2009年提出。该算法基于布谷鸟的寄生性育雏&#xff08;巢寄生&#xff09;行为…...

Mitel MiCollab企业协作平台存在任意文件读取漏洞(CVE-2024-41713)

免责声明: 本文旨在提供有关特定漏洞的深入信息,帮助用户充分了解潜在的安全风险。发布此信息的目的在于提升网络安全意识和推动技术进步,未经授权访问系统、网络或应用程序,可能会导致法律责任或严重后果。因此,作者不对读者基于本文内容所采取的任何行为承担责任。读者在…...

java+springboot+mysql党务(党员)管理系统

项目介绍&#xff1a; 使用javaspringbootmysql开发的党务管理系统&#xff0c;系统包含管理员、用户角色&#xff0c;功能如下&#xff1a; 管理员&#xff1a;用户管理&#xff1b;党支部管理&#xff1b;党员管理&#xff08;入党申请、积极分子、发展对象、预备党员、正式…...

gozero项目迁移与新服务器环境配置,包含服务器安装包括go版本,Nginx,项目配置包括Mysql,redis,rabbit,域名

迁移 **GoZero** 项目到新服务器并配置相关环境涉及多个步骤。以下是一个系统化的指南&#xff0c;涵盖服务器环境安装、数据库和缓存配置、项目部署以及域名绑定。 ### 步骤概述 1. **服务器环境配置** - 安装 Go 语言环境 - 安装 Nginx - 安装 MySQL 和 Redis -…...

使用WebDAV来上传和下载文件

WebDAV是什么 基于Web的分布式编写和版本控制&#xff08;WebDAV&#xff09;是超文本传输协议&#xff08;HTTP&#xff09;的扩展&#xff0c;有利于用户间协同编辑和管理存储在万维网服务器文档。WebDAV 由互联网工程任务组的工作组在 RFC 4918 中定义。许多现代操作系统为 …...

集成运算放大电路反馈判断

集成运算放大电路 一种具有很高放大倍数的多级直接耦合放大电路&#xff0c;因最初用于信号运算而得名&#xff0c;简称集成运放或运放 模拟集成电路中的典型组件&#xff0c;是发展最快、品种最多、应用最广的一种 反馈 将放大电路输出信号的一部分或全部通过某种电路引回到输…...

什么是绩效文化?

绩效文化是一种组织文化&#xff0c;它将绩效视为核心价值观&#xff0c;贯穿于组织的各个层面和活动之中。 一、绩效文化的内涵 目标导向 绩效文化强调组织成员都朝着共同的目标努力。这个目标通常是明确、可衡量的&#xff0c;如企业的年度利润目标、市场份额增长目标等。例…...

【力扣】409.最长回文串

问题描述 思路解析 因为同时包含大小写字母&#xff0c;直接创建个ASCII表大小的桶来标记又因为是要回文子串&#xff0c;所以偶数个数的一定可以那么同时&#xff0c;对于出现奇数次数的&#xff0c;我没需要他们的次数-1&#xff0c;变为偶数&#xff0c;并且可以标记出现过…...

android studio创建虚拟机注意事项

emulator 启动模拟器的时候&#xff0c;可以用 AVD 界面&#xff0c;也可以用命令行启动&#xff0c;但命令行启 动的时候要注意&#xff0c;系统有两个 emulator.exe &#xff0c;建议使用 emulator 目录下的那个&#xff01;&#xff01; 创建类型为google APIs的虚拟机可从…...

DP协议:缩略词

缩写代表的含义ACT分配更改触发&#xff08;Allocation Change Trigger&#xff09;API应用程序编程接口&#xff08;Application Programming Interface&#xff09;AUX辅助&#xff08;Auxiliary&#xff09;BER比特错误率&#xff08;Bit Error Rate&#xff09;bpc每色比特…...

【Rive】事件回调

1 前言 Android 中可以通过 RiveAnimationView 的 addEventListener 方法添加动画监听器&#xff0c;用于监听状态动画和过渡动画的开始和结束时机&#xff0c;实现动画开始和结束时的事件回调&#xff1b;也可以监听 Rive 事件触发的时机&#xff0c;在事件触发时响应回调。 …...

YOLOv8-ultralytics-8.2.103部分代码阅读笔记-tuner.py

tuner.py ultralytics\engine\tuner.py 目录 tuner.py 1.所需的库和模块 2.class Tuner: 1.所需的库和模块 # Ultralytics YOLO &#x1f680;, AGPL-3.0 license# 模块提供用于对象检测、实例分割、图像分类、姿势估计和多对象跟踪的 Ultralytics YOLO 模型的超参数调整…...

【数据结构】堆的概念、结构、模拟实现以及应用

本篇我们来介绍一下数据结构中的堆。 1.堆的概念及结构 1&#xff09;堆是一颗完全二叉树。 2&#xff09;堆又分为大堆和小堆&#xff0c;大堆就是树里面任何一个父节点都大于子节点&#xff0c;所以根节点是最大值&#xff1b;小堆就是树里面任何一个父节点都小于子节点&am…...

推送(push)项目到gitlab

文章目录 1、git init1.1、在当前目录中显示隐藏文件&#xff1a;1.2、查看已有的远程仓库1.3、确保你的本地机器已经生成了 SSH 密钥&#xff1a;1.4、将生成的公钥文件&#xff08;通常位于 ~/.ssh/id_rsa.pub&#xff09;复制到 GitLab 的 SSH 设置中&#xff1a;1.5、测试 …...

springboot/ssm宠物商城网站系统Java代码web项目宠物用品购物论坛源码

springboot/ssm宠物商城网站系统Java代码web项目宠物用品购物论坛源码 基于springboot(可改ssm)htmlvue项目 开发语言&#xff1a;Java 框架&#xff1a;springboot/可改ssm vue JDK版本&#xff1a;JDK1.8&#xff08;或11&#xff09; 服务器&#xff1a;tomcat 数据库&…...

前端基础的讲解-JS(22)

什么是JSON&#xff1f; 1.json 是一种轻量级的数据交换格式 简单来说&#xff1a;json 就是一种在各个编程语言中流通的数据格式&#xff0c;负责不同编程语言中的数据传递和交互。 类似于&#xff1a; 国际通用语言 - 英语 中国 56 个民族不同地区的通用语言 - 普通话 …...

zerotier实现内网穿透(访问内网服务器)

moo 内网穿透工具 实用工具&#xff1a;zerotier 目录 内网穿透工具 Windows下zerotier安装 ubuntu系统下的zerotier安装 使用moon加速 Windows下zerotier安装 有了网络之后&#xff0c;会给你一个网络id&#xff0c;这个网络id是非常重要的&#xff0c;其它设备要加入…...

python语言中怎么调用不同级文件夹中数据文件

python语言中怎么调用文件夹中数据文件 python 怎么调用同一级文件夹中数据1. **读取同一级文件夹中的数据文件&#xff08;如 .txt, .csv, .json 等&#xff09;**示例&#xff1a; 2. **导入同一级文件夹中的 Python 模块**3. **使用相对路径导入模块**4. **使用 os.path 或 …...

spring事务源码解析

1 引入 在企业级应用开发中&#xff0c;事务管理 是确保数据一致性和完整性的重要手段。而在 Spring 框架中&#xff0c;事务管理提供了高度抽象和灵活的实现&#xff0c;开发者只需通过简单的注解或配置即可轻松实现复杂的事务逻辑。然而&#xff0c;Spring 事务背后的实现机…...

【每日刷题】Day165

【每日刷题】Day165 &#x1f955;个人主页&#xff1a;开敲&#x1f349; &#x1f525;所属专栏&#xff1a;每日刷题&#x1f34d; &#x1f33c;文章目录&#x1f33c; 1. LCR 092. 将字符串翻转到单调递增 - 力扣&#xff08;LeetCode&#xff09; 2. 424. 替换后的最长…...

基于51单片机的智能公交车报站系统GPS定位语音播报智能安全检测人数统计

功能描述 1.LCD12864可显示当前年月日&#xff0c;星期&#xff0c;时间&#xff0c; 当前站名&#xff0c;经纬度&#xff0c;是否连接GPS&#xff0c;自动/手动模式&#xff0c; 2.自带GPS定位&#xff0c;可实时显示经纬度&#xff1b; 3.通过DS1302时钟芯片&#xff0c;获…...

计算机网络安全 —— 实体鉴别与生成大随机数

一、实体鉴别# ​ 实体鉴别&#xff08;经常简称为鉴别&#xff09;就是一方验证另一方身份的技术。一个实体可以是人、客户/服务器进程等。这里仅讨论如何鉴别通信对端 实体的身份&#xff0c;即验证正在通信的对方确实是所认为的通信实体&#xff0c;而不是其他的假冒者。进…...

Vue3+Pinia 状态管理持久化

一、Pinia 简介 &#x1f396;️Pinia 起始于 2019 年 11 月左右的一次实验&#xff0c;其目的是设计一个拥有组合式 API 的 Vue 状态管理库。Vue3VitePinia 新三剑客逐渐替代了Vue2WebpackVuex 了&#xff0c;性能啥的各方面吊打。 1.1 什么是状态管理&#xff1f; &#x1f…...

开源项目:轻型图像分割 unet_lite

DataBall 助力快速掌握数据集的信息和使用方式&#xff0c;会员享有 百种数据集&#xff0c;持续增加中。 需要更多数据资源和技术解决方案&#xff0c;知识星球&#xff1a; “DataBall - X 数据球(free)” -------------------------------------------------------------…...

C# 向上取整多种实现方法

1.使用 Math.Ceiling 方法&#xff1a; 在 C# 中&#xff0c;可以利用 System.Math 类下的 Math.Ceiling 方法来实现向上取整。它接受一个 double 或 decimal 类型的参数&#xff0c;并返回大于或等于该参数的最小整数&#xff08;以 double 或 decimal 类型表示&#xff09;。…...

Linux 权限及管理

目录 一、Linux权限 1、概念 2、超级用户和普通用户的相关操作 a. 添加用户&#xff0c;删除用户 b. 超级用户和普通用户的切换 c. sduo提权以及白名单设置 二、Linux权限管理 1、文件访问者的分类 2、文件访问类型和权限 a. 文件类型 b. 基本权限 3、文件权限值…...

【JVM】JVM基础教程(一)

目录 初识JVM JVM是什么&#xff1f; JVM的功能 解释、即时编译和运行 内存管理 常见的JVM JVM虚拟机规范 HotSpot的发展历程 JVM的组成 字节码文件详解 应用场景 以正确姿势打开字节码文件 ​编辑字节码文件的组成 基本信息 Magic魔数 主副版本号 常量池 接口…...

企业国内外网络互联方案全解析

面对国内市场日益饱和的现状&#xff0c;企业纷纷将目光投向海外&#xff0c;而实现国内外网络的高效互联&#xff0c;则成为支撑其跨国业务顺利运行的关键。本文将为您详细介绍几种实现国内外网络互联的有效策略&#xff0c;助您轻松应对全球化挑战。 有些企业选择使用虚拟专用…...

【优选算法 位运算】位运算算法入门详解:位运算小专题

判定字符是否唯一 题目解析 算法原理 解法一 &#xff1a;哈希数组 从前往后扫描字符串&#xff0c;把扫描到的字符先进行判断&#xff0c;如果对应的 val 0 &#xff0c;则放入哈希表中&#xff0c;否则返回 false&#xff0c;知道扫描完整个字符&#xff1b;时间…...

大文件分块上传后端服务器

一、背景&#xff1a; 后台系统需要上传大文件、大视频等数据&#xff0c;耗时过长&#xff0c;接口等待超时&#xff0c;故需优化通过前端多线程分片方式进行文件上传&#xff0c;显著提升上传速度。 二、流程&#xff1a; 前端逻辑&#xff1a; 前端使用分片技术&#xff…...

perl Window安装教程

perl Window安装教程 下载地址 https://platform.activestate.com/tangxing806/ActivePerl-5.28/distributions 运行state-remote-installer.exe 按下图截图步骤 检查perl版本 参考文献&#xff1a; perl安装教程...

Scrcpy投影之后为什么声音在电脑端显示?

关于安卓设备和电脑端扬声器优先级 在使用安卓设备与电脑进行某些连接操作&#xff08;比如通过 adb 相关工具交互时&#xff09;&#xff0c;确实存在音频输出的优先级选择情况。通常情况下&#xff0c;可能默认音频会输出到电脑端&#xff08;比如通过投屏等相关操作连接后&…...

2025年山东省职业院校技能大赛“信息安全管理与评估”(山东省) 任务书

2025年山东省职业院校技能大赛“信息安全管理与评估”(山东省 任务书 模块一网络平台搭建与设备安全防护任务1&#xff1a;网络平台搭建 &#xff08;50分&#xff09;任务2&#xff1a;网络安全设备配置与防护&#xff08;250分&#xff09; 模块二网络安全事件响应、数字取证…...

java+ssm+mysql收纳培训网

项目介绍&#xff1a; 使用javassmmysql开发的收纳视频培训网&#xff0c;系统包含超级管理员&#xff0c;系统管理员、培训师、用户角色&#xff0c;功能如下&#xff1a; 超级管理员&#xff1a;管理员管理&#xff1b;用户管理&#xff08;培训师、用户&#xff09;&#…...

多表查询-概述内连接外连接子查询

一.数据准备&#xff1a; 1.部门表&#xff1a; 代码&#xff1a; -- 部门管理 create table tb_dept (id int unsigned primary key auto_increment comment 主键ID,name varchar(10) not null unique comment 部门名称,create_time datetime not null c…...

H5游戏出海如何获得更多增长机会?

海外H5小游戏的崛起给了国内众多中小厂商出海发展的机会&#xff0c;开发者如何在海外市场获得更多的增长机会&#xff1f;#APP出海# H5游戏如何在海外获得核心用户&#xff1f; HTML5游戏的开发与运营者们首先可以利用量多质高的HTML5游戏&#xff0c;维持海外用户粘性&…...

element plus的表单校验,明明输入内容了,但提示红字还是会显示着

下拉框的不隐藏&#xff0c;可能是 trigger为blur的原因&#xff0c;改为change即可 const rules reactive({name: [{ required: true, message: "请输入名称", trigger: "blur" }],price: [{ required: true, message: "请输入价格", trigger…...

MobaXterm Sessions 批量录入导入,会话批量添加,解决导入配置中文乱码

一、创建表格 创建 Excel 表格&#xff0c;将服务器信息写入表格 二、写入文件 新建 list.txt 文件将表格中的服务器信息复制粘贴进去 三、修改脚本 这是你需要修改的变量&#xff0c;其他变量不需要动 # 登录用户 ssh_userroot # 目录名称 folder_name资源池四、执行脚本 …...

Vue项目中的权限控制实践与方案详解

在现代前端开发中&#xff0c;权限控制是一个不可或缺的重要环节。一个完善的权限控制系统不仅能够保护应用的安全性&#xff0c;还能为不同角色的用户提供更好的使用体验。让我们深入探讨Vue项目中权限控制的实现方案和最佳实践。 权限控制本质上是对用户操作的一种限制&…...