在Objective-C中,有一些我们之前并不熟悉但是经常见到的数据类型,比如id、nil、Nil、SEL等等。在很多文章里,我们都见过这些数据类型的介绍,但是都没有说的太清楚。

这篇文章从最底层的定义开始,介绍一下这些类型到底是怎么定义的,这会帮助我们更加深入地了解Objective-C。

参考:
http://unixjunkie.blogspot.com/2006/02/nil-and-nil.html

http://blog.csdn.net/itudou_2010/article/details/5501840

Objective-C中有一些很有趣的数据类型经常会被错误地理解。他们中的大多数都可以在/usr/include/objc/objc.h或者这个目录中的其他头文件中找到。下面是从objc.h中摘录的一段,定义了一些数据类型:

// objc.htypedef struct objc_class *Class;typedef struct objc_object {Class isa;
} *id;typedef struct objc_selector  *SEL;
typedef id      (*IMP)(id, SEL, …);
typedef signed char   BOOL;#define YES             (BOOL)1
#define NO              (BOOL)0#ifndef Nil
#define Nil 0   /* id of Nil class */
#endif#ifndef nil
#define nil 0   /* id of Nil instance */
#endif

我们在这里解释一下它们的细节:

id
id和void *并非完全一样。在上面的代码中,id是指向struct objc_object的一个指针,这个意思基本上是说,id是一个指向任何一个继承了Object(或者NSObject)类的对象。需要注意的是id 是一个指针,所以你在使用id的时候不需要加星号。比如id foo=nil定义了一个nil指针,这个指针指向NSObject的一个任意子类。而id *foo=nil则定义了一个指针,这个指针指向另一个指针,被指向的这个指针指向NSObject的一个子类。

nil
nil和C语言的NULL相同,在objc/objc.h中定义。nil表示一个Objctive-C对象,这个对象的指针指向空(没有东西就是空)。

Nil
首字母大写的Nil和nil有一点不一样,Nil定义一个指向空的类(是Class,而不是对象)。

SEL
这个很有趣。SEL是“selector”的一个类型,表示一个方法的名字。比如以下方法:

-[Foo count] 和 -[Bar count] 使用同一个selector,它们的selector叫做count。

在上面的头文件里我们看到,SEL是指向 struct objc_selector的指针,但是objc_selector是什么呢?那么实际上,你使用GNU Objective-C的运行时间库和NeXT Objective-C的运行运行时间库(Mac OS X使用NeXT的运行时间库)时,它们的定义是不一样的。实际上Mac OSX仅仅将SEL映射为C字符串。比如,我们定义一个Foo的类,这个类带有一个- (int) blah方法,那么以下代码:

NSLog (@"SEL=%s", @selector(blah));

会输出为 SEL=blah。

说白了SEL就是返回方法名。

这样的机制大大的增加了我们的程序的灵活性,我们可以通过给一个方法传递SEL参数,让这个方法动态的执行某一个方法;我们也可以通过配置文件指定需要执行的方法,程序读取配置文件之后把方法的字符串翻译成为SEL变量然后给相应的对象发送这个消息。

在 Objective-C 运行时库中,selector 是作为数组来管理的。这都是从效率的角度出发:函数调用的时候,不是通过方法名字比较而是指针值的比较来查找方法,由于整数的查找和匹配比字符串要快得多,所以这样可以在某种程度上提高执行的效率。

这样就必须保证所有类中的 selector 须指向同一实体(数组)。一旦有新的类被定义,其中的 selector 也需要映射到这个数组中。

实际情况下,总共有两种 selector 的数组:预先定义好的内置selector数组 和用于动态追加的selector数组 。

  • 内置selector
简单地说,内置的selector就是一个大的字符串数组。定义在objc-sel-table.h文件中:
#define NUM_BUILTIN_SELS 16371
/* base-2 log of greatest power of 2 < NUM_BUILTIN_SELS */
#define LG_NUM_BUILTIN_SELS 13static const char * const _objc_builtin_selectors[NUM_BUILTIN_SELS] = {".cxx_construct",".cxx_destruct","CGColorSpace","CGCompositeOperationInContext:","CIContext","CI_affineTransform","CI_arrayWithAffineTransform:","CI_copyWithZone:map:","CI_initWithAffineTransform:","CI_initWithRect:","CI_rect","CTM","DOMDocument","DTD",...};
可以看到,数组的大小NUM_BUILTIN_SELS定义为16371。字符串按照字母顺序排序,简单的都是为了运行时检索的速度(二分法查找)。
从定义好的 selector 名称我们可以看到一些新的方法名称,比如 CIConetext,CI开头的方法是由Tiger开始导入的程序库。
每次系统更新的时候,这个数组也是需要更新的。
  • 动态追加selector

