object-c 三

记忆体管理

到目前为止我都刻意避开 Objective-C 的记忆体管理议题。你可以唿叫物件上的 dealloc,但是若物件裡包含其他物件的指标的话,要怎么办呢?要释放那些物件所佔据的记忆体也是一个必须关注的问题。当你使用 Foundation framework 建立 classes 时,它如何管理记忆体?这些稍后我们都会解释。

注意:之前所有的範例都有正确的记忆体管理,以免你混淆。

Retain and Release(保留与释放)

Retain 以及 release 是两个继承自 NSObject 的物件都会有的 methods。每个物件都有一个内部计数器,可以用来追踪物件的 reference 个数。如果物件有 3 个 reference 时,不需要 dealloc 自己。但是如果计数器值到达 0 时,物件就得 dealloc 自己。[object retain] 会将计数器值加 1(值从 1 开始),[object release] 则将计数器值减 1。如果唿叫 [object release] 导致计数器到达 0,就会自动 dealloc。

 
  1. Fraction.m
  2. ...
  3. -(void)dealloc{
  4. printf("Deallocingfraction\n");
  5. [superdealloc];
  6. }
  7. ...

 
  1. main.m
  2. #import"Fraction.h"
  3. #import
  4. intmain(intargc,constchar*argv[]){
  5. Fraction*frac1=[[Fractionalloc]init];
  6. Fraction*frac2=[[Fractionalloc]init];
  7. //printcurrentcounts
  8. printf("Fraction1retaincount:%i\n",[frac1retainCount]);
  9. printf("Fraction2retaincount:%i\n",[frac2retainCount]);
  10. //incrementthem
  11. [frac1retain];//2
  12. [frac1retain];//3
  13. [frac2retain];//2
  14. //printcurrentcounts
  15. printf("Fraction1retaincount:%i\n",[frac1retainCount]);
  16. printf("Fraction2retaincount:%i\n",[frac2retainCount]);
  17. //decrement
  18. [frac1release];//2
  19. [frac2release];//1
  20. //printcurrentcounts
  21. printf("Fraction1retaincount:%i\n",[frac1retainCount]);
  22. printf("Fraction2retaincount:%i\n",[frac2retainCount]);
  23. //releasethemuntiltheydeallocthemselves
  24. [frac1release];//1
  25. [frac1release];//0
  26. [frac2release];//0
  27. }

output

 
  1. Fraction1retaincount:1
  2. Fraction2retaincount:1
  3. Fraction1retaincount:3
  4. Fraction2retaincount:2
  5. Fraction1retaincount:2
  6. Fraction2retaincount:1
  7. Deallocingfraction
  8. Deallocingfraction

Retain call 增加计数器值,而 release call 减少它。你可以唿叫 [obj retainCount] 来取得计数器的 int 值。 当当 retainCount 到达 0,两个物件都会 dealloc 自己,所以可以看到印出了两个 "Deallocing fraction"。

Dealloc

当你的物件包含其他物件时,就得在 dealloc 自己时释放它们。Objective-C 的一个优点是你可以传递讯息给 nil,所以不需要经过一堆防错测试来释放一个物件。

