Получение объекта из NSSet - PullRequest
       14

Получение объекта из NSSet

91 голосов
/ 30 сентября 2010

Если вы не можете получить объект с objectAtIndex: из NSSet, тогда как вы получаете объекты?

Ответы [ 8 ]

138 голосов
/ 30 сентября 2010

Есть несколько вариантов использования для набора.Вы можете перечислить через (например, с помощью enumerateObjectsUsingBlock или NSFastEnumeration), позвонить containsObject, чтобы проверить членство, используйте anyObject, чтобы получить участника (не случайным образом), или преобразуйте его в массив (в произвольном порядке) с помощью allObjects.

Набор подходит, если вы не хотите дублировать, не заботьтесь озаказать и хотите быстро проверить членство.

30 голосов
/ 30 сентября 2010

NSSet не имеет метода objectAtIndex:

Попробуйте вызвать allObjects, который возвращает NSArray всех объектов.

17 голосов
/ 04 августа 2013

можно использовать FilterSetUsingPredicate, если у вас есть какой-то уникальный идентификатор для выбора нужного вам объекта.

Сначала создайте предикат (при условии, что ваш уникальный идентификатор в объекте называется «идентификатор», и этоNSString):

NSPredicate *myPredicate = [NSPredicate predicateWithFormat:@"identifier == %@", identifier];

И затем выберите объект, используя предикат:

NSObject *myChosenObject = [mySet filteredSetUsingPredicate:myPredicate].anyObject;
13 голосов
/ 04 марта 2013

NSArray *myArray = [myNSSet allObjects];

MyObject *object = [myArray objectAtIndex:(NSUInteger *)]

замените NSUInteger на индекс нужного вам объекта.

8 голосов
/ 28 ноября 2016

Для Swift3 и iOS10:

//your current set
let mySet : NSSet
//targetted index
let index : Int
//get object in set at index
let object = mySet.allObjects[index]
6 голосов
/ 20 апреля 2012

NSSet использует метод isEqual: (который объекты, которые вы помещаете в этот набор, должен переопределять, кроме того, метод хэширования), чтобы определить, находится ли объект внутри него.

Так, например, если выиметь модель данных, которая определяет ее уникальность по значению id (скажем, свойство:

@property NSUInteger objectID;

, тогда вы реализуете isEqual: as

- (BOOL)isEqual:(id)object
{
  return (self.objectID == [object objectID]);
}

, и вы можете реализовать хеш:

- (NSUInteger)hash
{
 return self.objectID;  // to be honest, I just do what Apple tells me to here
                       // because I've forgotten how Sets are implemented under the hood
}

Затем вы можете получить объект с этим идентификатором (а также проверить, находится ли он в NSSet) с помощью:

MyObject *testObject = [[MyObject alloc] init];
testObject.objectID = 5; // for example.  

// I presume your object has more properties which you don't need to set here 
// because it's objectID that defines uniqueness (see isEqual: above)

MyObject *existingObject = [mySet member: testObject];

// now you've either got it or existingObject is nil

Но да, единственный способ получитьчто-то из NSSet заключается в рассмотрении того, что в первую очередь определяет его уникальность.

Я не проверял, что быстрее, но я избегаю использования перечисления, потому что оно может быть линейным, тогда как использование member: method будетнамного быстрее. Это одна из причин, по которой вы предпочитаете использовать NSSet вместо NSArray.

0 голосов
/ 09 августа 2018

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

Если вас не волнует какой объект, который вы получите, используйте -anyObject, который просто дает вам один объект из набора, как положить руку в сумку и схватить что-нибудь.

Dog *aDog = [dogs anyObject]; // dogs is an NSSet of Dog objects

Если вас волнует, какой объект вы получите, используйте -member, который возвращает вам объект, или ноль, если его нет в наборе. Вы должны уже иметь объект, прежде чем вызывать его.

Dog *spot = [Dog dogWithName:@"Spot"];
// ...
Dog *aDog = [dogs member:spot]; // Returns the same object as above

Вот код, который вы можете запустить в Xcode, чтобы понять больше

NSString *one = @"One";
NSString *two = @"Two";
NSString *three = @"Three";

NSSet *set = [NSSet setWithObjects:one, two, three, nil];

// Can't use Objective-C literals to create a set.
// Incompatible pointer types initializing 'NSSet *' with an expression of type 'NSArray *'
//  NSSet *set = @[one, two, three];

NSLog(@"Set: %@", set);
// Prints looking just like an array but is actually not in any order
//Set: {(
//     One,
//     Two,
//     Three
//     )}

// Get a random object
NSString *random = [set anyObject];
NSLog(@"Random: %@", random); // Random: One

// Iterate through objects. Again, although it prints in order, the order is a lie
for (NSString *aString in set) {
    NSLog(@"A String: %@", aString);
}

// Get an array from the set
NSArray *array = [set allObjects];
NSLog(@"Array: %@", array);

// Check for an object
if ([set containsObject:two]) {
    NSLog(@"Set contains two");
}

// Check whether a set contains an object and return that object if it does (nil if not)
NSString *aTwo = [set member:two];
if (aTwo) {
    NSLog(@"Set contains: %@", aTwo);
}
0 голосов
/ 21 июня 2016
for (id currentElement in mySet)
{
    // ** some actions with currentElement 
}
...