iOS开发中的对象系统基础

简介:

[0] Outline

  -- [1] id和Class

  -- [2] 动态地操作类

  -- [3] 实例化


[1] id和Class

在Objective-C中有一个特别的数据类型作为对象标识符:id,可以指向任何类型的对象。

通过 “可以指向任何类型的对象” 这一描述,猜想id实际上是指向Objective-C对象系统中的基类(继承体系中的祖先结构)的指针,在运行时是指向对象内存布局的基类部分。


第一眼看到id数据类型,我联想到了Python中的PyObject结构:

typedef struct _object {
    int ob_refcnt;
    struct _typeobject *ob_type;
} PyObject;

该数据类型也是Python对象系统中的祖先类型,不过与id相对应的应该是PyObject *类型。


id数据类型是一个指向struct objc_object结构的指针:

typedef struct objc_class *Class;
typedef struct objc_object {
    Class isa;
} *id;

更确切地说,id是指向Class类型的指针,而Class又是指向struct objc_class结构的指针:

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;
};

至此,可以看到Objective-C对象系统的基石:struct objc_class结构:

     isa指针:指向该对象所属类型的类型对象(Class Object)。在Objective-C中,类也是用对象来表示的,而类的isa指针指向它的metaclass(存储静态成员变量和类方法)。
     super_class指针:指向父类。
     name:类名称。
     version:类的版本信息。
     info:运行期使用的标志位,比如0x1(CLS_CLASS)表示该类为普通class,0x2(CLS_META)表示该类为     metaclass。
     instance_size:实例大小,即内存所占空间。
     ivars:指向成员变量列表的指针。
     methodLists:根据标志位的不同可能指向不同,比如可能指向实例方法列表,或者指向类方法列表。
     cache:因为Objective-C的消息转发需要查找dispatch table甚至可能需要遍历继承体系,所以缓存最近使用的方法。
     protocols:类需要遵守的协议。

[2] 动态地操作类

由上知道了类也是一种对象,那么类对象也有一种类型,这种类型就是类的metaclass,因此类方法其实就是metaclass的成员方法,类和metaclass是配套出现的。

那么metaclass的isa指针和super_class指针怎么指向的呢?

如果metaclass对应基类,那么它的isa指向自身、super_class指针指向对应的类(基类);如果不是则isa指针指向基类的metaclass、super_class指针指向父类的metaclass。

基类的isa指针为nil。


这不禁让我又想起了Python中类似的设计思想,比如整型数字2的类型是PyIntObject,而PyIntObject类的类型是PyTypeObject,PyTypeObject的类型是PyTypeObject。最终止于此。

同样地,Python中也有metaclass的存在。


知道了类的表示结构,我们可以动态地对类进行操作,加深理解。

//
//  main.m
//  HelloOC
//
//  Created by Jason Lee on 12-2-17.
//  Copyright (c) 2012年 XXX. All rights reserved.
//

#import <Foundation/Foundation.h>

#import <objc/objc.h>
#import <objc/runtime.h>

void selfIntro(idself, SEL_cmd);

int main (int argc, constchar * argv[])
{
    
    @autoreleasepool {
        //Create class & metaclass as a pair
        Class demoClass = objc_allocateClassPair([NSObjectclass], "NSDemo", 0);
        
        BOOL isOk = NO;
        
        //"v@:" indicates the function type : v - void, @ - object, : - SEL
        //向methodLists指向添加数据
        isOk = class_addMethod(demoClass, @selector(intro), (IMP)&selfIntro, "v@:");
        isOk == YES ? nil : NSLog(@"failed on class_addMethod\n");
        
        //向ivars指向添加数据
        isOk = class_addIvar(demoClass,"myVar", sizeof(id), log2(sizeof(id)), "@"); 
        isOk == YES ? nil : NSLog(@"failed on class_addIvar\n");
        
        objc_registerClassPair(demoClass);
        
        id demo = class_createInstance(demoClass, 0);
        
        if ([demo respondsToSelector:@selector(intro)]) {
            [demo performSelector:@selector(intro)];
        }
        
        object_setInstanceVariable(demo, "myVar", nil);
        void *outValue = (void *)0x1;
        object_getInstanceVariable(demo, "myVar", &outValue);
        if (nil == outValue) {
            NSLog(@"Hello,nil\n");  //printed
        }
        
        object_dispose(demo);
    }
    
    return0;
}

void selfIntro(idself, SEL_cmd) {
    NSLog(@"selfIntro\n");
    
    Class isa = self->isa;  //At first, isa is class NSDemo
    while (1) {
        if (isa == isa->isa) {  //Finally, NSObject's metaclass points to itself
            NSLog(@"%p, %@", isa, objc_getMetaClass(class_getName([isa class])));
            break;
        }
        
        NSLog(@"%p, %@", isa, objc_getMetaClass(class_getName([isa class])));
        isa = isa->isa; //Then, isa is assigned to NSDemo's metaclass
    }
}


[3] 实例化

要实例化出一个对象,需要根据类的定义来进行。类的定义包括类名称、数据和操作数据的方法。

编译过程,类的信息会被记录下来,供runtime system使用,同时编译器会为每个类创建唯一的一个对象来表示它:class object。如果从功能上说,class object也是factory object,它能够执行类方法,负责创建实例。


