深入解析alloc/retain/release/dealloc实现
首先通过GNUstep上得源码来叙述各个函数的实现(GNUstep是Cocoa框架的互换框架,二者的行为和实现方式很相似)
GNUstep源码中NSObject类的alloc方法:
id = obj = [NSObject alloc];
/**********************************/
+(id) alloc{
return [self allocWithZone : NSDefaultMallocZone()];
}
+(id) allocWithZone:(NSZone *) z{
return NSAllocateObject(self,0,z);
}
//NSZone是为了防止内存碎片化而引入的结构,对内存分配的区域本身进行多重画管理
NSAllocateObject函数例如以下:
struct obj_layout{
NSUInteger retained;
}
inline id NSAllocateObject(Class aClass,NSUInteger extraBytes,NSZone *zone){
int size =
计算容纳对象所需的内存;
id new = NSZoneMalloc(Zone ,size);
memset(new ,0 ,size);
new = (id) &((struct obj_layout *) new)[1];
}
NSAllocateObject函数通过调用NSZoneMalloc函数来分配存放对象所需的内存空间,之后将该内存空间置为0,最后返回对象而使用的指针.
/**************************/
-(NSUInteger) retainCount{
return NSExtraRefCount(self) + 1;
}
inline NSUInteger NSExtraRefCount(id anObject){
return ((struct obj_layout *) anObject)[-1].retained;
//由对象寻址找到对象内存头部,从而訪问当中的retained变量
}
/***************************/
-(id) retain{
NSInCrementExtraRefCount(self);
return self;
}
inline void NSInCrementExtraRefCount(id anObject){
if((( struct obj_layout *) anObject)[-1].retained == UINT_MAX - 1){
[NSException raise:NSInternalInconsistencyException format:@“NSIncrementExtraRefCount()ask to increment too far”];
(( struct obj_layout *)anObject) [-1].retained++;
}
}
/********************************/
-(void) release{
if(NSDecrementExtraREfCountWasZero(self))
[self dealloc];
}
BOOL NSDecrementExtraREfCountWasZero(id anObject){
if((struct obj_layout *) anObject)[-1].retained == 0){
return YES;
}else{
(struct obj_layout *) anObject)[-1].retained—;
return NO;
}
}
/**********************************/
-(void ) dealloc{
NSDeallocateObject(self);
}
inline void NSDeallocateObject(id anObject){
struct obj_layout *o = &((struct obj_layout *) anObject)[-1];
free(o);
}
以上就是GNUstep中的实现
//////////////////////////////////////////////////////////////////////////
苹果的实现
通过在NSObject类的alloc类方法上设置断点,查看其运行的函数为:
+alloc
+allocWithZone
class_createInstance
calloc
各个方法都调用了_CFDoExternRefOperation函数
/************************************************/
int _CFDoExternRefOperation(uintptr_t op,id obj){
CFBasicHashRef table =
取得对象所相应的散列表(obj);
int count;
int count;
switch (op) {
case OPERATION_retainCount:
count = CFBasicHashGetCountOfKey(table, obj);
return count;
case OPERATION_retain:
CFBasicHashAddValue(table, obj);
return obj;
case OPERATION_release:
count = CFBasicHashRemoveValue(table, obj);
return 0 == count;
}
}
何为散列表
散列表(也叫哈希表)是一种查找算法,与链表、树等算法不同的是。散列表算法在查找时不须要进行一系列和keyword(keyword是数据元素中某个数据项的值,用以标识一个数据元素)的比較操作。
散列表算法希望能尽量做到不经过不论什么比較。通过一次存取就能得到所查找的数据元素,因而必需要在数据元素的存储位置和它的keyword(可用key表示)之间建立一个确定的相应关系,使每一个keyword和散列表中一个唯一的存储位置相相应。因此在查找时,仅仅要依据这个相应关系找到给定keyword在散列表中的位置就可以。这样的相应关系被称为散列函数(可用h(key)表示)。
依据设定的散列函数h(key)和处理冲突的方法将一组keywordkey映像到一个有限的连续的地址区间上,并以keyword在地址区间中的像作为数据元素在表中的存储位置,这样的表便被称为散列表,这一映像过程称为散列,所得存储位置称为散列地址。
由上可知苹果是将计数器保存在引用记数表的记录中
优点:
(1) 对象用内存块的分配无需考虑内存头部
(2) 引用计数表个记录中存有内存块地址,可追溯到个内存块.
深入解析alloc/retain/release/dealloc实现的更多相关文章
- alloc retain release函数
- OC语法6——内存管理之引用计数器(retain,release)
OC内存管理: 一.引用计数器: Java有垃圾回收机制(Garbage Collection,GC).也就是说当我们创建对象后,不需要考虑回收内存的事,Java的垃圾回收机制会自动销毁该对象,回收它 ...
- (转)IOS内存管理 retain release
obj-c本质就是"改进过的c语言",大家都知道c语言是没有垃圾回收(GC)机制的(注:虽然obj-c2.0后来增加了GC功能,但是在iphone上不能用,因此对于iOS平台的程序 ...
- ios中的关键词retain release
内存分析 在函数中只要用new alloc copy 这样的分配空间时 则计算器retain就要为一 每调用一次就要加一 在.m文件中引用手动计数时 一定要调用[super dealloc]这 ...
- 内存管理1 retain & release
内存管理法则 1:谁创建谁释放alloc /new/ copy------>release/autorelease.一一对应,不是你创建的就不用你释放. 2:除了alloc /new/ copy ...
- Alloc and release
https://developer.apple.com/library/ios/documentation/Cocoa/Conceptual/MemoryMgmt/Articles/mmPractic ...
- 浅谈cocosd之autorelease\retain\release的理解
三种情况,引出问题: 1) new出来的对象需要释放,而释放时,如果有其他人引用了这个对象,再次使用这个对象时,则会出现野指针情况. ==> 于是出现了引用计数的释放管理机制. 2) 对于一 ...
- Linked dylibs built for GC-only but object files built for retain/release for architecture x86_64
编译开源Xcode 插件 SCXcodeSwitchExpander 源码地址: https://github.com/stefanceriu/SCXcodeSwitchExpander 编译环境:X ...
- 2016 - 2 - 19 ARC内存管理知识总结(一,arc基本概念及alloc等方法的实现)
一. ARC的基本概念 1. 在objc中采用automatic reference counting 机制, 让编译器来进行内存管理.在降低程序崩溃,内存管理泄漏等风险的同时,很大程度减少了程序员的 ...
随机推荐
- matlab 分析wav波形
[x,fs,bits]=wavread('d.wav', [1 5000]); % sound(x, fs, bits); N = length(x); n = 0 : N-1; t = n/fs; ...
- Git 的使用教程
Git是一个开源的分布式版本控制系统,用以有效.高速的处理从很小到非常大的项目版本管理. 安装Git 1.方法一:安装homebrew,然后通过homebrew安装Git. 第一步:在终端输入如下命令 ...
- Appium+python自动化12-appium元素定位【转载】
前言 appium定位app上的元素,可以通过id,name.class这些属性定位到 一.id定位 1.appium的id属性也就是通过UI Automator工具查看的resource-id属性
- MSSQL—字符串分离(Split函数)
前面提到了记录合并,有了合并需求肯定也会有分离需求,说到字符串分离,大家肯定会想到SPLIT函数,这个在.NET,Java和JS中都有函数,很可惜在SQL SERVER中没有,我们只能自己来写这么一个 ...
- 大牛教你如何循序渐进,有效的学习JavaScript?
首先要说明的是,咱现在不是高手,最多还是一个半桶水,算是入了JS的门.谈不上经验,都是一些教训. 这个时候有人要说,“靠,你丫半桶水,凭啥教我们”.您先别急着骂,先听我说! 你叫一个大学生去教小学数学 ...
- (转)Ubuntu安装g++-4.8
sudo add-apt-repository ppa:ubuntu-toolchain-r/test sudo apt-get update sudo apt-get install g++-4.8 ...
- Codeforces Beta Round #4 (Div. 2 Only) C. Registration system【裸hash/map】
C. Registration system time limit per test 5 seconds memory limit per test 64 megabytes input standa ...
- servlet方法
1.每一个Servlet都必须要实现Servlet接口,GenericServlet是个通用的.不特定于任何协议的Servlet,它实现了Servlet接口,而HttpServlet继承于Generi ...
- 拓扑排序+数学+DP【p1685】游览
Description 顺利通过了黄药师的考验,下面就可以尽情游览桃花岛了! 你要从桃花岛的西头开始一直玩到东头,然后在东头的码头离开.可是当你游玩了一次后,发现桃花岛的景色实在是非常的美丽!!!于是 ...
- android如何取消闹铃
取消闹钟: Intent intent = new Intent(context, TestReceiver.class); PendingIntent pi = PendingIntent.getB ...