5

我试图了解 ARC 的工作方式,据我所知,我应该在这里做错事。这是我正在使用的代码:

界面:

@interface ViewController : UIViewController{

}

@property (strong, nonatomic) NSString * myString ; 
@property (weak, nonatomic) NSString * myPointer ;

执行:

 - (void)viewDidLoad{

     [super viewDidLoad];
     self.myString = @"Hello world!" ; // myString is strong
     self.myPointer = self.myString ; // myPointer var is weak

     [self performSelector:@selector(makeNilMyValue) withObject:nil afterDelay:1];    
     [self performSelector:@selector(printValues) withObject:nil afterDelay:2];    
}

 - (void) makeNilMyValue{
     self.myString = nil ; 
}

 - (void) printValues{
     NSLog(@"myString: %@", self.myString) ;
     NSLog(@"myPointer: %@", self.myPointer) ; 
 }

执行此操作后,我得到:

2012-02-26 11:40:41.652 test1[933:207] myString: (null)

2012-02-26 11:40:41.653 test1[933:207] myPointer: Hello world!

如果我没记错的话,由于 myPointer 很弱,它不应该保留对象的内容。因此,它应该显示 nil 而不是“Hello World!”。

我究竟做错了什么?

根据 Caleb 的回答,我创建了另一个弱指针,请参见下面的代码:

- (void)viewDidLoad{
    [super viewDidLoad];
    self.myString = @"Hello world!" ; // myString is strong
    self.myPointer = self.myString ; // myPointer var is weak
    self.myPointer2 = self.myString ; // myPointer2 var is weak

    [self performSelector:@selector(makeNilMyValue) withObject:nil afterDelay:1];    
    [self performSelector:@selector(printValues) withObject:nil afterDelay:2];    
}

- (void) makeNilMyValue{
    self.myPointer2 = @"value changed!" ;
    self.myString = nil ;

}

- (void) printValues{
    NSLog(@"myString: %@", self.myString) ;
    NSLog(@"myPointer: %@", self.myPointer) ;
}

关键是我仍然得到了与以前相同的答案:

2012-02-26 12:08:13.426 test1[1333:207] myString: (null)
2012-02-26 12:08:13.427 test1[1333:207] myPointer: Hello world!
4

2 回答 2

8

正如 Caleb 指出的,在这个例子中使用常量 NSString 不是一个好主意。

在源代码中创建字符串对象的最简单方法是使用 Objective-C 的 @"..." 构造:

NSString *temp = @"/tmp/scratch"; 请注意,以这种方式创建字符串常量时,应避免使用 7 位 ASCII 字符以外的任何字符。这样的对象是在编译时创建的,并且存在于程序的整个执行过程中。编译器使此类对象常量在每个模块的基础上都是唯一的,并且它们永远不会被释放,尽管您可以像处理任何其他对象一样保留和释放它们。您也可以像执行任何其他字符串一样将消息直接发送到字符串常量:

BOOL 相同 = [@"comparison" isEqualToString:myString];

文档解释说常量字符串永远不会消失。

尝试使用其他东西进行实验。我尝试了 NSObject,它产生了预期的结果。

界面:

@interface ViewController : UIViewController

@property (strong, nonatomic) NSObject * myString; 
@property (weak, nonatomic) NSObject * myPointer;

@end

执行:

@implementation ViewController

@synthesize myString = _myString;
@synthesize myPointer = _myPointer;

- (void)viewDidLoad{

    [super viewDidLoad];

    self.myString = [[NSObject alloc] init];
    self.myPointer = self.myString;
    self.myString = nil; 
    NSLog(@"myString: %@", self.myString);
    NSLog(@"myPointer: %@", self.myPointer);
}

- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation
{
    return (interfaceOrientation != UIInterfaceOrientationPortraitUpsideDown);
}

@end

如文档 - Apple Developerllvm中所述,当没有指向内存的强指针时,弱指针设置为 nil 。

__weak 指定一个不使被引用对象保持活动状态的引用。当对象没有强引用时,弱引用设置为 nil。

于 2012-02-26T12:33:25.367 回答
5

因此,它应该显示 nil 而不是“Hello World!”。

常量字符串永远不会被释放,所以你的 `@"Hello World!" 永远不会消失。这就是为什么你的弱引用永远不会设置为零的原因。

于 2012-02-26T11:57:33.250 回答