AddressCard.h

 
  1. #import
  2. #import
  3. @interfaceAddressCard:NSObject{
  4. NSString*first;
  5. NSString*last;
  6. NSString*email;
  7. }
  8. -(AddressCard*)initWithFirst:(NSString*)f
  9. last:(NSString*)l
  10. email:(NSString*)e;
  11. -(NSString*)first;
  12. -(NSString*)last;
  13. -(NSString*)email;
  14. -(void)setFirst:(NSString*)f;
  15. -(void)setLast:(NSString*)l;
  16. -(void)setEmail:(NSString*)e;
  17. -(void)setFirst:(NSString*)f
  18. last:(NSString*)l
  19. email:(NSString*)e;
  20. -(void)setFirst:(NSString*)flast:(NSString*)l;
  21. -(void)print;
  22. @end
  23. AddressCard.m
  24. #import"AddressCard.h"
  25. #import
  26. @implementationAddressCard
  27. -(AddressCard*)initWithFirst:(NSString*)f
  28. last:(NSString*)l
  29. email:(NSString*)e{
  30. self=[superinit];
  31. if(self){
  32. [selfsetFirst:flast:lemail:e];
  33. }
  34. returnself;
  35. }
  36. -(NSString*)first{
  37. returnfirst;
  38. }
  39. -(NSString*)last{
  40. returnlast;
  41. }
  42. -(NSString*)email{
  43. returnemail;
  44. }
  45. -(void)setFirst:(NSString*)f{
  46. [fretain];
  47. [firstrelease];
  48. ffirst=f;
  49. }
  50. -(void)setLast:(NSString*)l{
  51. [lretain];
  52. [lastrelease];
  53. llast=l;
  54. }
  55. -(void)setEmail:(NSString*)e{
  56. [eretain];
  57. [emailrelease];
  58. eemail=e;
  59. }
  60. -(void)setFirst:(NSString*)f
  61. last:(NSString*)l
  62. email:(NSString*)e{
  63. [selfsetFirst:f];
  64. [selfsetLast:l];
  65. [selfsetEmail:e];
  66. }
  67. -(void)setFirst:(NSString*)flast:(NSString*)l{
  68. [selfsetFirst:f];
  69. [selfsetLast:l];
  70. }
  71. -(void)print{
  72. printf("%s%s<%S>
  73. ",[firstcString],
  74. [lastcString],
  75. [emailcString]);
  76. }
  77. -(void)dealloc{
  78. [firstrelease];
  79. [lastrelease];
  80. [emailrelease];
  81. [superdealloc];
  82. }
  83. @end
  84. main.m
  85. #import"AddressCard.h"
  86. #import
  87. #import
  88. intmain(intargc,constchar*argv[]){
  89. NSString*first=[[NSStringalloc]initWithCString:"Tom"];
  90. NSString*last=[[NSStringalloc]initWithCString:"Jones"];
  91. NSString*email=[[NSStringalloc]initWithCString:"[email protected]"];
  92. AddressCard*tom=[[AddressCardalloc]initWithFirst:first
  93. last:last
  94. email:email];
  95. //we'redonewiththestrings,sowemustdeallocthem
  96. [firstrelease];
  97. [lastrelease];
  98. [emailrelease];
  99. //printtoshowtheretaincount
  100. printf("Retaincount:%i\n",[[tomfirst]retainCount]);
  101. [tomprint];
  102. printf("\n");
  103. //freememory
  104. [tomrelease];
  105. return0;
  106. }

output

 
  1. Retaincount:1
  2. TomJones

如 AddressCard.m,这个範例不仅展示如何撰写一个 dealloc method,也展示了如何 dealloc 成员变数。

每个 set method 裡的叁个动作的顺序非常重要。假设你把自己当参数传给一个自己的 method(有点怪,不过确实可能发生)。若你先 release,「然后」才 retain,你会把自己给解构(destruct,相对于建构)!这就是为什么应该要 1) retain 2) release 3) 设值 的塬因。

通常我们不会用 C 形式字串来初始化一个变数,因为它不支援 unicode。下一个 NSAutoreleasePool 的例子会用展示正确使用并初始化字串的方式。

这只是处理成员变数记忆体管理的一种方式,另一种方式是在你的 set methods 裡面建立一份拷贝。

Autorelease Pool

当你想用 NSString 或其他 Foundation framework classes 来做更多程式设计工作时,你需要一个更有弹性的系统,也就是使用 Autorelease pools。

