Каков основной механизм синтеза ивара в современной среде выполнения Objective C? - PullRequest
22 голосов
/ 08 ноября 2008

Одной из особенностей современной (64-битной OS X и iPhone OS) среды выполнения Objective C является возможность свойств динамически синтезировать ивары без явного объявления их в классе:

@interface MyClass : NSObject {
//  NSString *name; unnecessary on modern runtimes
}

@property (retain) NSStrng *name;

@end

@implementation MyClass

@synthesize name;

@end

В довольно большой части моего кода я использую пользовательские реализации геттеров для инициализации свойств:

- (NSString *) name {
  if (!name) {
    name = @"Louis";
  }

  return name;
}

Вышеприведенное несовместимо с синтезированными иварами, так как ему необходим доступ к ивару, который не объявлен в заголовке. По разным причинам я хотел бы обновить ряд моих личных фреймворков для использования синтезированных иваров при построении на современных средах исполнения. Для достижения этой цели необходимо изменить приведенный выше код для работы с синтезированными иварами.

Хотя в документации Objective C 2.0 говорится, что синтезированные средства доступа в современной среде выполнения будут синтезировать ivar при первом использовании. Он не указывает, какой механизм низкого уровня используется для этого. Это сделано с помощью class_getInstanceVariable (), ослаблены ли ограничения на class_addIvar (), это недокументированная функция в целевой среде выполнения C 2.0? Хотя я мог бы реализовать свое собственное стороннее хранилище для данных, поддерживающих мои свойства, я бы предпочел использовать механизм, который используют синтезированные методы доступа.

Ответы [ 3 ]

21 голосов
/ 08 ноября 2008

Я пошел и снова посмотрел документацию, и я думаю, что вы неправильно ее прочитали. Синтезированные ивары создаются во время компиляции, а не во время выполнения.

Согласно документации Objective-C 2.0 :

Существуют различия в поведении, которые зависят от времени выполнения (см. Также «Различия во время выполнения»):

Для устаревших сред выполнения переменные экземпляра уже должны быть объявлены в блоке @interface. Если переменная экземпляра с тем же именем и совместимым типом, что и у свойства, существует, она используется - в противном случае вы получите ошибку компилятора.

Для современных сред выполнения переменные экземпляра синтезируются по мере необходимости. Если переменная экземпляра с таким именем уже существует, она используется.

Так что все, что вам нужно сделать, это объявить переменную экземпляра, которая вам нужна, и один и тот же код будет работать в обе среды выполнения ...

0 голосов
/ 28 января 2012

То, что вы ищете, это @synthesized name, например:

@synthesize name = _name;

...

- (NSString *) name {
    if (!name) {
        _name = @"Louis";
    }

    return _name;
}
0 голосов
/ 04 апреля 2011

Вы добавляете свойства во время выполнения с NSKeyValueCoding Protocol .

[myObject setValue:@"whatever" forKey:@"foo"];
...