В отношении вашего примера необходимо помнить две вещи.
Как объяснил MByD, доступ к объекту, который был освобожден, является неопределенным поведением.Это может или не может привести к сбою вашей программы - это зависит от того, была ли память, которая была выделена для этого объекта, была повторно использована и что было помещено туда.В некоторых случаях память еще не была повторно использована (и вы могли бы подумать, что объект все еще жив, но этот объект на самом деле является призрачным объектом ), в других случаях память могла быть повторно использована другой Целью-C объект (программа не будет аварийно завершать работу, но вы увидите другой объект), и в других случаях память может быть повторно использована чем-то, что не является объектом Objective-C (в этом случае программа, скорее всего, - но не обязательно- крах).
Ваш строковый объект является константной строкой.Как объяснено в ответе на этот вопрос и его комментариях, константная строка никогда не освобождается.Когда вы отправляете -[NSString initWithString:]
, передавая константную строку в качестве аргумента, Какао возвращает исходную константную строку, поэтому ваш код фактически совпадает с NSString *myString = @"myString";
.Это внутренняя деталь реализации Какао, хотя.Производственный код должен всегда учитывать, что объекты, возвращаемые +alloc
(и, как правило, последующие -init
), принадлежат вызывающей стороне, должны быть освобождены, когда вызывающая сторона больше не заинтересована в них, и не будут доступны после освобождения.
Ради эксперимента попробуйте следующий код:
#import <Foundation/Foundation.h>
#include <stdio.h>
int main(void) {
[NSAutoreleasePool new];
NSString *s1 = [[NSString alloc] initWithString:@"myString"];
NSString *s2 = [[NSString alloc] initWithString:@"myString"];
NSString *s3 = [[NSString alloc] initWithString:@"myString"];
NSString *s4 = [[NSString alloc] initWithString:@"myString"];
printf("s1 = %p\n", s1);
printf("s2 = %p\n", s2);
printf("s3 = %p\n", s3);
printf("s4 = %p\n", s4);
[s1 release];
[s2 release];
[s3 release];
[s4 release];
return 0;
}
Концептуально, s1
, s2
, s3
, s4
быть разными объектами.Однако, запустив эту программу, вы можете видеть, что это фактически один и тот же объект (они имеют одинаковый адрес):
$ ./a.out
s1 = 0x10080b090
s2 = 0x10080b090
s3 = 0x10080b090
s4 = 0x10080b090