关于self与self class用法上的区别

在许多地方可以见到self和[self class]的调用方式,那么他们有区别吗?

1 首先self是什么,它是指向实例变量首地址的指针(同c++的this一样)可以访问对象的资源。

2 [self class] 首先看下class 它返回的是元类的指针,所以通过[self class]可以访问当前类的静态函数

为什么[self class] 可以访问到类指针呢,可以看下NSObject的结构:

@interface NSObject <NSObject> {

Class isa  OBJC_ISA_AVAILABILITY;

}

//任何类都是继承于NSObject所以在内存中isa是首地址 ,self指向的就是isa的的地址。

看下在runtime里面class是怎么调用的,首先是获取NSObject 类的isa变量,通过isa来返回元类的指针,所以也就是说class是返回了当前类的类指针。

- (Class)class {

return object_getClass(self);

}

Class object_getClass(id obj)

{

if (obj) return obj->getIsa();

else return Nil;

}

3 那么[self class]除了可以访问静态函数,变量还有其他用处吗?看下面这个例子

@interface Engine:NSObject<NSCopying>

@end

@implementation Engine

-(id)copyWithZone:(NSZone *)zone{

return [[[self class]allocWithZone:zone]init];

}

我们需要将消息发送给一个类,而不是一个实例变量。那么应该发送给哪个类呢?直觉告诉我们应该是Engine,像这样:[Engine allocWithZone:zone];

但是这行代码只适用于Engine类,不适合它的之类。如果给它的子类Slant6发送copy消息,最后使用的是Engine类的拷贝而不是slant6类的拷贝,创建出来的时Engine对象。

所以要使用[self class]他可以将消息发送给正在接受copy消息的对象所属的类。如果self是一个Slvant6的对象,就会发送给Slvant6。

时间: 2024-10-09 10:36:39

关于self与self class用法上的区别的相关文章

C与C++在const用法上的区别

首先,C和C++在大体结构上不同,却在语法上相同.  所以在使用的时候,我们会时常遇到一些莫名其妙的问题,觉得语法上是正确的,但是编译的时候却出现一个红色的 error! 比如今天我遇到的这个有意思的问题. 1. 今天看到一段代码,觉得很有意思: 1 #pragma warning(disable : 4090) 2 3 #include<stdio.h> 4 5 int main(int argv, char args []) { 6 7 const int var = 4; 8 int *

关于Java中this和super的用法介绍和区别

1.this&super 什么是this,this是自身的一个对象,代表对象本身,可以理解为:指向对象本身的一个指针.当你想要引用当前对象的某种东西,比如当前对象的某个方法,或当前对象的某个成员,你便可以利用this来实现这个目的.要注意的是this只能在类中的非静态方法中使用,静态方法和静态的代码块中绝对不能出现this.his也可作为构造函数来使用.在后面可以看到 而什么是super,可以理解为是指向自己超(父)类对象的一个指针,而这个超类指的是离自己最近的一个父类.super的作用同样是可

关于JavaScript中apply与call的用法意义及区别(转)

JavaScript中有一个call和apply方法,其作用基本相同,但也有略微的区别.先来看看JS手册中对call的解释: call 方法调用一个对象的一个方法,以另一个对象替换当前对象.call([thisObj[,arg1[, arg2[,   [,.argN]]]]])参数thisObj可选项.将被用作当前对象的对象.arg1, arg2,  , argN可选项.将被传递方法参数序列.说明call 方法可以用来代替另一个对象调用一个方法.call 方法可将一个函数的对象上下文从初始的上下

redhat6和centos7在yum应用上的区别

最近在虚拟机上尝试了各种版本安装oracle11数据库,发现在Linux配置上一些区别,在centos7上,yum是自带配置信息的,在安装oracle需要的rpm包时可以直接安装,但是在redhat上需要首先配置yum的相关信息: 1 首先挂载虚拟光驱 创建空文件夹 mkdir /dvd mount /dev/sr0 /dvd/ cd /dvd ls 可以看到Linux安装盘中的信息 2 cd /etc/yum.pepos.d/ ls  会出现 rhel-source.repo vim rhel

WITH RECOMPILE 和 OPTION(RECOMPILE) 使用上的区别

在考虑重编译T-SQL(或者存储过程)的时候,有两种方式可以实现强制重编译(前提是忽略导致重编译的其他因素的情况下,比如重建索引,更新统计信息等等), 一是基于WITH RECOMPILE的存储过程级别重编译,另外一种是基于OPTION(RECOMPILE)的语句级重编译. 之前了解的比较浅,仅仅认为是前者就是编译整个存储过程中的所有的语句,后者是重编译存储过程中的某一个语句,也没有追究到底是不是仅仅只有这么一点区别. 事实上在某些特定情况下,两者的区别并非仅仅是存储过程级重编译和语句级重编译的

理解BPDU Guard的意义(BPDU Guard在全局配置与接口配置上的区别)

理解BPDU Guard的意义(BPDU Guard在全局配置与接口配置上的区别)   本文截自于博主CCNP交换技术稿件内容   BPDU Guard(BPDU保护),简单的讲它的意义就是一个不该接收BPDU的端口,比如被启动了portfast的端口,一旦收到BPDU报文,那么BPDU保护功能将会立即关闭该端口,并将端口状态置为error-disabled状态.BPDU Guard的配置分为全局型的配置和接口级的配置,注意这两种配置将带来一些不同的效果. 全局配置BPDU Guard 全局配置

VC和gcc在保证函数static变量线程安全性上的区别

VC和gcc不同,不能保证静态变量的线程安全性.这就给我们的程序带来了很大的安全隐患和诸多不便.这一点应该引起我们的重视!尤其是在构造函数耗时比较长的时候,很可能给程序带来意想不到的结果.本文从测试代码开始,逐步分析原理,最后给出解决方案. 多线程状态下,VC不能保证在使用函数的静态变量的时候,它的构造函数已经被执行完毕,下面是一段测试代码: class TestStatic { public: TestStatic() { Sleep(1000*10); m_num = 999; } publ

引用类型与值类型在编码上的区别

一.引入类型与值类型简介 值类型:直接存放于栈中,取的时候是直接取得值.值类型继承自System.ValueType.(自定义对象) 引用类型:存在于托管堆中,取的时候取的是该对象的地址,然后用这个地址去托管堆中取值.引用类型继承自System.Object.(int,bool) 二.在代码编写上的区别 在赋值的时候,值类型是复制一份,新的和旧的在以后的操作中互不影响,而引用类型复制的只是地址,在以后的修改中,修改新的旧的也会受到影响. 代码示例: static void Main(string

Odoo9以后的社区版本和企业版功能上的区别

Odoo9以后的社区版本和企业版除了授权模式的区别外,整理功能上的区别 透过功能设置菜单整理的区别如下,主要功能模块. 未包括所有模块,毕竟模块太多了. 以下是企业版有,而社区版没有的功能.