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的内部实现是一个线程池,所有提交的异步任务都会在这个线程池中的工作线 ...
随机推荐
- ABP开发框架前后端开发系列---(12)配置模块的管理
一般来说,一个系统或多或少都会涉及到一些系统参数或者用户信息的配置,而ABP框架也提供了一套配置信息的管理模块,ABP框架的配置信息,必须提前定义好配置的各项内容,然后才能在系统中初始化或者通过接口查 ...
- Appium+python自动化(十二)- Android UIAutomator终极定位凶“胸”器(七)(超详解)
简介 乍眼一看,小伙伴们觉得这部分其实在异性兄弟那里就做过介绍和分享了,其实不然,上次介绍和分享的大哥是uiautomatorviewer,是一款定位工具.今天介绍的是一个java库,提供执行自动化测 ...
- kubernetes实战篇之helm填坑与基本命令
系列目录 其实前面安装部分我们已经分享一些互联网上其它网友分享的一些坑,本篇介绍helm的基本使用以及在使用过程中碰到的一些坑. 客户端版本和服务端版本不一致问题 有些朋友可能在使用helm init ...
- 【Spring源码解析】—— 策略模式在Spring中的应用
一. 什么是策略模式 策略模式的定义/含义:策略本身就是为了实现某一个目标而采取的一种工作方式,因此只要能够达成目标,则采取哪一种策略都可以:因此多种实际的策略之间是相互平行的. 注意 ...
- Linux 中文件和文件夹获取 MySQL 权限(SELinux)
今天在 Linux 系统上移动 MySQL 的数据库目录 配置如下: /etc/my.cnf [mysqld]datadir=/home/mysqlsocket=/var/lib/mysql/mysq ...
- Docker入门简介(一)
Docker 介绍 Docker是Docker.lnc公司开源的一个基于LXC技术智商构建的Container容器引擎,源代码托管在GitHub上,基于Go语言并遵从Apache2.0协议开源. Do ...
- 从Spring的几个阶段理解其工作过程
Spring框架非常强大,想要彻底弄懂Spring是非常困难的. 为了便于了解Spring的工作原理,我们来研究一下,Spring是怎么加载的,Spring会经过几个阶段. 我们站在Javaweb ...
- oralce中的dual详解 转 http://blog.sina.com.cn/s/blog_a5a24bcb0100zeay.html
dual是属于sys的只有一个X varchar2(1)列查询虚拟列不会产生逻辑IO========================================================== ...
- 仿照Spring自己实现有各种通知的AOP,AOP实现的步骤分解
一.需求: 仿照Spring的AOP写的 MyAOP 2.0,有环绕通知.前置通知.后置通知.返回通知.异常通知等. 已实现:①通过动态代理+通知的注解类,实现了前置通知.后置通知等各种通知:②切点( ...
- WordPress教程之如何入门WordPress
这篇文章将介绍如何设置 WordPress,并自定义其基本功能.WordPress 的安装包相对较小(低于 10 MB),非常易于安装和管理.为了托管自己的网站,你可以获得几个不同的选项. 你可以通过 ...