iOS ARC - 弱而强的属性

时间:2012-02-26 11:51:35

标签: iphone automatic-ref-counting retain weak

我正在尝试理解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!

2 个答案:

答案 0 :(得分:8)

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

  

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

     

NSString * temp = @“/ tmp / scratch”;请注意,创建字符串时   以这种方式保持不变,除了7位之外,你应该避免使用任何东西   ASCII字符。这样的对象是在编译时创建的并且存在   整个程序执行期间。编译器生成这样的对象   每个模块唯一的常量,它们永远不会被释放,   虽然你可以保留和释放它们,就像你做任何其他对象一样。您   也可以直接将消息发送到字符串常量   其他字符串:

     

BOOL same = [@“comparison”isEqualToString:myString];

documentation解释了常量字符串永远不会消失。

尝试使用其他东西进行实验。我尝试了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   对象的引用。

答案 1 :(得分:5)

  

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

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