- 不要等到明天,明天太遙遠,今天就行動。
須讀:看完該文章你能做什么?
知道property中的NSString為什么要用copy
為什么block要用copy
copy中的block怎么防止循環(huán)引用
學習前:你必須會什么?(在這里我已經(jīng)默認你具備C語言的基礎了)
什么property,property的關鍵字
一、本章筆記
一、為什么property中的NSString的屬性要使用 copy,
因為外面可能會進行處理,這樣就修改對象中的屬性
二、為什么block要使用copy
避免以后調(diào)用block的時候,外界的對象已經(jīng)釋放了
三、copy block之后引發(fā)循環(huán)引用
如果對象中的block 有用到了對象自己,那么為了避免內(nèi)存泄漏,應該將對象修飾為 __block
記住: 以后字符串屬性都用copy
二、code
main.m
#pragma mark 19-copy和Property
#pragma mark 概念
/*
一、為什么property中的NSString的屬性要使用 copy,
因為外面可能會進行處理,這樣就修改對象中的屬性
二、為什么block要使用copy
避免以后調(diào)用block的時候,外界的對象已經(jīng)釋放了
三、copy block之后引發(fā)循環(huán)引用
如果對象中的block 有用到了對象自己,那么為了避免內(nèi)存泄漏,應該將對象修飾為 __block
記住: 以后字符串屬性都用copy
*/
#pragma mark - 代碼
#import <Foundation/Foundation.h>
#pragma mark 類
#import "Person.h"
#import "Dog.h"
#pragma mark - main函數(shù)
int main(int argc, const char * argv[])
{
#pragma 1.copy的第一個用途,防止外界修改內(nèi)部的數(shù)據(jù)
NSMutableString *temp = [NSMutableString stringWithFormat:@"lyh"];
Person *p = [[Person alloc]init];
p.name = temp;
// 問題 : 修改了外面的變量, 影響了對象中的屬性
[temp appendFormat:@" cool"];
NSLog(@"name = %@",p.name);
#pragma 2.可以使用copy保存block,這樣可以保住block使用的外界對象的命 避免以后調(diào)用block的時候,外界的對象已經(jīng)釋放了
/*
__block int num = 10;
void (^myBlock)() = ^{
num = 20;
NSLog(@"%i",num);
};
myBlock();
*/
// block默認存儲在棧中, 棧中的block訪問到外界的對象,不會對應進行retain
// block如果在堆中, 如果block中訪問了外界的對象, 會對外界的對象進行一次retain
/*
Person *person = [[Person alloc]init];
NSLog(@"retainCount = %lu",[person retainCount]);
void (^myBlock)() = ^{
NSLog(@"%p",person);
NSLog(@"retainCount = %lu",[person retainCount]);
};
Block_copy(myBlock); // 將block 轉移到堆中
myBlock();
*/
/*
Dog *d = [[Dog alloc]init]; // 1
NSLog(@"retainCount = %lu",[d retainCount]);
Person *p1 = [[Person alloc]init];
p1.pBlcok = ^{
NSLog(@"%@",d);
};
NSLog(@"retainCount = %lu",[d retainCount]);
// 如果狗調(diào)用block之前釋放了, 那么程序就會崩潰
[d release]; // 0
p1.pBlcok();
[p1 release];
*/
#pragma 3.copy block之后引發(fā)循環(huán)引用
// 如果對象中的block 有用到了對象自己,那么為了避免內(nèi)存泄漏,應該將對象修飾為 __block
__block Person *p1 = [[Person alloc]init];
p1.name = @"lyh";
NSLog(@"retainCount = %lu",[p1 retainCount]);
p1.pBlcok = ^{
NSLog(@"name = %@",p1.name);
};
NSLog(@"retainCount = %lu",[p1 retainCount]);
p1.pBlcok();
[p1 release];
return 0;
}
Person
>>>.h
#import <Foundation/Foundation.h>
typedef void (^myBlock) ();
@interface Person : NSObject
//@property (nonatomic,retain) NSString *name;
@property (nonatomic,copy) NSString *name;
// 注意 : 如果是block使用copy并不是拷貝,而是轉移到堆當中
//@property (nonatomic,assign) myBlock pBlcok;
@property (nonatomic,copy) myBlock pBlcok;
@end
>>>.m
#import "Person.h"
@implementation Person
- (void)dealloc
{
// 只要給block發(fā)送一條release消息,block中使用的對象 也會收到該消息
Block_release(_pBlcok);
NSLog(@"%s",__func__);
[super dealloc];
}
@end
Person
>>>.h
#import <Foundation/Foundation.h>
@interface Dog : NSObject
@end
>>>.m
#import "Dog.h"
@implementation Dog
- (void)dealloc
{
NSLog(@"%s",__func__);
[super dealloc];
}
@end