ios学习路线—Objective-C(Runtime 协议与分类)

转载 2016年06月17日 01:01:40

Objective-C中的分类允许我们通过给一个类添加方法来扩充它(但是通过category不能添加新的实例变量),并且我们不需要访问类中的代码就可以做到。

Objective-C中的协议是普遍存在的接口定义方式,即在一个类中通过@protocol定义接口,在另外类中实现接口,这种接口定义方式也成为“delegation”模式,@protocol声明了可以被其他任何方法类实现的方法,协议仅仅是定义一个接口,而由其他的类去负责实现。

在本章中,我们来看看runtime对分类与协议的支持。

基础数据类型
Category
Category是表示一个指向分类的结构体的指针,其定义如下:

typedef struct objc_category *Category;

struct objc_category {
    char *category_name                          OBJC2_UNAVAILABLE; // 分类名
    char *class_name                             OBJC2_UNAVAILABLE; // 分类所属的类名
    struct objc_method_list *instance_methods    OBJC2_UNAVAILABLE; // 实例方法列表
    struct objc_method_list *class_methods       OBJC2_UNAVAILABLE; // 类方法列表
    struct objc_protocol_list *protocols         OBJC2_UNAVAILABLE; // 分类所实现的协议列表
}

这个结构体主要包含了分类定义的实例方法与类方法,其中instance_methods列表是objc_class中方法列表的一个子集,而class_methods列表是元类方法列表的一个子集。

Protocol
Protocol的定义如下:

typedef struct objc_object Protocol;

我们可以看到,Protocol其中实就是一个对象结构体。

Runtime并没有在

@interface RuntimeCategoryClass : NSObject
- (void)method1;
@end

@interface RuntimeCategoryClass (Category)
- (void)method2;
@end

@implementation RuntimeCategoryClass

- (void)method1 {

}

@end

@implementation RuntimeCategoryClass (Category)

- (void)method2 {

}

@end

NSLog(@"测试objc_class中的方法列表是否包含分类中的方法");
unsigned int outCount = 0;
Method *methodList = class_copyMethodList(RuntimeCategoryClass.class, &outCount);
for (int i = 0; i < outCount; i++) {
    Method method = methodList[i];
    const char *name = sel_getName(method_getName(method));
    NSLog(@"RuntimeCategoryClass's method: %s", name);
    if (strcmp(name, sel_getName(@selector(method2)))) {
        NSLog(@"分类方法method2在objc_class的方法列表中");
    }
}

其输出是:

2014-11-08 10:36:39.213 [561:151847] 测试objc_class中的方法列表是否包含分类中的方法
2014-11-08 10:36:39.215 [561:151847] RuntimeCategoryClass's method: method2
2014-11-08 10:36:39.215 [561:151847] RuntimeCategoryClass's method: method1
2014-11-08 10:36:39.215 [561:151847] 分类方法method2在objc_class的方法列表中

而对于Protocol,runtime提供了一系列函数来对其进行操作,这些函数包括:

// 返回指定的协议
Protocol * objc_getProtocol ( const char *name );

// 获取运行时所知道的所有协议的数组
Protocol ** objc_copyProtocolList ( unsigned int *outCount );

// 创建新的协议实例
Protocol * objc_allocateProtocol ( const char *name );

// 在运行时中注册新创建的协议
void objc_registerProtocol ( Protocol *proto );

// 为协议添加方法
void protocol_addMethodDescription ( Protocol *proto, SEL name, const char *types, BOOL isRequiredMethod, BOOL isInstanceMethod );

// 添加一个已注册的协议到协议中
void protocol_addProtocol ( Protocol *proto, Protocol *addition );

// 为协议添加属性
void protocol_addProperty ( Protocol *proto, const char *name, const objc_property_attribute_t *attributes, unsigned int attributeCount, BOOL isRequiredProperty, BOOL isInstanceProperty );

// 返回协议名
const char * protocol_getName ( Protocol *p );

// 测试两个协议是否相等
BOOL protocol_isEqual ( Protocol *proto, Protocol *other );

// 获取协议中指定条件的方法的方法描述数组
struct objc_method_description * protocol_copyMethodDescriptionList ( Protocol *p, BOOL isRequiredMethod, BOOL isInstanceMethod, unsigned int *outCount );

// 获取协议中指定方法的方法描述
struct objc_method_description protocol_getMethodDescription ( Protocol *p, SEL aSel, BOOL isRequiredMethod, BOOL isInstanceMethod );

// 获取协议中的属性列表
objc_property_t * protocol_copyPropertyList ( Protocol *proto, unsigned int *outCount );

// 获取协议的指定属性
objc_property_t protocol_getProperty ( Protocol *proto, const char *name, BOOL isRequiredProperty, BOOL isInstanceProperty );

// 获取协议采用的协议
Protocol ** protocol_copyProtocolList ( Protocol *proto, unsigned int *outCount );

