【iOS】isKindOfClass和isMemberOfClass方法
前言
这个归根结底还是在考察我们对isa走向图和类的继承的理解,也就是苹果官方这幅图:
接下来的函数调用流程请参考这张图。
1 isKindOfClass方法
1.1 objc_opt_isKindOfClass C函数
查看源码可发现,无论是谁调用isKindOfClass
方法都会进入这个C函数。(这个C函数位于NSObjective.mm
)
// Calls [obj isKindOfClass]
// 当obj调用isKindOfClass时,objc_opt_isKindOfClass会被触发
// obj是一个id类型,id是一个objc_object结构体指针,意味着,传进来的可以是时类,也可以是类的实例对象
// otherClass就是isKindOfClass的参数,我们当初传进去的cls
BOOL objc_opt_isKindOfClass(id obj, Class otherClass)
{
#if __OBJC2__if (slowpath(!obj)) return NO;Class cls = obj->getIsa(); // 此处的cls仅是obj的第一个isaif (fastpath(!cls->hasCustomCore())) {// otherClass 从obj的ISA开始,依次和ISA的父类比较,直到找到或者父类为nil结束// 当父类为nil意味着最后一个和otherClass比较的是NSObject根类。for (Class tcls = cls; tcls; tcls = tcls->superclass) {if (tcls == otherClass) return YES;}return NO;}
#endifreturn ((BOOL(*)(id, SEL, Class))objc_msgSend)(obj, @selector(isKindOfClass:), otherClass);
}
可知:
- 一切皆从调用者
obj
的isa
开始,然后顺着superclass
走下去,直到找到cls
或superclass
为nil
结束。 - 当
superclass
为nil
,意味着最后的根类NSObject
也不是cls
,返回flase
。
1.2 类SubClass调用+ (BOOL)isKinsOfClass:(Class)cls
流程:
- 从类的
isa
——元类开始,判断它不是cls
; - 如果是,返回
true
; - 如果不是,继续用元类的
superclass
和cls
比较,看是不是cls
; - 直到根类
NSObject
也比较完。
判断顺序: SubClass
与 MetaClass->MetaClass->...->RootMetaClass->NSObject
。
总结:
- 判断
cls
是不是 元类->父类的元类->父父类的元类->…->根元类->NSObject
(元类的superclass
继承链)其中一个。 cls
传除NSObject.class
外的任意类对象均为false
。
1.3 元类MetaClass 调用+ (BOOL)isKinsOfClass:(Class)cls
流程:
MetaClass
的ISA
指向RootMetaClass
,所以从RootMetaClass
开始比较判断是不是我们传入的cls;- 如果不是,再看根类
NSObject
是不是,如果NSObject
也不是,就彻底没有了,返回false
。
判断顺序:MetaClass
与 RootMetaClass->NSObject
。
总结:
- 判断
cls
是不是根元类->NSObject
中的任意一个。
1.4 对象obj 调用- (BOOL)isKinsOfClass:(Class)cls
流程:
- 从
isa
指向的类对象开始,判断是不是cls
; - 如果不是,看类对象的父类,逐级判断是不是
cls
; - 直到找到返回
true
,或者判断到NSObject
依然不是,返回false
结束。
判断顺序: object
与 SubClass -> SubClass ->...->NSObject
。
总结:
- 判断cls是不是 类对象->父类->…->NSObject (superclass继承链)其中一个。
2 isMemberOfClass
2.1 类对象SubClass调用+ (BOOL)isMemberOfClass
源码:
+ (BOOL)isMemberOfClass:(Class)cls {return self->ISA() == cls;
}
不用像isKindOfClass循环直到找到或nil,他只要比较cls是不是我当前的isa指向,是返回true,不是返回false。
判断:SubClass
与MetaClass
。
2.2 元类MetaClass 调用+ (BOOL)isMemberOfClass
因为元类的isa只指向根元类NSObejct ,所以除了NSObject的类SubClass以外,传入任何类对象也都是false。
验证传入NSObject的类SubClass的结果:
void demo(void) {
// BOOL re1 = [[NSObject class] isKindOfClass:[NSObject class]];Class rootMetaClass = object_getClass([NSObject class]);NSLog(@"%d", [[NSObject class] isMemberOfClass:rootMetaClass]);
}
结果:
2.3 对象obj调用 -(BOOL)isMemberofClass:(Class)clss
- (BOOL)isMemberOfClass:(Class)cls
底层源码:
- (BOOL)isMemberOfClass:(Class)cls {return [self class] == cls;
}
- (Class)class {return object_getClass(self); // 获取当前的isa指向的类
}
只要判断对象的isa,也就是图中的SubClass是不是我们传入的cls。
3. 测试
void demo(void) {BOOL f1 = [(id)[NSObject class] isKindOfClass:[NSObject class]];BOOL f2 = [(id)[MyClass class] isKindOfClass:[MyClass class]];BOOL f3 = [(id)[MySuperClass class] isKindOfClass:[MySuperClass class]];BOOL f4 = [(id)[MyClass class] isKindOfClass:[MySuperClass class]];BOOL f5 = [(id)[NSObject alloc] isKindOfClass:[NSObject class]];BOOL f6 = [(id)[MyClass alloc] isKindOfClass:[NSObject class]];BOOL f7 = [(id)[MySuperClass alloc] isKindOfClass:[NSObject class]];NSLog(@"NSObjectClass ISKindOf NSObjectClass:%d", f1);NSLog(@"MyClassClass ISKindOf MyClassClass:%d", f2);NSLog(@"MySuperClassClass ISKindOf MySuperCassClass:%d", f3);NSLog(@"MyClassClass ISKindOf MySuperClassClass:%d", f4);NSLog(@"NSObjectObj ISKindOf NSObjectClass:%d", f5);NSLog(@"MyClassObj ISKindOf NSObjectClass:%d", f6);NSLog(@"MySuperClassObj ISKindOf NSObjectClass:%d", f7);}
测试结果:
相关文章:

