Compiler error "expected method not found" when using subscript on NSArray
Asked Answered
B

7

33

I wrote this simple code to try out the new Objective-C literal syntax for NSArrays:

NSArray *array = @[@"foo"];
NSLog(@"%@", array[0]); 

The first line works fine, but the subscripting results in an error:

Expected method to read array element not found on object of type 'NSArray *'

Just wondering if I have done something wrong, or if the literals haven't been fully implemented yet. I'm compiling with Apple LLVM 4.0 and using the iOS 5 SDK.

Here's a screenshot of the error, too.

Error

Brigham answered 11/7, 2012 at 5:19 Comment(5)
The page on clang.llvm.org about Objective-C literals state that you must use Apple LLVM 4.0 or clang v3.1 to use the new features. Do you have one of these installed, and is it specified in your build settings?Bridgehead
Yes I am using LLVM 4.0. Should have mentioned thatBrigham
You've also got to be compiling with the iOS 6 or OS X 10.8 SDKs -- otherwise Foundation objects don't have the necessary methods for the subscripting bit of the literal syntax.Haemo
@JoshCaswell Thats the answer, I am compiling to iOS5. Post as an answer so I can accept!Brigham
#9348222Taiwan
H
36

You've got to be compiling with the iOS 6 or OS X 10.8 SDKs -- otherwise Foundation objects don't have the necessary methods for the subscripting bit of the literal syntax.* Specifically in this case, the subscripting expects objectAtIndexedSubscript: to be implemented by NSArray, and that's a new method that was created to interact with this compiler feature. The parts of the new syntax that just have to do with object creation should work fine, though -- I don't believe that requires any new methods.

Further reading at http://clang.llvm.org/docs/ObjectiveCLiterals.html


*I base this on a bit of research performed by borrrden: https://mcmap.net/q/453113/-can-llvm-4-0-be-used-in-xcode-4-3

I've gotten a lot of upvotes on this answer, which I really feel is founded on borrrden's. Please, if you think my answer is worth an upvote, click through and vote there too.

Haemo answered 11/7, 2012 at 8:6 Comment(3)
There is one small thing I'd like to warn about. Literal bools are also not supported because of this. However, a quick fix that I implemented was adding this to the beginning of one of my common headers (in an iOS project) -> #ifndef __IPHONE_6_0 #if __has_feature(objc_bool) #undef YES #undef NO #define YES __objc_yes #define NO __objc_no #endif #endifNarine
why don't the subscripting refer to just objectAtIndex (if objectAtIndexedSubscript isn't found, or only on -iOS6)?Galloping
Since Xcode 4.4, the new syntax will still deploy back to iOS 5 by calling objectAtIndex:. Your deployment target must be set correctly to 5.0 and the SDK to the latest. See developer.apple.com/library/ios/#releasenotes/ObjectiveC/… for a compatibility matrix.Aime
G
15

If you're not targeting iOS 6 or OS X 10.8, I would like to point out that it's still remarkably easy to get subscripting to work. All you have to do is add the required methods as a category the classes you want subscripting to work for, and implement those methods appropriately. So add to the following classes the methods:

NSArray : - (id)objectAtIndexedSubscript: (NSUInteger)index;

NSMutableArray : - (void)setObject: (id)obj atIndexedSubscript: (NSUInteger)index;

NSDictionary : - (id)objectForKeyedSubscript: (id)key;

NSMutableDictionary : - (void)setObject: (id)obj forKeyedSubscript: (id)key;

Implementing this is a simple as calling the appropriate method for the class. For example, to implement subscripting on NSArray you just implement:

- (id) objectAtIndexedSubscript:(NSUInteger)index{
    return [self objectAtIndex:index];
}

The only downside I can see is you need to make sure to import your category into any class that intends on using the subscripting. Of course, you can get around that requirement by including the #import in your prefix header, usually the file: <appname>-Prefix.pch. (thanks Josh Caswell for pointing that out).

