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的内部实现是一个线程池,所有提交的异步任务都会在这个线程池中的工作线 ...
随机推荐
- 关于exe4j打包问题
一.eclipse导出jar Export-->Runnable JAR file 这里有两种情况: 选择 Package required libraries into generated j ...
- 六种 主流ETL 工具的比较(DataPipeline,Kettle,Talend,Informatica,Datax ,Oracle Goldengate)
六种 主流ETL 工具的比较(DataPipeline,Kettle,Talend,Informatica,Datax ,Oracle Goldengate) 比较维度\产品 DataPipeline ...
- Codeblocks 批量注释与对齐快捷键的教学方法
Ctrl+Shift+C 批量注释 Ctrl+shift+X 批量取消注释 Click Settings->Editor->KeyboardShortcuts (in the left o ...
- 深度探索c++对象模型 第二章
1,c++转换函数:显示转换和隐式转换. 隐式转换为程序员提供了很大的变量.比如整形提升,普通类型转换为类类型(operator int())都为程序带来无尽的方便.试想,如果没有整形提升,一个sho ...
- Postman接口测试_基本功能
一. 安装与更新 1.安装的方式 方式1:chrome插件版本:chrome--->设置--->扩展程序: 方式2:native版本(具有更好的扩展性,推荐使用):https://ww ...
- PyCharm2018 汉化&激活
一.汉化 将下载好的resources_cn_PyCharm_2018.1_r2.jar 放入pycharm 的lib 目录中,启动app即可 下载链接: https://pan.baidu.com/ ...
- CPU缓存和内存屏障
CPU性能优化手段 - 缓存 为了提高程序的运行性能, 现代CPU在很多方面对程序进行了优化例如: CPU高速缓存, 尽可能的避免处理器访问主内存的时间开销, 处理器大多会利用缓存以提高性能 多级缓存 ...
- scrapy基础知识之 使用FormRequest.from_response()方法模拟用户登录:
通常网站通过 实现对某些表单字段(如数据或是登录界面中的认证令牌等)的预填充 使用Scrapy抓取网页时,如果想要预填充或重写像用户名.用户密码这些表单字段, 可以使用 FormRequest.fro ...
- .Net Core 学习依赖注入自定义Service
1. 定义一个服务,包含一个方法 public class TextService { public string Print(string m) { return m; } } 2. 写一个扩展方法 ...
- infiniband install driver
硬件:Mellanox InfiniBand,主要包括 HCA(主机通道适配器)和交换机两部分 软件:CentOS 6.4 MLNX_OFED_LINUX-2.1-1.0.0-rhel6.4-x86_ ...