【iOS】KVOKVC原理
1 KVO 键值监听
1.1 KVO简介
KVO的全称是Key-Value Observing
,俗称"键值监听",可以用于监听摸个对象属性值得改变。
KVO一般通过以下三个步骤使用:
// 1. 添加监听
[self.student1 addObserver:self forKeyPath:@"age" options:options context:nil];// 2. 重写- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context方法- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {NSLog(@"%@的%@被改变:%@", object, keyPath, change);
}// 3. 适当时机移除监听
[self.student1 removeObserver:self forKeyPath:@"age"];
1.2 KVO简单使用
- 建立
SXStudent
类和SXTeacher
类
//SXStudent.h #import <Foundation/Foundation.h>NS_ASSUME_NONNULL_BEGIN@interface SXStudent : NSObject
@property (nonatomic, assign) NSInteger age;
@endNS_ASSUME_NONNULL_END// SXTeacher.h
#import <Foundation/Foundation.h>
#import "SXStudent.h"NS_ASSUME_NONNULL_BEGIN@interface SXTeacher : NSObject
@property (nonatomic, strong) SXStudent *student1;
@property (nonatomic, strong) SXStudent *student2;
- (void)demo;
@endNS_ASSUME_NONNULL_END
- 实现
SXStudent
类。
// SXStudent.m#import "SXStudent.h"
@implementation SXStudent
@end
- 实现
SXTeacher
类,重写init
方法,为SXTeacher
的student1
属性添加监听。实现demo
方法,分别更改student1
和student2
的age
值。
// SXTeacher.m#import "SXTeacher.h"
#import <objc/runtime.h>@implementation SXTeacher- (id)init {if (self = [super init]) {self.student1 = [[SXStudent alloc] init];self.student2 = [[SXStudent alloc] init];self.student1.age = 1;self.student2.age = 2;// 添加监听NSKeyValueObservingOptions options = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;[self.student1 addObserver:self forKeyPath:@"age" options:options context:nil];}return self;
}- (void)demo {self.student1.age = 20;self.student2.age = 30;
}- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {NSLog(@"%@的%@被改变:%@", object, keyPath, change);
}- (void)dealloc {// 移除监听[self.student1 removeObserver:self forKeyPath:@"age"];
}@end
mian
函数内创建SXTeacher
的实例对象并调用demo
方法测试。
#import <Foundation/Foundation.h>
#import "SXTeacher.h"int main(int argc, const char * argv[]) {@autoreleasepool {SXTeacher *teacher = [[SXTeacher alloc] init];[teacher demo];}return 0;
}
- 查看运行结果。
1.3 实现原理探究
1.3.1 student1发生的变化
为什么student1
的setter
方法可以触发监听,添加监听的方法到底对student1
做了什么?
- 我们在添加监听后打一个断点。
- 试着利用lldb调试查看
student1
和student2
的isa
指针。
我们发现student1
的isa
指针的指向被更改成了NSKVONotifying_SXStudent
(NSKVONotifying_为前缀,原类名为后缀)类。
1.3.2 NSKVONotifying_XXX类
- 关于
NSKVONotifying_XXX
类
NSKVONotifying_XXX
类是Runtime
动态创建的一个类,在程序运行的过程中产生的一个新的类。NSKVONotifying_XXX
类是原类的一个子类。NSKVONotifying_XXX
类存在自己的setAge:
、class
、dealloc
、isKVOA
…方法。
试着验证NSKVONotifying_XXX
类的方法和父类,我们可以使用如下代码打印NSKVONotifying_SXStudent
类和SXStudent
类的方法列表和父类类型。
- (void)demo2 {[self printMethods:object_getClass(self.student1)];[self printMethods:object_getClass(self.student2)];
}- (void) printMethods:(Class)cls {unsigned int count;Method *methods = class_copyMethodList(cls, &count);NSMutableString *methodNames = [NSMutableString string];[methodNames appendFormat:@"%@ - ", cls];NSLog(@"%@ superClass ----> %@", NSStringFromClass(cls), NSStringFromClass(class_getSuperclass(cls)));for (int i = 0; i < count; i++) {Method method = methods[i];NSString *methodName = NSStringFromSelector(method_getName(method));[methodNames appendFormat:@"%@ ", methodName];}NSLog(@"%@", methodNames);free(methods);
}
打印结果:
可以看到NSKVONotifying_SXStudent
类有自己的setAge:
、class
、dealloc
、 _isKVOA
方法。
- 重写class方法是为了隐藏NSKVONotifying_XXX类的存在。重写后的class方法返回其父类(原来的类)类型,使用户以为类没有变化。
- _isKVOA则是用来标识当前类是否是通过runtime动态生成的类对象,如果是,就返回YES,不是,则返回NO。
- 当对像被销毁后,dealloc做一些收尾工作。
1.3.3 方法调用探究
由上面可分析出我们的student1
的isa
指针指向的类对象是NSKVONotifying_SXStudent
,并且NSKVONotifying_SXStudent
中还带有setAge:
方法,所以student1
的setAge:
方法走的应该是NSKVONotifying_SXStudent
类中的setAge:
方法。
我们试着使用下面的代码打印student1
被监听前后的setAge:
方法的地址,并使用lldb调试一探究竟。
- (id)init {if (self = [super init]) {self.student1 = [[SXStudent alloc] init];self.student2 = [[SXStudent alloc] init];self.student1.age = 1;self.student2.age = 2;NSLog(@"添加监听之前 - p1 = %p, p2 = %p", [self.student1 methodForSelector:@selector(setAge:)], [self.student2 methodForSelector:@selector(setAge:)]);// 添加监听NSKeyValueObservingOptions options = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;[self.student1 addObserver:self forKeyPath:@"age" options:options context:nil];NSLog(@"添加监听之后 - p1 = %p, p2 = %p", [self.student1 methodForSelector:@selector(setAge:)], [self.student2 methodForSelector:@selector(setAge:)]);}return self;
}
打断点使用lldb打印方法地址对应的方法名:
我们发现student1
的setAge:
方法实际上是调用了Foundation
框架的_NSSetLongLongValueAndNotify
函数。这又是怎么回事,我们先来了解一下这个函数。
1.3.3 _NSSetXXXValueAndNotify
经过查阅资料我们可以了解到。
NSKVONotifyin_XXX
中的setage:
方法中其实调用了Fundation
框架中C语言函数_NSsetXXXValueAndNotify
,_NSsetXXXValueAndNotify
内部做的操作相当于,首先调用willChangeValueForKey
将要改变方法,之后调用原来的setage
方法对成员变量赋值,最后调用didChangeValueForKey
已经改变方法。didChangeValueForKey
中会调用监听器的监听方法,最终来到监听者的observeValueForKeyPath
方法中。
Foundation
框架中会根据属性的类型,调用不同的方法。例如我们之前定义的NSInteger
类型的age
属性,那么我们看到Foundation
框架中调用的_NSsetLongLongValueAndNotify
函数。那么我们把age
的属性类型变为double
重新打印一遍。
我们发现调用的函数变为了_NSSetDoubleValueAndNotify
,那么这说明Foundation
框架中有许多此类型的函数,通过属性的不同类型调用不同的函数。
我们可以重写 SXStudent
类的willChangeValueForKey
方法和didChangeValueForKey
方法来验证上述说法。
#import "SXStudent.h"@implementation SXStudent
- (void)setAge:(NSInteger)age {NSLog(@"setAge");_age = age;
}
- (void)willChangeValueForKey:(NSString *)key {NSLog(@"willChangeValueForKey begin");[super willChangeValueForKey:key];NSLog(@"willChangeValueForKey end");
}- (void)didChangeValueForKey:(NSString *)key {NSLog(@"didChangeValueForKey begin");[super didChangeValueForKey:key];NSLog(@"didChangeValueForKey end");
}
@end
打印结果:
可知:
_NSSetXXXValueAndNotify
调用willChangeValueForKey:
;_NSSetXXXValueAndNotify
调用setter
实现;_NSSetXXXValueAndNotify
调用didChangeValueForKey:
;didChangeValueForKey
内部会调用observe
r的observeValueForKeyPath:ofObject:change:context:
方法。
1.3.4 伪代码
据上所述,可以写出NSKVONotifying_SXStudent类的伪代码:
///> NSKVONotifying_SXStudent.m 文件#import "NSKVONotifying_SXStudent.h"@implementation NSKVONotifying_SXStudent- (void)setAge:(int)age{_NSSetLongLongValueAndNotify(); ///> 文章末尾 知识点补充小结有此方法来源
}void _NSSetLongLongValueAndNotify(){[self willChangeValueForKey:@"age"];[super setAge:age];[self didChangeValueForKey:@"age"];
}- (void)didChangeValueForKey:(NSString *)key{///> 通知监听器 key发生了改变[observe observeValueForKeyPath:key ofObject:self change:nil context:nil];
}@end
2 KVC 键值编码
2.1 KVC简介
KVC的全称key - value - coding,俗称"键值编码",可以通过key来访问某个属性。
常见的API有:
- (void)setValue:(id)value forKey:(NSString *)key;
- (void)setValue:(id)value forKeyPath:(NSString *)keyPath;- (id)valueForKey:(NSString *)key
- (id)valueForKeyPath:(NSString *)keyPath;
2.2 KVC简单使用
2.2.1 自定义SXDog类、SXStudent类和SXTeacher类。
// SXDog.h
#import <Foundation/Foundation.h>NS_ASSUME_NONNULL_BEGIN@interface SXDog : NSObject
@property (nonatomic, assign) CGFloat weight;
@endNS_ASSUME_NONNULL_END// SXStudent.h
#import <Foundation/Foundation.h>
#import "SXDog.h"NS_ASSUME_NONNULL_BEGIN@interface SXStudent : NSObject
@property (nonatomic, assign) NSInteger age;
@property (nonatomic, strong) SXDog *dog;
@endNS_ASSUME_NONNULL_END// SXTeacher.h
#import <Foundation/Foundation.h>
#import "SXStudent.h"NS_ASSUME_NONNULL_BEGIN@interface SXTeacher : NSObject
@property (nonatomic, strong) SXStudent *student1;
@endNS_ASSUME_NONNULL_END
2.2.2 实现这三个类,为了方便使用,重写SXStudent和SXTeacher的初始化方法,在初始化方法里对属性进行初始化。
SXDog.m
#import "SXDog.h"@implementation SXDog
@end
SXStudent.m
#import "SXStudent.h"
#import <objc/runtime.h>@implementation SXStudent
- (id)init {if (self = [super init]) {self.dog = [[SXDog alloc] init];}return self;
}
@end
SXTeacher.m
#import "SXTeacher.h"
#import <objc/runtime.h>@implementation SXTeacher
- (id)init {if (self = [super init]) {self.student1 = [[SXStudent alloc] init];}return self;
}
@end
2.2.3 为SXTeacher添加两个方法,在这两个方法里试调用KVC。
SetValue:ForKey:
与ValueForKey:
方法
- (void)demoSetValueForKeyAndValueForKey {[self.student1 setValue:@20 forKey:@"age"];NSLog(@"点语法:%ld", self.student1.age);NSNumber *value = [self.student1 valueForKey:@"age"];NSLog(@"KVC:%@", value);
}
SetValue:ForKeyPath:
与ValueForKeyPath:
- (void)demoSetValueForKeyPathAndValueForKeyPath {[self.student1 setValue:@16 forKeyPath:@"dog.weight"];NSLog(@"点语法:%lf", self.student1.dog.weight);NSNumber *value = [self.student1 valueForKeyPath:@"dog.weight"];NSLog(@"KVC:%@", value);
}
- 调用上面两个函数,运行。
2.2.4 KeyPath 和 Key 的区别:
keyPath
相当于根据路径去寻找属性,一层一层往下找。key
是直接访问属性的名字,如果按路径找会报错。
2.3 KVC流程
2.3.1 setValue:forkey:赋值流程
- 首先会按照setKey:、_setKey:的顺序到对象的方法列表中寻找这两个方法,如果找到了方法,则传参并且调用方法。
- 如果没有找到方法,则通过accessInstanceVariablesDirectly方法的返回值来决定是否能够查找成员变量。如果accessInstanceVariablesDirectly返回YES,则会按照以下顺序到成员变量列表中查找对应的成员变量:
- _key
- _isKey
- key
- isKey
- 如果accessInstanceVariablesDirectly返回NO,则直接抛出NSUnknownKeyException异常。
如果在成员变量列表中找到对应的属性值,则直接进行赋值,如果找不到,则会抛出NSUnknownKeyException异常。
accessInstanceVariablesDirectly函数
+ (BOOL)accessInstanceVariablesDirectly{return YES; ///> 可以直接访问成员变量// return NO; ///> 不可以直接访问成员变量, ///> 直接访问会报NSUnkonwKeyException错误
}
2.3.2 valueForKey:取值流程
- 首先会按照以下顺序查找方法列表:
- getKey
- key
- isKey
- _key
- 如果找到就直接传递参数,调用方法,如果未找到则查看accessInstanceVariablesDirectly方法的返回值,如果返回NO,则直接抛出NSUnknownKeyException异常。
- 如果accessInstanceVariablesDirectly方法返回YES,则按如下顺序查找成员变量列表:
- _key
- _isKey
- key
- isKey
- 如果能找到对应的成员变量,则直接获取成员变量的值,如果未找到,则抛出NSUnknownKeyException异常。
2.3.3 试验证setValue:forkey:赋值流程
对上述例子进行小修改:
SXStudent.h
@interface SXStudent : NSObject {@publicint _age;int _isAge;int age;int isAge;
}
@end
SXTeacher.m
- (id)init {if (self = [super init]) {self.student1 = [[SXStudent alloc] init];NSKeyValueObservingOptions option = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;[self.student1 addObserver:self forKeyPath:@"age" options:option context:nil];[self.student1 setValue:@20 forKey:@"age"];NSLog(@"-----");}return self;
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {NSLog(@"%@的%@被改变:%@", object, keyPath, change);
}
在NSLog(@"-----");
处打下断点,运行,查看student1
中的成员变量。看看谁被赋值了。
可以看到_age
首先被赋值,我们注释掉SXStudent
中的_age
成员变量,看看下一个是谁被赋值。如此反复,就可以得到setValue:forkey:
赋值流程。结果与上述无误,我就不继续了。
通过本例,我们还可以知道KVC也可以触发KVO监听。
3 一些问题
3.1 iOS用什么方式实现对一个对象的KVO?(KVO的本质是什么?)
给一个实例对象添加KVO,系统内部是利用Runtime
动态的生成一个此实例对象的类对象的子类,具体的格式为_NSKVONotifying_XXX
,并且让实例对象的isa
指针指向这个新生成的类。
重写属性的set
方法,当调用set
方法时,会调用Foundation
框架的NSSetXXXValueAndNotify
函数
在_NSSetXXXValueAndNotify
中会执行以下步骤:
- 调用
willChangeValueForKey:
方法; - 调用父类的
set
方法,重新赋值; - 调用
didChangeValueForKey:
方法; didChangeValueForKey:
内部会触发监听器的observeValueForKeyPath:ofObject:change:context:
方法。
3.2 如何手动触发KVO?
手动调用willChangeValueForKey:
和didChangeValueForKey:
。
例:
- (id)init {if (self = [super init]) {self.student1 = [[SXStudent alloc] init];NSKeyValueObservingOptions option = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;[self.student1 addObserver:self forKeyPath:@"age" options:option context:nil];[self.student1 willChangeValueForKey:@"age"];[self.student1 didChangeValueForKey:@"age"];}return self;
}
运行结果:
虽然是在didChangeValueForKey:
内部会触发监听器的observeValueForKeyPath:ofObject:change:context:
方法,但是如果不调用willChangeValueForKey:
无法就无法触发监听器,这两个必须一起使用。
3.3 直接修改成员变量的值是否会触发KVO?
直接修改成员变量的值不会触发KVO,因为没有触发setter方法。
相关文章:

【iOS】KVOKVC原理
1 KVO 键值监听 1.1 KVO简介 KVO的全称是Key-Value Observing,俗称"键值监听",可以用于监听摸个对象属性值得改变。 KVO一般通过以下三个步骤使用: // 1. 添加监听 [self.student1 addObserver:self forKeyPath:"age"…...

当机器人变硬核:探索深度学习中的时间序列预测
收藏自:Wed, 15 Sep 2021 10:32:56 UTC 摘要:时间序列预测是机器学习和深度学习领域的一个重要应用,它可以用于预测未来趋势、分析数据模式和做出决策。本文将介绍一些基本概念和常用方法,并结合具体的案例,展示如何使…...

C# Solidworks二次开发:自动创建虚拟零件及使用注意事项
今天要讲的是关于在solidworks中如何自动创建虚拟零件的功能,也就是solidworks中插入新零件这个功能。 实现这个功能需要使用的API如下所示: InsertNewVirtualPart(swFaceOrPlane1, out swcomp2); 其中这个方法中使…...
vim工具 windows系统使用
vim常用命令: 编辑–>输入: i: 在当前光标所在字符的前面,转为输入模式; 粘贴命令 p p: 如果删除或复制为整行内容,则粘贴至光标所在行的下方,如果复制或删除的内容为非整行,则粘贴至光标所…...

Tesseract开源的OCR工具及python pytesseract安装使用
一 、介绍 Tesseract是一款由Google赞助的开源OCR。 pytesseract是python包装器,它为可执行文件提供了pythonic API。 Tesseract 已经有 30 年历史,开始它是惠普实验室的一款专利软件,在2005年后由Google接手并进一步开发和完善。Tesseract支…...
【数理知识】自由度 degree of freedom 及自由度的计算方法
放在最前的一句话:自由度是一个存在于两个学科中的概念,一个是存在于统计学中的自由度,另一个是存在于物理学中的自由度。而我本人需要的是研究物理学中的自由度概念,同时本笔记全篇也是在了解物理学中的自由度。 文章目录 自由度…...

苍穹外卖day09——历史订单模块(用户端)+订单管理模块(管理端)
查询历史订单——需求分析与设计 产品原型 业务规则 分页查询历史订单 可以根据订单状态查询 展示订单数据时,需要展示的数据包括:下单时间、订单状态、订单金额、订单明细(商品名称、图片) 接口设计 查询历史订单——代码开…...

正则表达式 —— Grep
文本处理三剑客:Grep、Sed、Awk 这三个工具都是基于对文本的内容进行增删改查的操作,此篇着重介绍grep与正则表达式的应用,以及扩展正则表达式。 正则表达式 什么是正则表达式? 它是由一类特殊字符以及文本字符所编写的一种模式…...

STC12C5A系列单片机片内看门狗的应用
wdt.c #include "wdt.h"void wdt_init(void) {WDT_CONTR 0x24; // 0010 0100 - 1.1377s }void wdt_feed(void) {WDT_CONTR | 0x10; // 喂狗 }wdt.h #ifndef _WDT_H_ #define _WDT_H_#include "stc12c5a60s2.h"// 函数声明 extern void wdt_init(void); …...

C语言指针详解
目录 指针是什么? 指针和指针类型 指针-整数 指针的解引用 野指针 野指针成因 如何规避野指针 指针运算 指针- 整数 指针-指针 指针的关系运算 指针和数组 二级指针 指针数组 指针数组 模拟二维数组 指针是什么? 指针理解的2个要点: 1. 指针是内存中一个…...

RTPS规范v2.5(中文版)
实时发布订阅协议 DDS互操作性有线协议 (DDSI-RTPS) 技术规范 V2.5 (2022-04-01正式发布) https://www.omg.org/spec/DDSI-RTPS/2.5/PDF 目 录 1 范围 8 2 一致性 8 3 规范性参考文献 8 4 术语和定义 9 5 标识 …...

LeetCode102.Binary-Tree-Level-Order-Traversal<二叉树的层序遍历>
题目: 思路: 写过N叉树的层序遍历,(8条消息) LeetCode429.N-Ary-Tree-Level-Order-Traversal<N 叉树的层序遍历>_Eminste的博客-CSDN博客 使用栈保存每一层的结点。然后每次当前层结束。将这一层的值添加进去res中。…...

yolov8系列[五]-项目实战-yolov8模型无人机检测
yolov8系列[五]-项目实战-yolov8模型无人机检测 项目介绍项目展示功能简介代码结构如何启动 开发者模式1. 安装依赖环境2. 启动程序 源代码下载其他 项目介绍 无人机识别项目,无人机搭载nvidia jetson边缘计算板子,进行实时识别。使用yolov8算法,训练了识别无人机的…...

Redis 笔记,基本数据类型、持久化、主从、集群等等问题
标题 😀😀😀创作不易,各位看官点赞收藏. 文章目录 标题Redis 基础笔记1、安装及环境搭建2、Redis 数据类型2.1、String2.2、List2.3、Hash2.4、Set2.5、Zset2.6、BitMap2.7、HyperLogLog2.8、Geospatial2.9、Stream 3、Redis 持久…...
JDK,JRE,JVM三者的关系
JDK(全称 Java Development Kit),Java开发工具包,能独立创建、编译、运行程序。 JDK JRE java开发工具(javac.exe/java.exe/jar.exe) JRE(全称 Java Runtime Environment),能运行…...
行为型-命令模式(Command Pattern)
说明 命令模式(Command Pattern)是一种行为设计模式,它将请求封装为一个对象,以便在不同的请求者和接收者之间进行解耦、参数化和操作的队列化。命令模式允许你将具体的请求封装为对象,这些对象之间彼此独立ÿ…...

总结942
5:40起床 6:00~7:00单词复习300个,记100个 7:15~8:00早读,《love is as strong as death》第一第二段 8:10~9:10三大计算回顾 9:15~10:06 习题880第一章基础选择纠错 10:10~10:30单词默写 10:30~11:40强化第一讲习题 11:40~12:30继续…...

MFC自定义控件使用
用VS2005新建一个MFC项目,添加一个Custom Control控件在窗体 我们需要为自定义控件添加一个类。项目,添加类,MFC类 设置类名字,基类为CWnd,你也可以选择CDialog作为基类 类创建完成后,在它的构造函数中注册一个新的自定义窗体,取名为"MyWindowClass" WNDCL…...
【学习笔记】「ROI 2018 Day 2」无进位加法
先放一个大佬的博客:「loj - 2850」「ROI 2018 Day 2」无进位加法 用数据结构来优化搜索🤔 神一样的 Kidulthood 考场上就已经意识到了这道题的正解是搜索😅 考虑搜索过程的本质🤔 首先是找到最小的满足 t i i t_ii tii最大…...

分布式I/O,IT和OT融合少不了它
长期以来信息技术IT和操作运营技术OT是相互隔离的,随着大数据分析和边缘计算业务的对现场级实时数据的采集需求,IT和OT有了逐渐融合的趋势。IT与OT融合,它赋予工厂的管理者监控运行和过程的能力大为增强,甚至可以预测到可能发生的…...

UE5 学习系列(二)用户操作界面及介绍
这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…...
Vue记事本应用实现教程
文章目录 1. 项目介绍2. 开发环境准备3. 设计应用界面4. 创建Vue实例和数据模型5. 实现记事本功能5.1 添加新记事项5.2 删除记事项5.3 清空所有记事 6. 添加样式7. 功能扩展:显示创建时间8. 功能扩展:记事项搜索9. 完整代码10. Vue知识点解析10.1 数据绑…...

C++初阶-list的底层
目录 1.std::list实现的所有代码 2.list的简单介绍 2.1实现list的类 2.2_list_iterator的实现 2.2.1_list_iterator实现的原因和好处 2.2.2_list_iterator实现 2.3_list_node的实现 2.3.1. 避免递归的模板依赖 2.3.2. 内存布局一致性 2.3.3. 类型安全的替代方案 2.3.…...
Cesium1.95中高性能加载1500个点
一、基本方式: 图标使用.png比.svg性能要好 <template><div id"cesiumContainer"></div><div class"toolbar"><button id"resetButton">重新生成点</button><span id"countDisplay&qu…...
MySQL中【正则表达式】用法
MySQL 中正则表达式通过 REGEXP 或 RLIKE 操作符实现(两者等价),用于在 WHERE 子句中进行复杂的字符串模式匹配。以下是核心用法和示例: 一、基础语法 SELECT column_name FROM table_name WHERE column_name REGEXP pattern; …...
Xen Server服务器释放磁盘空间
disk.sh #!/bin/bashcd /run/sr-mount/e54f0646-ae11-0457-b64f-eba4673b824c # 全部虚拟机物理磁盘文件存储 a$(ls -l | awk {print $NF} | cut -d. -f1) # 使用中的虚拟机物理磁盘文件 b$(xe vm-disk-list --multiple | grep uuid | awk {print $NF})printf "%s\n"…...
C#中的CLR属性、依赖属性与附加属性
CLR属性的主要特征 封装性: 隐藏字段的实现细节 提供对字段的受控访问 访问控制: 可单独设置get/set访问器的可见性 可创建只读或只写属性 计算属性: 可以在getter中执行计算逻辑 不需要直接对应一个字段 验证逻辑: 可以…...

AirSim/Cosys-AirSim 游戏开发(四)外部固定位置监控相机
这个博客介绍了如何通过 settings.json 文件添加一个无人机外的 固定位置监控相机,因为在使用过程中发现 Airsim 对外部监控相机的描述模糊,而 Cosys-Airsim 在官方文档中没有提供外部监控相机设置,最后在源码示例中找到了,所以感…...
C++课设:简易日历程序(支持传统节假日 + 二十四节气 + 个人纪念日管理)
名人说:路漫漫其修远兮,吾将上下而求索。—— 屈原《离骚》 创作者:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 专栏介绍:《编程项目实战》 目录 一、为什么要开发一个日历程序?1. 深入理解时间算法2. 练习面向对象设计3. 学习数据结构应用二、核心算法深度解析…...

宇树科技,改名了!
提到国内具身智能和机器人领域的代表企业,那宇树科技(Unitree)必须名列其榜。 最近,宇树科技的一项新变动消息在业界引发了不少关注和讨论,即: 宇树向其合作伙伴发布了一封公司名称变更函称,因…...