一、多用常量,少用#define预处理指令

const关键字,它限定一个变量不允许被改变。使用const在一定程度上可以提高程序的安全性和可靠性。便于进行类型检查,使编译器对处理内容有更多了解,消除了一些隐患。
const 推出的初始目的,正是为了取代#define预处理指令,消除它的缺点,同时继承它的优点。使用#define预处理指令,只是在预处理器里进行文本替换,这样定义出来的常量不含类型信息,编译器只是会在编译前据此执行查找和替换,即时有人重新定义此常量值,编译器也不会警告,这将导致应用内常量值不一致。

(1) 编译器处理方式不同
  define宏是在预处理阶段展开。
  const常量是编译运行阶段使用。
(2) 类型和安全检查不同
  define宏没有类型,不做任何类型检查,仅仅是展开。
  const常量有具体的类型,在编译阶段会执行类型检查。
(3) 存储方式不同
  define宏仅仅是展开,有多少地方使用,就展开多少次,不会分配内存。(宏定义不分配内存,变量定义分配内存。)如果是字符串类型,取的时候只需要给前面地址,如果是整形、浮点型会有多份拷贝,但这些数写在指令中。占的只是代码段而已,大量用宏会导致二进制文件变大
  const常量会在内存中分配(可以是堆中也可以是栈中)。共享一块内存空间,就算项目中N处用到,也不会分配N块内存空间。
(4)const 可以节省空间,避免不必要的内存分配。

二、const的使用

1、注意常量名称

1
static const NSTimeInterval kAninationDuration = 0.3

若常量局限于某“编译单元”(translation unit,也就是“实现文件”implementation file)之内,则在前面加上字母k;
若常量在类之外可见,则通常以类名为前缀。

2、变量一定要同时用static和const来声明。如果试图修改有const修饰的变量,那么编译器就会报错。而static修饰符则意味着该变量仅在定义变量的编译单元可见。编译器每收到一个编译单元,就会输出一份“目标文件”(object file)。那么如果声明上述变量不加static修饰,则编译器会为他创建一个“外部符号”(external symbol)。此时若是另一个编译单元也声明了同名变量,那么编译器就会报错。

实际上,如果变量既声明为static又声明为const,那么编译器根本不会创建符号,而是会像#define预处理指令一样,把所有遇到的变量都替换成常值。不过这种定义方法的常量带有类型信息。

3、公开常量

那么有时候如果需要公开常量的时候。那么此类常量需要放在“全局符号表”(global symbol table)中,以便可以在定义该常量的编译单元之外使用。

1
2
// In the header file 
extern NSString *const XYZStringConstant;
1
2
//In the implementation file
NSString *const XYZStringConstant = @"VALUE";

这个变量声明在头文件,且在实现文件“定义”。注意const修饰符在常量类型中的位置。常量定义应该从右至左阅读,所以,XYZStringConstant 就是“一个常量,而这个常量是指针,指向 NSString 对象”,这与需求相符。

FOUNDATION_EXPORT

在 OC 中系统库头文件或第三方库时,经常可以看到 FOUNDATION_EXPORT,作用与 extern 相同,可置于变量/常量或者函数前,以表示变量/常量或者函数的定义在别的文件中,提示编译器遇到此变量或函数时,在其它模块中寻找其定义。实际上,我们在 NSObjCRuntime.h 头文件中看一下 FOUNDATION_EXPORT的定义,就可以看到它实际上就是 extern 的一个宏定义,只不过是同时兼容了 C 与 C++。所以,如果在程序中用到了 C 与 C++ 混编,那么建议使用FOUNDATION_EXPORT来声明全局变量/常量或函数。

1
2
3
4
5
6
7
8
9
//NSObjCRuntime.h
#if defined(__cplusplus)
#define FOUNDATION_EXTERN extern "C"
#else
#define FOUNDATION_EXTERN extern
#endif

#define FOUNDATION_EXPORT FOUNDATION_EXTERN
#define FOUNDATION_IMPORT FOUNDATION_EXTERN

使用,如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
#import "ViewController.h"
#import "Student.h"

NSString *const XYZStringConstant = @"VALUE";
NSString * XYZStringValue = @"VALUE";

@implementation ViewController

- (void)viewDidLoad {
[super viewDidLoad];

XYZStringValue = @"123";

Student *s = [[Student alloc] init];
//...
}

1
2
3
4
5
6
7
8
9
10
11
12
13
#import "Student.h"

FOUNDATION_EXPORT NSString *const XYZStringConstant;
FOUNDATION_EXPORT NSString * XYZStringValue;

@implementation Student

- (instancetype)init {
if (self = [super init]) {
NSLog(@" student - %@ %@", XYZStringConstant, XYZStringValue);
}
return self;
}

如上例所示,在 ViewController 中定义了常量,然后再别的文件中用 FOUNDATION_EXPORT 声明使用。

4、注意const修饰符在常量类型中的位置。
const只修饰它右边的内容,被const修饰的内容都是常量

References

《Effective Objective-C 2.0》
iOS 宏(define)与常量(const)的正确使用