《Effective Objective-C 2.0》- 4:多

2018-05-08  本文已影响0人  WhistleCai

编写代码时经常定义常量,比如说,在一个页面上播放一个动画,在很多开发者来说都会把八方动画的时间提取为一个常量。

#define ANIMATION_DURATTON 0.5

这段预处理指令会把代码中的 ANIMATION_DURATTON 字符串替换成0.5,不过这样定义出来的常量没有类型信息,预处理过程中会把碰到的所有的ANIMATION_DURATTON替换成为0.5,这样的话,假设此指令声明在某个头文件中,那么所有引入这个头文件的代码,其ANIMATION_DURATTON都会被替换。

想要解决这个问题最好的办法就是设法利用编译器的某些特性,有个办法比用预处理指令来定义常量更好,如例:

static const NSSTimeInterval kAnimationDuration = 0.5;

此方式定义的常量包含类型信息,其好处是清楚的描述了常量的含义,由此可以知道该常量的类型为NSSTimeInterval。

定义常量的位置很重要,我们总喜欢在头文件里声明预处理指令,这样做很不好,当常量名称有可能互相冲突时更是如此。例如,ANIMATION_DURATTON这个常量名就不该定义在头文件,因为所有引入这份头文件的其他文件都会出现这个名字。其实就连用static const定义的那个常量也不应该出现在头文件里。因为Objective-C没有“名称空间”这一概念,所以那样做就等于声明了一个名叫kAnimationDuration全局变量,因此名称必须加上前缀,以表明所属的类。

如果不打算公开某个常量,就应该在使用该常量的实现文件里定义,例如:

EOCAnimatedView.h

#import <UIKit/UIKit.h>

@interface EOCAnimatedView : UIView

-(void)animate;

@end

//EOCAnimatedView.m

#import "EOCAnimatedView.h"

static const NSSTimeInterval kAnimationDuration = 1.5;

@implementation EOCAnimatedView

-(void)animate{

}

@end

变量一定要同时用static和const来声明,事实上,一个变量既声明为static,又声明为const,那么编译器根本不会创建符号,而会像#define预处理指令一样把所有遇到的变量都替换成常值,不过要记住,用这种方式定义的常量带有类型信息。

有时候需要对外公开一个常量,此类常量需要放在“全局符号表”中,以便可以在定义该常量的编译单元之外使用,因此其定义方式与上例演示的static const有所不一样:

//In the header file

extern NSString *const EOCStringConstant;

//In the implementation file

NSString *const EOCStringConstant=@"VALUE";

这个常量在头文件中“声明”,且在实现文件中“定义”,const修饰符在常量类型中的位置,常量从右至左解读。编译器看到头文件中的extern关键字,就能明白如何明白在引入此文件的代码中处理该常量了。这个关键字就会告诉编译器,在全局符号表中将会有一个名叫EOCStringConstant的符号,编译器无需查看其定义,既允许此代码使用此常量,因为它知道当链接成二进制文件之后,肯定能找到这个常量。

此类常量必须要定义,且只能定义一次,通常将其定义在与声明该常量的头文件相关的实现文件里,由实现文件生成目标文件时,编译器会在“数据段”为字符串分配存储空间。连接器会把此目标文件和其他的目标文件相链接,以生成最终的二进制文件,凡使用到EOCStringConstant这个全局符号的地方,链接器都能将其解析。

注意常量的名字,为避免名称冲突,最好使用与之相关的类名作为前缀。

这样定义的常量优于使用#define预处理命令,因为编译器确保常量的值不会变,总之勿使用预处理指令定义常量,而应该用戒指编译器的特性来确保常量的正确。

上一篇下一篇

猜你喜欢

热点阅读