编译此代码时,我收到错误“initializer element is not a compile-time constant”。任何人都可以解释原因吗?
#import "PreferencesController.h"
@implementation PreferencesController
- (id)init
{
self = [super init];
if (self) {
// Initialization code here.
}
return self;
}
NSImage* imageSegment = [[NSImage alloc] initWithContentsOfFile:@"/User/asd.jpg"];//error here
答案 0 :(得分:99)
当您在函数范围之外定义变量时,该变量的值实际上会写入您的可执行文件中。这意味着您只能使用常量值。由于您在编译时不了解运行时环境的所有内容(哪些类可用,它们的结构是什么等),因此除了常量字符串之外,您无法创建目标c对象,而这些字符串是特定的结构并保证保持这种状态。您应该做的是将变量初始化为nil并使用+initialize
来创建图像。 initialize
是一个类方法,它将在您的类上调用任何其他方法之前调用。
示例:
NSImage *imageSegment = nil;
+ (void)initialize {
if(!imageSegment)
imageSegment = [[NSImage alloc] initWithContentsOfFile:@"/User/asd.jpg"];
}
- (id)init {
self = [super init];
if (self) {
// Initialization code here.
}
return self;
}
答案 1 :(得分:19)
必须将全局变量初始化为常量值,例如4
或0.0
或@"constant string"
或nil
。对象构造函数(例如init
)不返回常量值。
如果你想拥有一个全局变量,你应该将它初始化为nil
,然后使用类方法返回它:
NSImage *segment = nil;
+ (NSImage *)imageSegment
{
if (segment == nil) segment = [[NSImage alloc] initWithContentsOfFile:@"/user/asd.jpg"];
return segment;
}
答案 2 :(得分:10)
因为您要求编译器使用本质上动态的代码初始化静态变量。
答案 3 :(得分:5)
原因是您在源代码(静态变量)中定义了函数之外的imageSegment
。
在这种情况下,初始化不能包括代码的执行,如调用函数或分配类。初始化程序必须是一个常量,其值在编译时是已知的。
然后,您可以在init
方法内初始化静态变量(如果您将其声明推迟到init)。
答案 4 :(得分:3)
您当然可以#define宏,如下所示。在编译之前,编译器将用其值替换“IMAGE_SEGMENT”。虽然您将实现为数组定义全局查找,但它与全局变量不同。扩展宏时,它就像内联代码一样工作,因此每次都会创建一个新图像。因此,如果您在使用宏的地方小心,那么您将有效地创建一个全局变量。
#define IMAGE_SEGMENT [[NSImage alloc] initWithContentsOfFile:@"/User/asd.jpg"];
然后在需要的地方使用它,如下所示。每次执行以下代码时,都会使用新的内存指针创建一个新对象。
imageSegment = IMAGE_SEGMENT