// 查看协议是否采用了另一个协议
BOOL protocol_conformsToProtocol ( Protocol *proto, Protocol *other );

1.objc_getProtocol函数,需要注意的是如果仅仅是声明了一个协议,而未在任何类中实现这个协议,则该函数返回的是nil。

2.objc_copyProtocolList函数,获取到的数组需要使用free来释放

3.objc_registerProtocol函数,创建一个新的协议后,必须调用该函数以在运行时中注册新的协议。协议注册后便可以使用,但不能再做修改,即注册完后不能再向协议添加方法或协议

需要强调的是,协议一旦注册后就不可再修改,即无法再通过调用protocol_addMethodDescription、protocol_addProtocol和protocol_addProperty往协议中添加方法等。

小结
Runtime并没有提供过多的函数来处理分类。对于协议,我们可以动态地创建协议,并向其添加方法、属性及继承的协议,并在运行时动态地获取这些信息。

Object-C非正式协议与正式协议的区别

这两个概念困扰我很久了,一直都很像搞清楚到非正式协议和正式协议有什么区别和联系,下面结合网上的资料和自己的看法谈谈这个问题。 一、非正式协议 显然这个名词是相对于正式协议而言的。在解释非正...
  • braddoris
  • braddoris
  • 2014年04月28日 09:32
  • 8789

iOS开发入门教程之Objective-C · 协议(Protocols)

上一篇:iOS开发入门教程之Objective-C · 引用计数 前言 本文将介绍Objective-C的一个重要概念——协议(Protocols),iOS编程将经常使用到协议。比如在iOS中经常...
  • taonull
  • taonull
  • 2015年02月28日 16:49
  • 1210

Objective-C Runtime 运行时之一:类与对象

Objective-C语言是一门动态语言,它将很多静态语言在编译和链接时期做的事放到了运行时来处理。这种动态语言的优势在于:我们写代码时能够更具灵活性,如我们可以把消息转发给我们想要的对象,或者随意交...
  • hengshujiyi
  • hengshujiyi
  • 2014年10月31日 17:05
  • 7972

ios学习路线—Objective-C(Runtime方法与消息)

基础数据类型 SEL又叫选择器,是表示一个方法的selector的指针,其定义如下:typedef struct objc_selector *SEL;objc_selector结构体的详细定义没有...
  • dubo413392749
  • dubo413392749
  • 2016年06月16日 23:01
  • 97

ios学习路线—Objective-C(Runtime成员变量与属性)

类型编码(Type Encoding) 作为对Runtime的补充,编译器将每个方法的返回值和参数类型编码为一个字符串,并将其与方法的selector关联在一起。这种编码方案在其它情况下也是非常有用...
  • dubo413392749
  • dubo413392749
  • 2016年06月16日 18:21
  • 184

ios学习路线—Objective-C(检查内存泄露方式(Instruments))

Time Profiler(时间探测器) 现在开始运行app,搜索一些图片,这时候你发现查找一个结果太慢了,而且搜索结果列表页面滚动起来也是让人无法忍受的, 首先,确保工具栏中的视图选择有选择的所...
  • dubo413392749
  • dubo413392749
  • 2016年06月15日 15:50
  • 1110

ios学习路线—Objective-C(检查内存泄露方式(Static Analyzer))

静态分析错误提示分类 Xocde的静态代码分析工具会接卸项目的源代码,并以以下集中错误类型加以标识: 1.逻辑缺陷,例如访问未初始化的变量或空指针的解引用 2.内存管理缺陷,如内存泄露 3.无...
  • dubo413392749
  • dubo413392749
  • 2016年06月15日 14:50
  • 731

ios学习路线—Objective-C(装箱和拆箱)

概述 从前面的博文我们也可以看到,数组和字典中只能存储对象类型,其他基本类型和结构体是没有办法放到数组和字典中的,当然你也是无法给它们发送消息的也就是说有些NSObject的方法是无法调用的,这个时...
  • dubo413392749
  • dubo413392749
  • 2016年06月15日 13:51
  • 252

ios学习路线—Objective-C(生命周期)

视图控制器生命周期 当一个视图控制器被创建,并在屏幕上显示的时候。 代码的执行顺序 1.alloc 创建对象,分配空间 2...
  • dubo413392749
  • dubo413392749
  • 2016年06月16日 01:52
  • 450

iOS学习笔记10—Objective-C的运行时编程(Runtime Programming)

说实话,当我接触iOS开发这么长时间以来 我还真没了解过Objective-C的运行时编程(Runtime Programming),今天特意在网上搜了下,原来这么深奥啊 表示现在理解不了,先转走了再...
  • jr773
  • jr773
  • 2013年08月12日 17:08
  • 6207
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:ios学习路线—Objective-C(Runtime 协议与分类)
举报原因:
原因补充:

(最多只允许输入30个字)