【iOS】isKindOfClass和isMemberOfClass方法
前言 这个归根结底还是在考察我们对isa走向图和类的继承的理解,也就是苹果官方这幅图: 接下来的函数调用流程请参考这张图。 1 isKindOfClass方法 1.1 objc_opt_isKindOfClass C函数 查看源码可发现,无论是谁调用isKindOfClass方法都会…...

李飞飞「具身智能」VoxPoser:0预训练完成复杂指令
机器人接入大模型听懂人话 论文地址: https://voxposer.github.io/voxposer.pdf 项目主页: https://voxposer.github.io/ 参考链接: [1]https://twitter.com/wenlong_huang/status/1677375515811016704 [1]https://www.amacad.org/publicatio…...
前端八股文
info 毕业设计(课题、方向 本科毕业设计:家庭医生签约管理系统后台开发(微信小程序) 硕士课题:医学图像分割(婴儿脑分割) 51062319991129351X 邮编 重庆市南岸区 400000 13183849783 // 18728097929 // 13158442955 中国广电四川网络股份有限公司中江…...

前端年度工作述职报告优秀
前端年度工作述职报告优秀篇1 尊敬的各位领导、各位同仁: 大家好!按照20__年度我公司就职人员工作评估的安排和要求,我认真剖析、总结了自己的工作情况,现将本人工作开展情况向各位领导、同仁做以汇报,有不妥之处,希…...

【MyBatis 学习一】认识MyBatis 第一个MyBatis查询
目录 一、认识MyBatis 1、MyBatis是什么? 2、为什么要学习MyBatis? 二、配置MyBatis环境 1、建库与建表 2、创建新项目 3、xml文件配置 (1)配置数据库连接 (2)配置 MyBatis 中的 XML 路径 三、测试&#x…...
TCP 和 UDP
TCP(Transmission Control Protocol,传输控制协议) 是面向连接的协议,即在收发数据前,必须和对方建立可靠的连接,TCP的头部为20个字节。 UDP(User Datagram Protocol,用户数据报协…...

springboot配置自定义数据源(Druid德鲁伊)的步骤。
今天和大家分享下在Springboot中配置自定义数据源Druid的两种方法及步骤。 方法一: 1.在pom.xml配置依赖(注释里面的内容) 2.配置自己的数据源设置,我是在yaml文件中配置的,顺便提醒一下,在配置yaml文件的时候缩进问题一定要注意…...

K8S:容器日志收集与管理
Kubernetes 里面对容器日志的处理方式,都叫作 cluster-level-logging,即:这个日志处理系统,与容器、Pod 以及 Node 的生命周期都是完全无关的。这种设计当然是为了保证,无论是容器挂了、Pod 被删除,甚至节点…...

Flutter系列文章-Flutter进阶
在前两篇文章中,我们已经了解了Flutter的基础知识,包括Flutter的设计理念、框架结构、Widget系统、基础Widgets以及布局。在本文中,我们将进一步探讨Flutter的高级主题,包括处理用户交互、创建动画、访问网络数据等等。为了更好地…...

【C++】C++11右值引用|新增默认成员函数|可变参数模版|lambda表达式
文章目录 1. 右值引用和移动语义1.1 左值引用和右值引用1.2 左值引用和右值引用的比较1.3右值引用的使用场景和意义1.4 左值引用和右值引用的深入使用场景分析1.5 完美转发1.5.1 万能引用1.5.2 完美转发 2. 新的类功能2.1 默认成员函数2.2 类成员变量初始化2.3 强制生成默认函数…...
rust学习-线程
Rust 标准库只提供了 1:1 线程模型 Rust 是较为底层的语言,如果愿意牺牲性能来换取抽象,以获得对线程运行更精细的控制及更低的上下文切换成本,使用实现了 M:N 线程模型的 crate 示例 use std::thread; use std::time::Duration;fn main() …...
题目:2180.统计各位数字之和为偶数的整数个数
题目来源: leetcode题目,网址:2180. 统计各位数字之和为偶数的整数个数 - 力扣(LeetCode) 解题思路: 暴力遍历即可。 解题代码: class Solution {public int countEven(int num) {int re…...

3dsmax制作一个机器人
文章目录 建模身子:眼睛:头饰:肩膀手臂腿调整细节 渲染导出objMarmoset Toolbag 3.08渲染给眼睛添加材质,设置为自发光添加背景灯光 建模 身子: 眼睛: 头饰: 肩膀 手臂 腿 调整细节 渲染 导出…...
C++的类型转换运算符:reinterpret_cast
C的类型转换运算符:reinterpret_cast reinterpret_cast 是 C 中与 C 风格类型转换最接近的类型转换运算符。它让程序员能够将一种对象类型转换为另一种,不管它们是否相关;也就是说,它使用如下所示的语法强制重新解释类型…...

flask中的cookies介绍
flask中的cookies介绍 “Cookie” 在 web 开发中是一种非常重要的技术,用于在客户端(即用户的浏览器)存储信息,以便在多个页面和多个访问会话之间保持状态。Cookies 通常用于记住用户的登录信息,跟踪用户在站点上的浏…...

adnroid 11. 0 Activity启动流程图解
从Launcher到ActivityTaskManager 从ActivityTaskManagerService 到 ApplicationThread 从ApplicationThread到onCreate...

了解Unity编辑器之组件篇Physics(四)
Physics:用于处理物理仿真和碰撞检测。它提供了一组功能强大的工具和算法,用于模拟真实世界中的物理行为,使游戏或应用程序更加真实和可信。 主要用途包括: 碰撞检测:Unity Physics 提供了高效的碰撞检测算法&#x…...

“数字中华 点亮未来”中华线上客户节 盛大开幕
2023年是中华保险数字化转型落地之年,峥嵘37载,中华保险在数字化转型上已经涌现了一批彰显辨识度、具有影响力的应用成果。7月15日,中华保险围绕数字化转型之路开展以“数字中华 点亮未来”为主题的37周年线上客户节活动,倾力打造…...
中文分词入门:使用IK分词器进行文本分词(附Java代码示例)
1. 介绍 中文分词是将连续的中文文本切分成一个个独立的词语的过程,是中文文本处理的基础。IK分词器是一个高效准确的中文分词工具,采用了"正向最大匹配"算法,并提供了丰富的功能和可定制选项。 2. IK分词器的特点 细粒度和颗粒…...

CTFSHOW web 信息收集
web入门的刷题 web1 教我们多看看源代码 web2 抓包 web3 抓包 web4 robots.txt robots.txt web5 phps源代码泄露 phps 就是php的源代码 用户无法访问 php 只可以通过phps来访问 web6 源代码备份 web7 git web8 svn web9 swp /index.php.swp web10 cookie web11 查域名…...
谷歌浏览器插件
项目中有时候会用到插件 sync-cookie-extension1.0.0:开发环境同步测试 cookie 至 localhost,便于本地请求服务携带 cookie 参考地址:https://juejin.cn/post/7139354571712757767 里面有源码下载下来,加在到扩展即可使用FeHelp…...

大数据学习栈记——Neo4j的安装与使用
本文介绍图数据库Neofj的安装与使用,操作系统:Ubuntu24.04,Neofj版本:2025.04.0。 Apt安装 Neofj可以进行官网安装:Neo4j Deployment Center - Graph Database & Analytics 我这里安装是添加软件源的方法 最新版…...

工业安全零事故的智能守护者:一体化AI智能安防平台
前言: 通过AI视觉技术,为船厂提供全面的安全监控解决方案,涵盖交通违规检测、起重机轨道安全、非法入侵检测、盗窃防范、安全规范执行监控等多个方面,能够实现对应负责人反馈机制,并最终实现数据的统计报表。提升船厂…...

如何在看板中体现优先级变化
在看板中有效体现优先级变化的关键措施包括:采用颜色或标签标识优先级、设置任务排序规则、使用独立的优先级列或泳道、结合自动化规则同步优先级变化、建立定期的优先级审查流程。其中,设置任务排序规则尤其重要,因为它让看板视觉上直观地体…...

涂鸦T5AI手搓语音、emoji、otto机器人从入门到实战
“🤖手搓TuyaAI语音指令 😍秒变表情包大师,让萌系Otto机器人🔥玩出智能新花样!开整!” 🤖 Otto机器人 → 直接点明主体 手搓TuyaAI语音 → 强调 自主编程/自定义 语音控制(TuyaAI…...

AI,如何重构理解、匹配与决策?
AI 时代,我们如何理解消费? 作者|王彬 封面|Unplash 人们通过信息理解世界。 曾几何时,PC 与移动互联网重塑了人们的购物路径:信息变得唾手可得,商品决策变得高度依赖内容。 但 AI 时代的来…...

Netty从入门到进阶(二)
二、Netty入门 1. 概述 1.1 Netty是什么 Netty is an asynchronous event-driven network application framework for rapid development of maintainable high performance protocol servers & clients. Netty是一个异步的、基于事件驱动的网络应用框架,用于…...

关于easyexcel动态下拉选问题处理
前些日子突然碰到一个问题,说是客户的导入文件模版想支持部分导入内容的下拉选,于是我就找了easyexcel官网寻找解决方案,并没有找到合适的方案,没办法只能自己动手并分享出来,针对Java生成Excel下拉菜单时因选项过多导…...

若依登录用户名和密码加密
/*** 获取公钥:前端用来密码加密* return*/GetMapping("/getPublicKey")public RSAUtil.RSAKeyPair getPublicKey() {return RSAUtil.rsaKeyPair();}新建RSAUti.Java package com.ruoyi.common.utils;import org.apache.commons.codec.binary.Base64; im…...
前端调试HTTP状态码
1xx(信息类状态码) 这类状态码表示临时响应,需要客户端继续处理请求。 100 Continue 服务器已收到请求的初始部分,客户端应继续发送剩余部分。 2xx(成功类状态码) 表示请求已成功被服务器接收、理解并处…...