另一个用于动态追加的 selector,其定义在 objc-sel.m 和 objc-sel-set.m  文件中 新的 selector 都被追加到 _buckets 成员中,其中追加和搜索使用 Hash 算法。

static struct __objc_sel_set *_objc_selectors = NULL;struct __objc_sel_set {uint32_t _count;uint32_t _capacity;uint32_t _bucketsNum;SEL *_buckets;
};

IMP
从上面的头文件中我们可以看到,IMP定义为

id (*IMP) (id, SEL, …)

这样说来, IMP是一个指向函数的指针,这个被指向的函数包括id(“self”指针),调用的SEL(方法名),再加上一些其他参数。

说白了IMP就是实现方法。

我们取得了函数指针之后,也就意味着我们取得了执行的时候的这段方法的代码的入口,这样我们就可以像普通的 C语言函数调用一样使用这个函数指针。当然我们可以把函数指针作为参数传递到其他的方法,或者实例变量里面,从而获得极大的动态性。我们获得了动态性,但 是付出的代价就是编译器不知道我们要执行哪一个方法所以在编译的时候不会替我们找出错误,我们只有执行的时候才知道,我们写的函数指针是否是正确的。所 以,在使用函数指针的时候要非常准确地把握能够出现的所有可能,并且做出预防。尤其是当你在写一个供他人调用的接口API的时候,这一点非常重要。

Method
在objc/objc-class.h中定义了叫做Method的类型,是这样定义的:

typedef struct objc_method *Method;
struct objc_method {SEL method_name;char *method_types;IMP method_imp;
};

这个定义看上去包括了我们上面说过的其他类型。也就是说,Method(我们常说的方法)表示一种类型,这种类型与selector和实现(implementation)相关。

最初的SEL是方法的名称method_name。char型的method_types表示方法的参数。最后的IMP就是实际的函数指针,指向函数的实现。

Class
从上文的定义看,Class(类)被定义为一个指向struct objc_class的指针,在objc/objc-class.h中它是这么定义的:

struct objc_class {struct objc_class *isa;struct objc_class *super_class;const char *name;long version;long info;long instance_size;struct objc_ivar_list *ivars;struct objc_method_list **methodLists;struct objc_cache *cache;struct objc_protocol_list *protocols;
};
由以上的结构信息,我们可以像类似于C语言中结构体操作一样来使用成员。比如下面取得类的名称:
Class cls;
cls = [NSString class];printf("class name %s\n", ((struct objc_class*)cls)->name);
发送消息与函数调用的不同

Objective-C的消息传送如下图所示 :

Objective-C的消息传送

发送消息的过程,可以总结为以下内容 :

  • 首先,指定调用的方法
  • 为了方法调用,取得 selector

源代码被编译以后,方法被解释为 selector。这里的 selector 只是单纯的字符串。

  • 消息发送给对象B

消息传送使用到了 objc_msgSend 运行时API。这个API只是将 selector 传递给目标对象B。

  • 从 selector 取得实际的方法实现

首先,从对象B取得类的信息,查询方法的实现是否被缓存(上面类定义中的struct objc_cache *cache;)。如果没有被缓 存,则在方法链表中查询(上面类定义中的struct objc_method_list **methodLists;)。

  • 执行

利用函数指针,调用方法的实现。这时,第一个参数是对象实例,第二个是 selector。

  • 传送返回值

利用 objc_msgSend API 经方法的返回值传送回去。

简单地从上面发送消息的过程可以看到,最终还是以函数指针的方式调用了函数。为什么特意花那么大的功夫绕个大圈子呢?

这些年,随着程序库尺寸的扩大,动态链接库的使用已经非常普遍。就是说,应用程序本身并不包括库代码,而是在启动时或者运行过程中动态加载程序库。这样一来一方面可以减小程序大小,另一方面可以提升了代码重用(不用再造轮子)。但是,随之带来了向下兼容的问题。

如果程序库反复升级,添加新的方法的时候,开发者与用户间必须保持一致的版本,否则将产生运行时错误。一般,解决这个问题是,调用新定义的方法的时 候,实现检查当前系统中是否存在新方法的实现。如果没有,跳过它或者简单地产生警告信息。 Objective-C中的respondsToSelector:方法就可以用来实现这样的动作。

但是,这并不是万全的解决方案。如果应用程序与新的动态程序库(含有新定义的API)一起编译后,新定义的API符号也被包含进去。而这样的应用程 序放到比较旧的系统(旧的动态程序库)中运行的时候,因为找不到链接符号,程序将不能启动。这就是 win32系统中常见的「DLL地域」。

