NSNotification Name 最佳写法

时间:2023-12-19 08:57:08

本文主要借探讨NSNotificationName的最佳写法的机会,学习下extern, static, const, #define 和常量指针与指针常量等的特性和用法。

1.NSNotification标准使用方法

发送通知

[[NSNotificationCenter defaultCenter] postNotificationName:QLPosterDidPostNotification object:nil];

接收通知

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(didReceive:) name:QLPosterDidPostNotification object:nil];

还要注意在dealloc中移除通知哈

2.通知的发送者和接收者都需要知道NSNotification name,如何保证双方统一

在iOS中,通知是为了方便解耦,因此通知的发送者和接收者往往不在同一个类/文件中

但是无论是发送者还是接收者,他们使用的通知名字却需要一致,如上例的QLPosterDidPostNotification在两处地方都使用到了

那么如何保证2处地方使用的通知名字是同一个呢?

最蠢的做法就是在各处需要用到这个通知的地方自己定义一个QLPosterDidPostNotification

这种做法违背了DRY原则,简直忍无可忍,一旦违背了DRY原则,代码的可读性将会变得非常差,维护起来会非常恶心!

最佳实践是一处地方定义,多处地方使用

3.最佳写法

QLPoster.h

extern NSString *const QLPosterDidPostNotification;

QLPoster.m

NSString *const QLPosterDidPostNotification = @"QLPosterDidPostNotification";

QLReceiver.h

extern NSString *const QLPosterDidPostNotification;

4.怎么用

首先,很自然的,我们先在QLPoster.m中声明定义一个通知的名字

NSString *const QLPosterDidPostNotification = @"QLPosterDidPostNotification";

其次,为了让外面的人统一使用该变量,我们需要在QLPoster.h中将该变量暴露出去,并且加上extern,但不需要赋值(因为是声明)。目的就是为了告诉外部调用者,我有QLPosterDidPostNotification这个变量,它是什么内容你们别管,直接用就行了

然后QLReceiver要来调用啦,直接把QLPoster.h中关于变量声明的这句代码直接copy到自己的.h就行啦(因为这也是声明)

5.为什么不用#define

static const 是十分常见的组合,这里可以先把问题转为 static const 和 #define 的区别。

static const和#define的异同是:

相同点:编译器根本不会创建符号,预编译时把所有遇到的变量都替换为常值

不同点:但是static const定义的常量带有类型信息(可以指明常量是int, double, ...),便于理解

由于#define定义的常量不含类型信息,编译器只是会在编译前据此执行查找与替换操作。即使有人重新定义了常量值,编译器也不会产生警告信息,这将导致应用程序中的常量不一致

6.extern是什么(为什么不用static const)

extern int a;//声明一个全局变量a
int a; //定义一个全局变量a
extern int a = ;//定义一个全局变量a并给初值
int a = ;//定义一个全局变量a并给初值

定义只能有1处,但声明可以有多处

定义引起内存分配,声明则不会

A.m

NSString *hello = @"hello";

B.m中不需要#import "A.m",直接

extern NString *hello;
NSLog(@"%@", hello); //输出 hello

非常厉害,不需要引用相关头文件就能使用其值!

变量的声明默认就是extern

变量的声明默认就是extern

变量的声明默认就是extern

重要的事情说3次了

假如A.m不想被B用extern找到,怎么办?只需要加个static

A.m

static NSString *hello = @"hello";

static表示该值只会在本文件(.m)中使用哈,别人看不到的

假如声明此变量时不加static,则编译器会为它创建一个“外部符号”(external symbol)

假如C.m也和A.m一样,都有一个NSString *hello = @"hello";

那就会编译失败,因为有2个外部变量hello

7.为什么加const

通知名称一般是固定不可变的,且不希望外部修改其值,所以加上const无非是最正确的选择

8.为什么是NSString *const 而不是 NSString const *

这就是C/C++中说到的常量指针和指针常量的区别了

const写在*前面,则该指针可被重新赋值,而指针指向的内容不可被重新赋值

const写在*后面,则该指针指向的内容可被重新赋值,而指针本身不可被重新赋值(即不能指向其他内存)

回到iOS中来,我们修改NSString的值,是怎么修改

NSString *str = @"";
str = @"";

注意哦,str = @"456";相当于(不代表就一定是)str = [[NSString alloc] initWithString:"456"];

就是说,iOS中,对NSString的修改一般都是让其指向一块新的内存

所以为了防止外部改变通知名字的值,我们一般将const写在*后面

9.规范的NSNotification Name命名方式

[Name of associated class] + [Did | Will] + [UniquePartOfName] + Notification

(1)QLPosterDidPostNotification 而不是 kDidPostNotification

以k开头的一般是类内自己使用的static const变量,如果需要提供给外部使用,最好将自己的类名作为前缀

通知一般都不是只给自己用的,所以需要将自己的类名作为前缀

声明为extern的常量要出现在全局符号表中,所以其名称应加以区隔,通常用与之相关的类名做前缀。

(2)Did/Will表示动作的已经结束或即将开始,以明确通知抛出的时机

(3)QLPosterDidPostNotification 而不是 QLPosterDidPostNoti

通知名称这里宁愿名字长也不要引起歧义(苹果的命名一向被吐槽太长了)