当开发 Mac Cocoa 应用程式时,autorelease pool 会自动地帮你设定好。

 
  1. main.m
  2. #import
  3. #import
  4. #import
  5. intmain(intargc,constchar*argv[]){
  6. NSAutoreleasePool*pool=[[NSAutoreleasePoolalloc]init];
  7. NSString*str1=@"constantstring";
  8. NSString*str2=[NSStringstringWithString:@"stringmanagedbythepool"];
  9. NSString*str3=[[NSStringalloc]initWithString:@"selfmanagedstring"];
  10. //printthestrings
  11. printf("%sretaincount:%x\n",[str1cString],[str1retainCount]);
  12. printf("%sretaincount:%x\n",[str2cString],[str2retainCount]);
  13. printf("%sretaincount:%x\n",[str3cString],[str3retainCount]);
  14. //freememory
  15. [str3release];
  16. //freepool
  17. [poolrelease];
  18. return0;
  19. }

output

 
  1. constantstringretaincount:ffffffff
  2. stringmanagedbythepoolretaincount:1
  3. selfmanagedstringretaincount:1

如果你执行这个程式,你会发现几件事:第一件事,str1 的 retainCount 为 ffffffff。

另一件事,虽然我只有 release str3,整个程式却还是处于完美的记忆体管理下,塬因是第一个常数字串已经自动被加到 autorelease pool 裡了。还有一件事,字串是由 stringWithString 产生的。这个 method 会产生一个 NSString class 型别的字串,并自动加进 autorelease pool。

千万记得,要有良好的记忆体管理,像 [NSString stringWithString: @"String"] 这种 method 使用了 autorelease pool,而 alloc method 如 [[NSString alloc] initWithString: @"String"] 则没有使用 auto release pool。

在 Objective-C 有两种管理记忆体的方法, 1) retain and release or 2) retain and release/autorelease。

对于每个 retain,一定要对应一个 release 「或」一个 autorelease。

下一个範例会展示我说的这点。

 
  1. Fraction.h
  2. ...
  3. +(Fraction*)fractionWithNumerator:(int)ndenominator:(int)d;
  4. ...
  5. Fraction.m
  6. ...
  7. +(Fraction*)fractionWithNumerator:(int)ndenominator:(int)d{
  8. Fraction*ret=[[Fractionalloc]initWithNumerator:ndenominator:d];
  9. [retautorelease];
  10. returnret;
  11. }
  12. ...
  13. main.m
  14. #import
  15. #import"Fraction.h"
  16. #import
  17. intmain(intargc,constchar*argv[]){
  18. NSAutoreleasePool*pool=[[NSAutoreleasePoolalloc]init];
  19. Fraction*frac1=[FractionfractionWithNumerator:2denominator:5];
  20. Fraction*frac2=[FractionfractionWithNumerator:1denominator:3];
  21. //printfrac1
  22. printf("Fraction1:");
  23. [frac1print];
  24. printf("\n");
  25. //printfrac2
  26. printf("Fraction2:");
  27. [frac2print];
  28. printf("\n");
  29. //thiscausesasegmentationfault
  30. //[frac1release];
  31. //releasethepoolandallobjectsinit
  32. [poolrelease];
  33. return0;
  34. }

output

 
  1. Fraction1:2/5
  2. Fraction2:1/3

在这个例子裡,此 method 是一个 class level method。在物件建立后,在它上面唿叫 了 autorelease。在 main method 裡面,我从未在此物件上唿叫 release。

这样行得通的塬因是:对任何 retain 而言,一定要唿叫一个 release 或 autorelease。物件的 retainCount 从 1 起跳 ,然后我在上面唿叫 1 次 autorelease,表示 1 - 1 = 0。当 autorelease pool 被释放时,它会计算所有物件上的 autorelease 唿叫次数,并且唿叫相同次数的 [obj release]。

如同註解所说,不把那一行註解掉会造成分段错误(segment fault)。因为物件上已经唿叫过 autorelease,若再唿叫 release,在释放 autorelease pool 时会试图唿叫一个 nil 物件上的 dealloc,但这是不允许的。最后的算式会变为:1 (creation) - 1 (release) - 1 (autorelease) = -1

管理大量暂时物件时,autorelease pool 可以被动态地产生。你需要做的只是建立一个 pool,执行一堆会建立大量动态物件的程式码,然后释放这个 pool。你可能会感到好奇,这表示可能同时有超过一个 autorelease pool 存在。

你可能感兴趣的:(object)