从这个角度来看,我在思考class object是否就是metaclass,但是不能确认。

Apple官方文档TOCPL中Class Objects一章有这么一句,“a class object keeps the prototype of a class instance”,但metaclass并不能作为实例原型。

于是我认为class object是运行时class和metaclass结合起来的受限表现,能够访问编译器捕捉下来的类信息,没有成员变量,不能调用成员方法,但是可以执行类方法。

从源码层次来看,class object是由类名来表示,比如下述代码中:

int version = [NSString version];

NSString代表着class object。


首先,class object会被runtime system发送initialize消息进行初始化,让其做好运行时的准备,比如初始化静态变量。

之后,可以调用class object的方法(类方法)alloc来为新的实例对象分配内存空间,将其所有变量初始化为0,isa指针指向所属类。

最后再调用init函数进行必要的初始化。


写到这里的时候,突然要变更办公位置,思路被打断了,就先写到这里。

最后,留一个在SO上面看到的问题,我也疑惑,只能有几分猜测:http://stackoverflow.com/questions/8847146/whats-is-methodlists-attribute-of-the-structure-objc-class-for

[这篇文章是我对SO上的问题的解答:http://blog.csdn.net/jasonblog/article/details/7303618]


[Last Updated] 2012-03-17


Jason Lee @ 杭州

博客:http://blog.csdn.net/jasonblog

微博:http://weibo.com/jasonmblog

GitHub: https://github.com/siqin
目录
相关文章
|
3天前
|
iOS开发 开发者
uniapp开发ios打包Error code = -5000 Error message: Error: certificate file(p12) import failed!报错问题如何解决
uniapp开发ios打包Error code = -5000 Error message: Error: certificate file(p12) import failed!报错问题如何解决
uniapp开发ios打包Error code = -5000 Error message: Error: certificate file(p12) import failed!报错问题如何解决
|
2月前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
28天前
|
iOS开发 开发者 MacOS
深入探索iOS开发中的SwiftUI框架
【10月更文挑战第21天】 本文将带领读者深入了解Apple最新推出的SwiftUI框架,这一革命性的用户界面构建工具为iOS开发者提供了一种声明式、高效且直观的方式来创建复杂的用户界面。通过分析SwiftUI的核心概念、主要特性以及在实际项目中的应用示例,我们将展示如何利用SwiftUI简化UI代码,提高开发效率,并保持应用程序的高性能和响应性。无论你是iOS开发的新手还是有经验的开发者,本文都将为你提供宝贵的见解和实用的指导。
120 66
|
14天前
|
存储 监控 API
app开发之安卓Android+苹果ios打包所有权限对应解释列表【长期更新】-以及默认打包自动添加权限列表和简化后的基本打包权限列表以uniapp为例-优雅草央千澈
app开发之安卓Android+苹果ios打包所有权限对应解释列表【长期更新】-以及默认打包自动添加权限列表和简化后的基本打包权限列表以uniapp为例-优雅草央千澈
|
1月前
|
开发框架 Android开发 iOS开发
安卓与iOS开发中的跨平台策略:一次编码,多平台部署
在移动应用开发的广阔天地中,安卓和iOS两大阵营各占一方。随着技术的发展,跨平台开发框架应运而生,它们承诺着“一次编码,到处运行”的便捷。本文将深入探讨跨平台开发的现状、挑战以及未来趋势,同时通过代码示例揭示跨平台工具的实际运用。
121 3
|
1月前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
2月前
|
存储 前端开发 Swift
探索iOS开发:从新手到专家的旅程
本文将带您领略iOS开发的奇妙之旅,从基础概念的理解到高级技巧的掌握,逐步深入iOS的世界。文章不仅分享技术知识,还鼓励读者在编程之路上保持好奇心和创新精神,实现个人成长与技术突破。
|
2月前
|
传感器 iOS开发 UED
探索iOS生态系统:从App Store优化到用户体验提升
本文旨在深入探讨iOS生态系统的多个方面,特别是如何通过App Store优化(ASO)和改进用户体验来提升应用的市场表现。不同于常规摘要仅概述文章内容的方式,我们将直接进入主题,首先介绍ASO的重要性及其对开发者的意义;接着分析当前iOS平台上用户行为的变化趋势以及这些变化如何影响应用程序的设计思路;最后提出几点实用建议帮助开发者更好地适应市场环境,增强自身竞争力。
|
2月前
|
安全 Android开发 iOS开发
深入探讨Android与iOS系统的差异及未来发展趋势
本文旨在深入分析Android和iOS两大移动操作系统的核心技术差异、用户体验以及各自的市场表现,进一步探讨它们在未来技术革新中可能的发展方向。通过对比两者的开放性、安全性、生态系统等方面,本文揭示了两大系统在移动设备市场中的竞争态势和潜在变革。
|
2月前
|
安全 Android开发 数据安全/隐私保护
深入探索Android与iOS系统安全性的对比分析
在当今数字化时代,移动操作系统的安全已成为用户和开发者共同关注的重点。本文旨在通过比较Android与iOS两大主流操作系统在安全性方面的差异,揭示两者在设计理念、权限管理、应用审核机制等方面的不同之处。我们将探讨这些差异如何影响用户的安全体验以及可能带来的风险。
55 1