it-swarm.com.ru

Как объявить свойства уровня класса в Objective-C?

Может быть, это очевидно, но я не знаю, как объявить свойства класса в Objective-C.

Мне нужно кэшировать для каждого класса словарь и удивляться, как поместить его в класс.

195
mamcx

свойства имеют конкретное значение в Objective-C, но я думаю, что вы имеете в виду нечто, эквивалентное статической переменной? Например. только один экземпляр для всех типов Foo?

Чтобы объявить функции класса в Objective-C, вы используете префикс + вместо - так что ваша реализация будет выглядеть примерно так:

// Foo.h
@interface Foo {
}

+ (NSDictionary *)dictionary;

// Foo.m
+ (NSDictionary *)dictionary {
  static NSDictionary *fooDict = nil;
  if (fooDict == nil) {
    // create dict
  }
  return fooDict;
}
184
Andrew Grant

Я использую это решение:

@interface Model
+ (int) value;
+ (void) setValue:(int)val;
@end

@implementation Model
static int value;
+ (int) value
{ @synchronized(self) { return value; } }
+ (void) setValue:(int)val
{ @synchronized(self) { value = val; } }
@end

И я нашел это чрезвычайно полезным в качестве замены шаблона Singleton.

Чтобы использовать его, просто получите доступ к своим данным с точечной нотацией:

Model.value = 1;
NSLog(@"%d = value", Model.value);
108
spooki

Как видно из WWDC 2016/XCode 8 ( что нового в сеансе LLVM @ 5: 05). Свойства класса могут быть объявлены следующим образом

@interface MyType : NSObject
@property (class) NSString *someString;
@end

NSLog(@"format string %@", MyType.someString);

Обратите внимание, что свойства класса никогда не синтезируются

@implementation
static NSString * _someString;
+ (NSString *)someString { return _someString; }
+ (void)setSomeString:(NSString *)newString { _someString = newString; }
@end
71
Alex Nolasco

Если вы ищете эквивалент уровня @property на уровне класса, то ответ "такого нет". Но помните, @property в любом случае является только синтаксическим сахаром; он просто создает методы объекта с соответствующим именем.

Вы хотите создать методы класса, которые обращаются к статическим переменным, которые, как уже говорили другие, имеют только немного другой синтаксис.

63
Jim Puls

Вот потокобезопасный способ сделать это:

// Foo.h
@interface Foo {
}

+(NSDictionary*) dictionary;

// Foo.m
+(NSDictionary*) dictionary
{
  static NSDictionary* fooDict = nil;

  static dispatch_once_t oncePredicate;

  dispatch_once(&oncePredicate, ^{
        // create dict
    });

  return fooDict;
}

Эти изменения гарантируют, что fooDict создается только один раз.

From документация Apple : "dispatch_once - выполняет объект блока один раз и только один раз за время существования приложения."

21
Quentin

Начиная с Xcode 8 Objective-C теперь поддерживает свойства класса:

@interface MyClass : NSObject
@property (class, nonatomic, assign, readonly) NSUUID* identifier;
@end

Поскольку свойства класса никогда не синтезируются, вам нужно написать собственную реализацию.

@implementation MyClass
static NSUUID*_identifier = nil;

+ (NSUUID *)identifier {
  if (_identifier == nil) {
    _identifier = [[NSUUID alloc] init];
  }
  return _identifier;
}
@end

Вы обращаетесь к свойствам класса, используя обычный синтаксис точки на имени класса:

MyClass.identifier;
11
berbie

Свойства имеют значения только в объектах, а не в классах.

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

Вы также можете рассмотреть возможность использования определенных отношений между вашими объектами: вы приписываете роль мастера определенному объекту вашего класса и связываете другие объекты с этим мастером. Мастер будет содержать словарь как простое свойство. Я думаю о дереве, подобном тому, которое используется для иерархии представления в приложениях Какао.

Другой вариант - создать объект выделенного класса, который состоит из словаря "класса" и набора всех объектов, связанных с этим словарем. Это что-то вроде NSAutoreleasePool в Какао.

7
mouviciel

Начиная с Xcode 8, вы можете использовать атрибут свойства class в ответе Берби.

Однако в реализации вам необходимо определить и класс getter, и setter для свойства класса, используя статическую переменную вместо iVar.

Sample.h

@interface Sample: NSObject
@property (class, retain) Sample *sharedSample;
@end

Sample.m

@implementation Sample
static Sample *_sharedSample;
+ ( Sample *)sharedSample {
   if (_sharedSample==nil) {
      [Sample setSharedSample:_sharedSample];
   }
   return _sharedSample;
}

+ (void)setSharedSample:(Sample *)sample {
   _sharedSample = [[Sample alloc]init];
}
@end
5
Jean-Marie D.

Если у вас много свойств уровня класса, то может быть в порядке одноэлементный шаблон. Что-то вроде этого:

// Foo.h
@interface Foo

+ (Foo *)singleton;

@property 1 ...
@property 2 ...
@property 3 ...

@end

А также

// Foo.m

#import "Foo.h"

@implementation Foo

static Foo *_singleton = nil;

+ (Foo *)singleton {
    if (_singleton == nil) _singleton = [[Foo alloc] init];

    return _singleton;
}

@synthesize property1;
@synthesize property2;
@synthesise property3;

@end

Теперь получите доступ к своим свойствам на уровне класса, как это:

[Foo singleton].property1 = value;
value = [Foo singleton].property2;
2
Pedro Borges