美文网首页
二十一、修饰属性的关键字

二十一、修饰属性的关键字

作者: MethodSwizzling | 来源:发表于2017-06-17 13:45 被阅读0次

    二十一、修饰属性的关键字

    1.属性和成员变量

           必须先把有关属性和成员变量的话说开了,否则有关剩下的那些关键字眼一个也甭想看明白。

    总的概括:

           属性是.h文件里@property后面定义的那个,成员变量是会在.m文件里自动生成一个前面加了下划线的那个,方便用。self.那个也能用,但不叫成员变量,叫调用settergetter方法的,它俩在考你的问题面前根本就不是一个东西,水火不容,不能乱说。

           是这样的,你在.h文件里洋洋洒洒的写了一句

           @property (nonatomic,copy) NSString *str(我就是被定义出来的那个“属性”);

           之后,意味着两件事的发生:

    第一件事:

            @property决定系统会自动帮你在.m文件里创建一个以下划线加上copy修饰的你那个属性名字构成的一个东西,叫做成员变量,(“_str”,没错我就是那个“成员变量”)   这个东西是专门给你在.m文件里调用的,外面的类访问不到,用不了这个。这也就促成了第二样东西的产生......

    第二件事:

            @property决定系统又帮你生成了一套针对这个属性的setter和getter方法,给你用的,从类的外部访问它的时候。

           额外总结一下@property做的三件事情:

           1)生成一个带下划线的成员变量;

           2)生成一套默认的settergetter方法;

           3)将settergetter方法的声明放到头文件中,不然外面的类想用也用不了。

            (我先把我记得的写出来,这里留白用来写setter和getter方法具体长什么样还有他们延伸出的问题--

    mrc下的基本数据类型的setter方法:

    - (void)setAge:(NSInteger)age {

            //因为基本数据类型是不需要release或者copy或者retain的       

             _age = age;

    }

    mrc下类类型的setter方法:

    - (void)setName:(NSString *)name {

            [_name release];

            _name = [name retain];

    }

    mrc下类类型的getter方法,这个基本数据类型的也一样:

    - (NSString *)name {

             return _name;

    }

    然而,一旦同时重写了setter和getter方法,那么使用的就不是@property给生成的成员变量了,所以得在.m文件里把成员变量也重新声明一遍。

    好了,settergetter方法写完了)

    2.copy

            在讲完属性,成员变量,settergetter这堆乱糟糟的关系后,再来讲copy。特别地讲copy是因为它修饰的都是带有指针类型的属性。

            先从事情的发生说起。有个controller,它里面有个cell,这个cell里面,有个label。现在要想给这个label赋值text,怎么办?当然是要通过cell的对象去“.”点出来label属性然后给它赋值。这个“.”点它的时候,就是上面一小节中说的,从外面调用了一个类的属性,要这个属性的setter和getter方法出马了。点属性和settergetter这两件事一绑定,就叫做点语法

            那么,当我们利用点语法给这个label的text属性赋值的时候发生了什么呢?

            cell.label.text = @"zjy";

            [cell.label setText@“zjy”];

            上面这两句话表示的意思是完全等同的,意思是当你从类外部给一个对象的一个属性赋值的时候,调用到的就是该属性的setter方法,例子举得不是很好,但是没错,setText也是那种setter方法。

           总结一下目前已知的事情,是从类外部“点”一个属性的时候必然会用到它的setter方法,无论是默认的还是重写过的,目的都是要通过这个setter方法去给这个属性的成员变量赋值。那么copy的setter方法里面长什么样就也得知道比较好了。

    它长这样(还是换回setStr的例子举):

           -  (void)setStr:(NSString *) str {

                 //把该属性的成员变量原先指向的那块内存空间的引用计数减一(至于是不是能释放掉那是人家自己的事情了,这也能很好的解释野指针,which means一个对象指向了一块内存区域,但那里已经被释放掉了,所以再调用该对象”.“点什么的时候程序就会报内存错误,然后崩掉)(另一个备注是:如果一个对象的引用计数减到零的时候,那么这个对象就会被释放了,具体是谁来做这件事呢?说是系统,其实是runloop在循环检测,发现谁的引用计数为零了就让它去调用dealloc方法,也就是领盒饭了)

                 [ _str release ];

                 //利用copy新new出来一块内存空间,放上该放的东西,然后让你家成员变量指     向这块内存空间。

                 _ str = [str copy];

             }

             当然,以上代码的具体写法是要搁在MRC的年代,现在不用这么写了,但是这个发生的过程依然是存在的。

    3.retain

           retain就当做是跟copy一组的cp吧,提完copy肯定得说retain,而且对比着看更有助于记住它们。

           上面那个例子,除了把“copy“字眼换成”retain“以外,它们在MRC下的setter方法里的实现上没有任何区别。但是,意思上有区别:

            当然,共同点是,首先它俩都得先允许它们的成员变量把原先不管指向的是哪旮沓的内存空间的引用计数减一。

            copy要经历的是:new一块新内存空间,把存放传入值的那块内存空间的内容复制一份,放进新内存空间后;这时,原存放传入值空间引用计数不变,新内存空间引用计数由零变为一;然后,让成员变量去指向新内存空间(没错,成员变量指向”谁“或者等于”谁“是不会改变这个”谁“的引用计数的)。也就是说,copy的过程中是改变了引用计数了,但不是我们通常以为的那块。

            retain要经历的是:使原存放传入值的空间的引用计数加一,如果原来引用计数是一,那你现在就是二咯;然后,让成员变量去指向这块内存空间。(需要再次明确,先加一再指向,指向与引用计数加一无关)

           @property有两个对应的词:@synthesize和@dynamic,如果需要用到的话在.m文件里的@implemention下面(不是里面)写,例如@dynamic name;

           @dynamic告诉编译器,setter和getter方法由用户自己定义,不用生成;

           @synthesize告诉编译器默认生成setter和getter方法;当然什么都不写的请情况下默认就是用它啦。

    1.nsstring(用什么,为什么不可以用strong,会不会崩溃)

    2.delegate

    3.block

    4.strong,weak,assign,copy,retain

    5.深拷贝,浅拷贝

    相关文章

      网友评论

          本文标题:二十一、修饰属性的关键字

          本文链接:https://www.haomeiwen.com/subject/tfovqxtx.html