static:
1、static可以修饰OC中所有的成员变量,有局部和全局之分;static还可以修饰方法
2、static修饰的对象的内存是由系统来管理的
3、static修饰的对象只能被初始化一次
self:
指当前类的对象本身,或者说是当前消息的接收者
super:
指父类的对象本身、向父类发送消息
switch和if
1、它们都是条件判断语句
2、switch只能处理整型、字符型、枚举类型(其实都是整型)的数据,if则没有这样的限制
3、switch的效率更高: if 语句通过一层层的逻辑判断来执行符合条件的语句;而swith可以通过(expresion)直接找到对应的case来执行。
@public、@protected、@private
1、@public:
由它声明的实例变量能在任何其他类中访问
2、@protected:
由它声明的实例变量可以在本类以及其子类中访问。系统默认的是@protected类型的。
3、@private:
由它声明的实例变量只能在本类中访问
include、#import、@class
1、#include与#import功能一样,都是查找该类中的属性和方法;
2、#import可以防止#include具有的交叉导入的问题;
3、@class XX 只是告诉编译器 把XX作为类来处理,但并没有导入该类,所以编译效率更高一些。
nil、Nil、NULL
nil:指向一个对象的指针为空
Nil:指向一个类的指针为空
NULL:指向一个C类型的指针为空
__unused
不使用变量时,加这个前缀可以不报警告。
_Nonnull
表示对象不可以是 nil,null
Nonnull区域设置(Audited Regions)
如果需要每个属性或每个方法都去指定nonnull和nullable,是一件非常繁琐的事。苹果为了减轻我们的工作量,专门提供了两个宏:NS_ASSUME_NONNULL_BEGIN和NS_ASSUME_NONNULL_END。在这两个宏之间的代码,所有简单指针对象都被假定为nonnull,因此我们只需要去指定那些nullable的指针。如下代码所示:
NS_ASSUME_NONNULL_BEGIN
@interface TestNullabilityClass ()
@property (nonatomic, copy) NSArray * items;
- (id)itemWithName:(nullable NSString *)name;
@end
NS_ASSUME_NONNULL_END
在上面的代码中,items属性默认是nonnull的,itemWithName:方法的返回值也是nonnull,而参数是指定为nullable的。
_Nullable
表示对象可以是 nil, null。如果不遵循这个规则,编译器会报错。
NS_DESIGNATED_INITIALIZER
实现指定构造器的。这里之所以要用这个宏,往往是想告诉调用者要用这个方法去初始化(构造)类对象。
如果子类指定了新的初始化器,那么在这个初始化器内部必须调用父类的Designated Initializer。并且需要重写父类的Designated Initializer,将其指向子类新的初始化器。
如下:
// .h
- (instancetype)initWithName:(NSString *)name NS_DESIGNATED_INITIALIZER;
// .m
- (instancetype)init
{
return [self initWithName:@""];
}
- (instancetype)initWithName:(NSString *)name {
self = [super init];
if (self) {
// do something
}
return self;
}
更好的做法
如果定义NS_DESIGNATED_INITIALIZER,大多是不想让调用者调用父类的初始化函数,只希望通过该类指定的初始化进行初始化,这时候就可以用NS_UNAVAILABLE宏。如下:
// .h
- (instancetype)init NS_UNAVAILABLE;
- (instancetype)initWithName:(NSString *)name
NS_DESIGNATED_INITIALIZER;
如果调用者使用init 初始化,编译器就会给出一个编译错误。使用NS_UNAVAILABLE后,就不需要在.m中重写父类初始化函数了。如果要允许调用者使用init就需要在.m中重写父类的初始化函数,如上提到的,否则就会报警告。