Skip to main content

iOS ARC - weak and strong properties



I'm trying to understand the way ARC works, and as far as I know, I should be doing something wrong here. This is the code I'm using:





Interface:







@interface ViewController : UIViewController{



}



@property (strong, nonatomic) NSString * myString ;

@property (weak, nonatomic) NSString * myPointer ;







Implementation:







- (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) ;

}







After executing this, I get:







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!







If I'm not wrong, due to myPointer is weak, it shouldn't retain the content of the object. So, it should show nil instead of "Hello World!".





What am I doing wrong?





Following Caleb's answer, I have created another weak pointer, see code below:







- (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) ;

}







The point is that I still got the same answer I used to have:







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!




Comments

  1. As Caleb pointed out, using a constant NSString for this example is not a good idea.


    The simplest way to create a string object in source code is to use the Objective-C @"..." construct:

    NSString *temp = @"/tmp/scratch"; Note that, when creating a string
    constant in this fashion, you should avoid using anything but 7-bit
    ASCII characters. Such an object is created at compile time and exists
    throughout your program’s execution. The compiler makes such object
    constants unique on a per-module basis, and they’re never deallocated,
    though you can retain and release them as you do any other object. You
    can also send messages directly to a string constant as you do any
    other string:

    BOOL same = [@"comparison" isEqualToString:myString];


    The documentation explains that constant strings will never disappear.

    Try using something else for your experiment. I tried NSObject and it produced expected results.

    Interface:

    @interface ViewController : UIViewController

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

    @end


    Implementation:

    @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


    Weak pointers are set to nil when there are no strong pointers to the memory as explained in the documentation - Apple Developer or llvm.


    __weak specifies a reference that does not keep the referenced object alive. A weak reference is set to nil when there are no strong
    references to the object.

    ReplyDelete
  2. So, it should show nil instead of "Hello World!".


    First, constant strings are never deallocated, so your `@"Hello World!" never goes away.

    Second, even if the string were released and deallocated, that wouldn't change the value of self.myPointer. That property would still hold the same memory address, and so would appear to point to the same data unless that memory were modified. So your pointer could very easily point to something that used to be a valid object and just happens to still work. Relying on such a pointer is bad news, though -- it's a frequent source of intermittent crashes.

    ReplyDelete

Post a Comment

Popular posts from this blog

[韓日関係] 首相含む大幅な内閣改造の可能性…早ければ来月10日ごろ=韓国

div not scrolling properly with slimScroll plugin

I am using the slimScroll plugin for jQuery by Piotr Rochala Which is a great plugin for nice scrollbars on most browsers but I am stuck because I am using it for a chat box and whenever the user appends new text to the boxit does scroll using the .scrollTop() method however the plugin's scrollbar doesnt scroll with it and when the user wants to look though the chat history it will start scrolling from near the top. I have made a quick demo of my situation http://jsfiddle.net/DY9CT/2/ Does anyone know how to solve this problem?

Why does this javascript based printing cause Safari to refresh the page?

The page I am working on has a javascript function executed to print parts of the page. For some reason, printing in Safari, causes the window to somehow update. I say somehow, because it does not really refresh as in reload the page, but rather it starts the "rendering" of the page from start, i.e. scroll to top, flash animations start from 0, and so forth. The effect is reproduced by this fiddle: http://jsfiddle.net/fYmnB/ Clicking the print button and finishing or cancelling a print in Safari causes the screen to "go white" for a sec, which in my real website manifests itself as something "like" a reload. While running print button with, let's say, Firefox, just opens and closes the print dialogue without affecting the fiddle page in any way. Is there something with my way of calling the browsers print method that causes this, or how can it be explained - and preferably, avoided? P.S.: On my real site the same occurs with Chrome. In the ex