- 不要等到明天租漂,明天太遙遠(yuǎn)阶女,今天就行動(dòng)。
須讀:看完該文章你能做什么哩治?
知道property中的NSString為什么要用copy
為什么block要用copy
copy中的block怎么防止循環(huán)引用
學(xué)習(xí)前:你必須會(huì)什么?(在這里我已經(jīng)默認(rèn)你具備C語言的基礎(chǔ)了)
什么property,property的關(guān)鍵字
一秃踩、本章筆記
一、為什么property中的NSString的屬性要使用 copy,
因?yàn)橥饷婵赡軙?huì)進(jìn)行處理,這樣就修改對(duì)象中的屬性
二业筏、為什么block要使用copy
避免以后調(diào)用block的時(shí)候,外界的對(duì)象已經(jīng)釋放了
三憔杨、copy block之后引發(fā)循環(huán)引用
如果對(duì)象中的block 有用到了對(duì)象自己,那么為了避免內(nèi)存泄漏,應(yīng)該將對(duì)象修飾為 __block
記住: 以后字符串屬性都用copy
二、code
main.m
#pragma mark 19-copy和Property
#pragma mark 概念
/*
一蒜胖、為什么property中的NSString的屬性要使用 copy,
因?yàn)橥饷婵赡軙?huì)進(jìn)行處理,這樣就修改對(duì)象中的屬性
二消别、為什么block要使用copy
避免以后調(diào)用block的時(shí)候,外界的對(duì)象已經(jīng)釋放了
三、copy block之后引發(fā)循環(huán)引用
如果對(duì)象中的block 有用到了對(duì)象自己,那么為了避免內(nèi)存泄漏,應(yīng)該將對(duì)象修飾為 __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的第一個(gè)用途,防止外界修改內(nèi)部的數(shù)據(jù)
NSMutableString *temp = [NSMutableString stringWithFormat:@"lyh"];
Person *p = [[Person alloc]init];
p.name = temp;
// 問題 : 修改了外面的變量, 影響了對(duì)象中的屬性
[temp appendFormat:@" cool"];
NSLog(@"name = %@",p.name);
#pragma 2.可以使用copy保存block,這樣可以保住block使用的外界對(duì)象的命 避免以后調(diào)用block的時(shí)候,外界的對(duì)象已經(jīng)釋放了
/*
__block int num = 10;
void (^myBlock)() = ^{
num = 20;
NSLog(@"%i",num);
};
myBlock();
*/
// block默認(rèn)存儲(chǔ)在棧中, 棧中的block訪問到外界的對(duì)象,不會(huì)對(duì)應(yīng)進(jìn)行retain
// block如果在堆中, 如果block中訪問了外界的對(duì)象, 會(huì)對(duì)外界的對(duì)象進(jìn)行一次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 轉(zhuǎn)移到堆中
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之前釋放了, 那么程序就會(huì)崩潰
[d release]; // 0
p1.pBlcok();
[p1 release];
*/
#pragma 3.copy block之后引發(fā)循環(huán)引用
// 如果對(duì)象中的block 有用到了對(duì)象自己,那么為了避免內(nèi)存泄漏,應(yīng)該將對(duì)象修飾為 __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并不是拷貝,而是轉(zhuǎn)移到堆當(dāng)中
//@property (nonatomic,assign) myBlock pBlcok;
@property (nonatomic,copy) myBlock pBlcok;
@end
>>>.m
#import "Person.h"
@implementation Person
- (void)dealloc
{
// 只要給block發(fā)送一條release消息,block中使用的對(duì)象 也會(huì)收到該消息
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