One upside is you can alter the subscripting methods to suit your needs. For example, Apple doesn't allow you to add/remove objects to NSMutableArray using subscripting, but this can be accomplished easily enough:

 - (void) setObject:(id)obj atIndexedSubscript:(NSUInteger)index{
    if (index < self.count){
        if (obj)
            [self replaceObjectAtIndex:index withObject:obj];
        else
            [self removeObjectAtIndex:index];
    } else {
        [self addObject:obj];
    }
}
Galacto answered 25/7, 2012 at 20:21 Comment(4)
«you need to make sure to import your category into any class that intends on using the subscripting» No problem there; just put it in your prefix header! Nice post!Haemo
Thanks @JoshCaswell! I'd forgotten about the prefix. I edited the post to include your info.Galacto
Good point, but it's worth adding that you don't need to do this to target iOS5, only to compile with an earlier SDK. You can target iOS5 (and use the new constructs) and compile with the iOS6 sdk.Gymnastic
That's true, but for the moment iOS 6 is beta only. Not particularly useful for production code. Of course, in a month or so the point will be mute.Galacto
M
2

I got the following code from this link:

You can just add this category to NSObject and collection subscripting will work. I put it in my .pch file.

// Add support for subscripting to the iOS 5 SDK.
#if __IPHONE_OS_VERSION_MAX_ALLOWED < 60000
@interface NSObject (SubscriptingSupport)

- (id)objectAtIndexedSubscript:(NSUInteger)idx;
- (void)setObject:(id)obj atIndexedSubscript:(NSUInteger)idx;
- (void)setObject:(id)obj forKeyedSubscript:(id <NSCopying>)key;
- (id)objectForKeyedSubscript:(id)key;

@end
#endif

Of course you will need the newest version of CLANG which is already in XCode 4.4.

Mauceri answered 17/8, 2012 at 0:21 Comment(5)
This works really well and is really simple compared to the other answers. Thank you!Eureka
How does this differ from Aaron's answer?Haemo
His answer suggests creating multiple categories. This is a simpler way as you only need create a single category. While this may be obvious to experienced developers, it is likely to be less obvious to developers who are new to categories.Mauceri
I missed that this is a category on NSObject, and I have to say that I don't like it. This way, the category is going to be implemented on objects for whom the methods have no meaning. There was a language feature (@optional protocol methods) introduced a while back to avoid exactly that. The methods should be added to the classes where they are needed.Haemo
That's a good point. Using @optional would have been a good idea. Of course, Xcode 4.5 is officially released now, and this whole thing is no longer necessary.Mauceri
C
1

I got the same problem, but then it wasn't b/c of a lacking iOS version.. but it was simply because the original array was set as an NSArray rather than an NSMutableArray. Changing it to NSMutableArray fixed it for me

Camus answered 17/4, 2013 at 12:35 Comment(0)
A
0

I'm adding this because this is a common error that still exists in Xcode as of 7/2015 and it's not easy to figure out how to resolve it.

I received this error when attempting to call a method on an object without having created an instance of the object. My solution was to create an instance of the object, then call the method on the property on the instance of that object.

Example: What didn't work: [self methodCall:arrayItem] (see full example below)

[self tappedUser:self.activities[indexPath.row].followItem.user.givenName];

What fixed it: ObjectClass newObject = arrayItem; [self methodCall:newObject] (see full example below)

FollowActivityItem *followItem = self.activities[indexPath.row];
[self tappedUser:followItem.user.givenName];
Alvarez answered 17/7, 2015 at 17:51 Comment(0)
P
0

add #import <Foundation/Foundation.h> to your h file, and use an NSMutableArray instead

Pornocracy answered 13/4, 2016 at 12:43 Comment(0)
F
-1

If anyone gets to this old thread after getting this error in Xcode 9.3 beta 4 with some legacy Objective-C code like I did, here was my fix.

Update:

@property (nonatomic, strong) id<CustomClass> myObject;

To:

@property (nonatomic, strong) NSMutableArray<CustomClass> *myObject;
Fairchild answered 23/3, 2018 at 22:17 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.