为了解决这个问题,Objective-C 编译得到的二进制文件中,函数是作为 selector 来保存的。就是说,不管调用什么函数,二进制文件中不会包含符号信息。为了验证 Objective-C 编译的二进制文件是否包含符号信息,这里用 nm 命令来查看。

int main (int argc, const char * argv[])
{NSString*   string;int         length;string = [[NSString alloc] initWithString:@"Objective-C"];length = [string length];return  0;
}

这里调用了 alloc、initWithString:、length 等方法。

% nm TestU .objc_class_name_NSString
00003000 D _NXArgc
00003004 D _NXArgvU ___CFConstantStringClassReference
00002b98 T ___darwin_gcc3_preregister_frame_infoU ___keymgr_dwarf2_register_sectionsU ___keymgr_global
0000300c D ___progname
000025ec t __call_mod_init_funcs
000026ec t __call_objcInitU __cthread_init_routine
00002900 t __dyld_func_lookup
000028a8 t __dyld_init_checkU __dyld_register_func_for_add_imageU __dyld_register_func_for_remove_image
...

可以看到,这里没有alloc、initWithString:、length3个方法的符号。所以,即使我们添加了新的方法,也可以在任何新旧系统中运 行。当然,函数调用之前,需要使用 respondsToSelector: 来确定方法是否存在。正是这样的特性,使得程序可以运行时动态地查询要执行的方法,提高了 Objective-C 语言的柔韧性。

Target-Action Paradigm

Objective-C 语言中,GUI控件对象间的通信利用 Target-Action Paradigm。不像其他事件驱动的 GUI 系统实现的那样,需要以回调函数的形式注册消息处理函数(Win32/MFC,Java AWT, X Window)。Target-Action Paradigm 完全是面向对象的事件传递机制。

例如用户点击菜单的事件,用Target-Action Paradigm来解释就是,调用菜单中被设定目标的Action。这个Action对应的方法不一定需要实现。目标与Action的指定与方法的实现没有关系,源代码编译的时候不会检测,只是在运行时确认(参考前面消息传送的机制)。

运行时,通过respondsToSelector: 方法来检查实现的情况。如果有实现,那么使用performSelector:withObject:来调用具体的Action,像是下面的代码:

// 目标对象
id target;
// 具体Action的 selector
SEL action;
...// 确认目标是否实现Action
if ([target respondsToSelector:actioin]) {// 调用具体Action[target performSelector:action withObject:self];
}
通过这样的架构,利用 setTarget: 可以更该其他的目标,或者 setAction: 变换不同的Action。实现动态的方法调用。

类型

常量实例

NSlog字符

Char

‘a’,’/n’

%c

Short int

--

%hi,%hx,%ho

Unsigned short int

--

%hu,%hx,%ho

Int

12,-97,0xFFE0,0177

%i,%x,%o

Unsigned int

12u,100U,0xFFu

%u,%x,%o

Long int

12L,-200l,0xffffL

%li,%lx,%lo

Unsigned long int

12UL,100ul,0xffeeUL

%lu,%lx,%lo

Long long int

0xe5e5c5e5LL,500ll

%lli,%llx,%llo

Unsigned long long int

12ull,0xffeeULL

%llu,%llx,%llo

Float

12.34f,3.1e-5f,

%f,%e,%g,%a

Double

12.34,3.1e-5,0x.1p3

%f,%e,%g,%a

Long double

12.34l,3.1e-5l

%Lf,%Le,%Lg

id

nil

%p

NSLog的格式如下所示:

  • %@     对象
  • %d, %i 整数
  • %u     无符整形
  • %f     浮点/双字
  • %x, %X 二进制整数
  • %o     八进制整数
  • %zu    size_t
  • %p     指针
  • %e     浮点/双字 (科学计算)
  • %g     浮点/双字
  • %s     C 字符串
  • %.*s   Pascal字符串
  • %c     字符
  • %C     unichar
  • %lld   64位长整数(long long)
  • %llu   无符64位长整数
  • %Lf    64位双字

