isMemberOfClass、isKindOfClass原理分析
isMemberOfClass
- 调用者必须是传入的类的实例对象才返回YES
- 判断调用者是否是传入对象的实例,别弄反了,如 [s1 isMemberOfClass:p1] ,意思是s1是否是p1的实例对象
- 不进行父类递归去查找判断
源码:
+ (BOOL)isMemberOfClass:(Class)cls {
return object_getClass((id)self) == cls;
}
- (BOOL)isMemberOfClass:(Class)cls {
return [self class] == cls;
}
有两个方法,一个实例方法,一个类方法,两者区别:
- 实例方法:是根据实例对象取得类对象,再去判断
- 类方法:是根据对象取得元类对象,再去判断
实例代码:
XPerson* p1 = [[XPerson alloc]init];
XStudent* s1 = [[XStudent alloc]init]; // true (用s1的类对象和 [s1 class] 判断,肯定是一样的了)
NSLog(@"s1是否是s1 实例: %i",[s1 isMemberOfClass:[s1 class]]);
// true ([s1 class] 与 [XStudent class] 等同,一个类只会有一个类对象,一个元类对象,可以有多个实例对象)
NSLog(@"s1是否是XStudent 实例: %i",[s1 isMemberOfClass:[XStudent class]]);
// false (s1的类对象 != p1的类对象)
NSLog(@"s1是否是p1 实例: %i",[s1 isMemberOfClass:[p1 class]]);
// false (同上)
NSLog(@"s1是否是XPerson 实例: %i",[s1 isMemberOfClass:[XPerson class]]); // false (XStudent元类对象 != XStudent类对象)(底层是获取XStudent的元类去比较)
NSLog(@"XStudent是否是XStudent 实例: %i",[XStudent isMemberOfClass:[XStudent class]]);
// true (XStudent元类对象 = XStudent元类对象)
NSLog(@"XStudent是否是XStudent元类 实例: %i",[XStudent isMemberOfClass:object_getClass([XStudent class])]);
// false (XStudent元类对象 != XPerson元类对象)
NSLog(@"XStudent是否是XPerson元类 实例: %i",[XStudent isMemberOfClass:objet_getClass([XPerson class])]);
isKindOfClass
- 调用者是传入的类的实例对象,或者调用者是传入类的继承者链中的类的实例对象,则返回YES
- 判断调用者是否是传入对象的子类,别弄反了
- 去父类递归查找判断
源码:
+ (BOOL)isKindOfClass:(Class)cls {
for (Class tcls = object_getClass((id)self); tcls; tcls = tcls->super_class) {
if(tcls == cls) return YES;
}
return NO;
}
-(BOOL)isKindOfClass:(Class)cls {
for(Class tcls = [self class]; tcls; tcls = tcls->super_class) {
if(tcls == cls) return YES;
}
return NO;
}
实例代码:
XPerson* p1 = [[XPerson alloc]init];
XStudent* s1 = [[XStudent alloc]init]; // true (用s1的类对象、父类类对象、基类类对象([NSObject class]) 去和 p1的类对象比较,[p1 class]是s1的父类类对象)
NSLog(@"s1是否是p1 子类: %i",[s1 isKindOfClass:[p1 class]]);
// true (同上:[XPerson class] 与 [p1 class]是等同的)
NSLog(@"s1是否是XPerson 子类: %i",[s1 isKindOfClass:[XPerson class]]);
// true ([NSObject class]是s1的基类类对象)
NSLog(@"s1是否是NSObject 子类: %i",[s1 isKindOfClass:[NSObject class]]); // false (用 XStudent的元类、父元类、基类(NSObject类对象) 去与 XPerson的对象比较)
NSLog(@"XStudent是否是XPerson 子类: %i",[XStudent isKindOfClass:[XPerson class]]);
// true (类方法需要传入 元类进去判断,里面会取 XStudent 元类去比较)
NSLog(@"XStudent是否是XPerson元类 子类: %i",[XStudent isKindOfClass:object_getClass([XPerson class])]);
// true (元类的最上层就是基元类)
NSLog(@"XStudent是否是NSObject元类 子类: %i",[XStudent isKindOfClass:object_getClass([NSObject class])]);
isMemberOfClass、isKindOfClass原理分析的更多相关文章
- WebViewJavascriptBridge 原理分析
WebViewJavascriptBridge 原理分析 网上好多都是在介绍 WebViewJavascriptBridge如何使用,这篇文章就来说说 WebViewJavascriptBridge ...
- Handler系列之原理分析
上一节我们讲解了Handler的基本使用方法,也是平时大家用到的最多的使用方式.那么本节让我们来学习一下Handler的工作原理吧!!! 我们知道Android中我们只能在ui线程(主线程)更新ui信 ...
- Java NIO使用及原理分析(1-4)(转)
转载的原文章也找不到!从以下博客中找到http://blog.csdn.net/wuxianglong/article/details/6604817 转载自:李会军•宁静致远 最近由于工作关系要做一 ...
- 原子类java.util.concurrent.atomic.*原理分析
原子类java.util.concurrent.atomic.*原理分析 在并发编程下,原子操作类的应用可以说是无处不在的.为解决线程安全的读写提供了很大的便利. 原子类保证原子的两个关键的点就是:可 ...
- Android中Input型输入设备驱动原理分析(一)
转自:http://blog.csdn.net/eilianlau/article/details/6969361 话说Android中Event输入设备驱动原理分析还不如说Linux输入子系统呢,反 ...
- 转载:AbstractQueuedSynchronizer的介绍和原理分析
简介 提供了一个基于FIFO队列,可以用于构建锁或者其他相关同步装置的基础框架.该同步器(以下简称同步器)利用了一个int来表示状态,期望它能够成为实现大部分同步需求的基础.使用的方法是继承,子类通过 ...
- Camel运行原理分析
Camel运行原理分析 以一个简单的例子说明一下camel的运行原理,例子本身很简单,目的就是将一个目录下的文件搬运到另一个文件夹,处理器只是将文件(限于文本文件)的内容打印到控制台,首先代码如下: ...
- NOR Flash擦写和原理分析
NOR Flash擦写和原理分析 1. NOR FLASH 的简单介绍 NOR FLASH 是很常见的一种存储芯片,数据掉电不会丢失.NOR FLASH支持Execute On Chip,即程序可以直 ...
- 使用AsyncTask异步更新UI界面及原理分析
概述: AsyncTask是在Android SDK 1.5之后推出的一个方便编写后台线程与UI线程交互的辅助类.AsyncTask的内部实现是一个线程池,所有提交的异步任务都会在这个线程池中的工作线 ...
随机推荐
- Storm 学习之路(二)—— Storm核心概念详解
一.Storm核心概念 1.1 Topologies(拓扑) 一个完整的Storm流处理程序被称为Storm topology(拓扑).它是一个是由Spouts 和Bolts通过Stream连接起来的 ...
- 安装Flume——海量日志收集聚合系统
下载flume: 1.官方网站下载: http://flume.apache.org/download.html 2.百度网盘资源: apache-flume-1.9.0-bin.tar 链接:ht ...
- 基于STM32之UART串口通信协议(二)发送
一.前言 1.简介 在上一篇UART详解中,已经有了关于UART的详细介绍了,也有关于如何使用STM32CubeMX来配置UART的操作了,而在该篇博客,主要会讲解一下如何实现UART串口的发送功能. ...
- asyncio源码分析之基本执行流程
基于async关键字的原生协程 # 定义一个简单的原生协程cor async def cor(): print('enter cor') print('exit cor') print(type(co ...
- java虚拟机-GC-新生代的GC、老年代的GC
名词解释: GC:垃圾收集器 Minor GC:新生代GC,指发生在新生代的垃圾收集动作,所有的Minor GC都会触发全世界的暂停(stop-the-world),停止应用程序的线程,不过这个过程非 ...
- django基础知识之Ajax:
使用Ajax 使用视图通过上下文向模板中传递数据,需要先加载完成模板的静态页面,再执行模型代码,生成最张的html,返回给浏览器,这个过程将页面与数据集成到了一起,扩展性差 改进方案:通过ajax的方 ...
- netty实现的RPC框架
自己手撸了一个nettyRPC框架,希望在这里给有兴趣的同学们做个参考. 要想实现nettyrpc需要了解的技术要点如下: spring的自定义注解.spring的bean的有关初始化. 反射和动态代 ...
- PCB 板边倒圆角的实现方法(基本算法一)
PCB外形是直角时外形时,通常工程制作时,外是直角或尖角的地方倒圆角,主要是为了防止板边容易划伤板且容易扎伤人 所以当客户没有特殊要求时,PCB外形是直角时一般会默认倒角0.5mm圆角(如下图所示) ...
- 详叙BeanWrapper和PropertyDescriptor
每篇一句 千古以来要饭的没有要早饭的,知道为什么吗? 相关阅读 [小家Spring]聊聊Spring中的数据转换:Converter.ConversionService.TypeConverter.P ...
- numpy表示图片详解
我自己的一个体会,在学习机器学习和深度学习的过程里,包括阅读模型源码的过程里,一个比较大的阻碍是对numpy掌握的不熟,有的时候对矩阵的维度,矩阵中每个元素值的含义晕乎乎的. 本文就以一个2 x 2 ...