编译错误:“初始化元素不是编译时常量”

时间:2011-05-26 18:07:34

标签: objective-c cocoa initialization compiler-errors global-variables

编译此代码时,我收到错误“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

5 个答案:

答案 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)

必须将全局变量初始化为常量值,例如40.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