Objective-C中的一些特殊的数据类及NSLog的输出格式相关推荐

  1. python vector 初始化_一文带你走进Python中的数据类

    全文共2607字,预计学习时长14分钟 图源:unsplash 数据类适用于Python3.7或更高版本,它不仅可以用作数据容器,还可以编写样板代码,简化创建类的过程. 创建第一个数据类 创建一个数据 ...

  2. 使用Mutability Detector对Java数据类的不变性进行单元测试

    在我们所有的项目中,我们使用的数据类根据定义包含数据(字段),但不包含(业务)逻辑. 根据最佳编码实践,数据类最好应该是不可变的,因为不可变性意味着线程安全. 这里的主要参考是Joshua Bloch ...

  3. kotlin 读取json文件_Kotlin数据类及json解析

    通过数据类,可以方便地得到很多有趣的函数,一部分是来自属性,比如编写getter和setter函数,还有下面这些函数:equals(): 比较两个对象的属性来确保他们是相同的. hashCode(): ...

  4. matlab中normfit,MATLAB中如何得到一组统计数据的分布特征

    MATLAB中如何得到一组统计数据的分布特征 我想通过MATLAB知道一组统计数据的分布特征,如属于哪种分布,均值方差等,请问如何实现?谢谢! function f=p_judge(A,alpha) ...

  5. Postgresql:删除及查询字段中包含单引号的数据

    Postgresql:删除及查询字段中包含单引号的数据 1. 假设pg表t_info的属性att,值为固定的:'test' 2. 假设值为不固定的,'abcde' 参考 1. 假设pg表t_info的 ...

  6. 在python中使用json格式存储数据

    在python中使用json格式存储数据 代码如下: import jsonlist1 = [{'A': [1, 2, 3, 4, 5, 6], 'B': [3, 4, 5, 6, 7]},{'C': ...

  7. python pandas库读取excel/csv中指定行或列数据详解

    通过阅读表格,可以发现Pandas中提供了非常丰富的数据读写方法,下面这篇文章主要给大家介绍了关于python利用pandas库读取excel/csv中指定行或列数据的相关资料,需要的朋友可以参考下 ...

  8. pandas使用dropna函数删除dataframe中全是缺失值的数据列(drop columns with all missing values in dataframe)

    pandas使用dropna函数删除dataframe中全是缺失值的数据列(drop columns with all missing values in dataframe) 目录

  9. R语言ggplot2可视化:使用长表数据(窄表数据)( Long Data Format)可视化多个时间序列数据、在同一个可视化图像中可视化多个时间序列数据(Multiple Time Series)

    R语言ggplot2可视化:使用长表数据(窄表数据)( Long Data Format)可视化多个时间序列数据.在同一个可视化图像中可视化多个时间序列数据(Multiple Time Series) ...

最新文章

  1. 判断两个树是否相等和判断tree1是否包含tree2 python实现
  2. sap事务代码_SAP事务码太多,记不住怎么办?
  3. iOS之深入解析CocoaPods的插件机制和如何加载插件整合开发工具
  4. 解决客户端从服务器请求数据乱码问题
  5. 大剑无锋之TCP和HTTP的区别【面试推荐】
  6. python3(十五)np.linalg.norm
  7. python清除列表内容_Python 列表的清空方式
  8. 在加州最古老的监狱教课是种什么样的体验?
  9. mysql ssh错误_通过SSH隧道连接时,MySQL访问被拒绝错误
  10. NeurIPS 2019论文盘点:谷歌系最多,国内清华第一
  11. spring cloud 微服务的版本介绍与内部组件详解
  12. [Android Pro] 告别编译运行 ---- Android Studio 2.0 Preview发布Instant Run功能
  13. 运算符优先级(cpp/c)
  14. 黑客之门的魅力:感染与加载
  15. 国产操作系统之优麒麟安装
  16. html messagebox确定取消,Element MessageBox弹框的详细使用
  17. python gpio 接口_树莓派GPIO接口常见的命令
  18. 组织行为学笔记(4)——人格与价值观
  19. flutter图标按钮_flutter 按钮封装 带图片的按钮,带边框的按钮,纯文字的按钮
  20. django 快速实现文件上传(django2.0)

热门文章

  1. 设置jstree只展示到2级_你做的私域流量属于什么级别?80%的商家都还只在第2级...
  2. cad插件加载bplot成功用不了_新手入门,学习CAD必须掌握,教你使用标注命令,绘图效率翻一倍...
  3. oracle substr(table),oracle中的substr()函数
  4. [Unity] 在 3DsMax 中将骨骼调整为适应 Unity 的 Humanoid 的结构的记录
  5. Ubuntu系统显卡驱动、CUDA、CUDNN安装(二CUDA和CUDNN)
  6. 部分Caffe 报错解决方案
  7. 搜狗输入法电脑版_搜狗输入法Mac版更新:找不到哪里下载?看这里
  8. 华为ac控制器web配置手册_欧姆龙AC伺服系统1S系列产品型号说明及功能介绍
  9. 流量卡官网源码【全解无后门】 修复添加教程
  10. CSS框架+响应式设计