Say we have parent class
ParentViewController.h
@interface ParentViewController
....
@end
ParentViewController.m
@interface ParentViewController()
- (NSArray *)selectedItems;
@end
@implementation ParentViewController
.....
@end
And then we subclass it
ChildViewController.h
@interface ChildViewController : ParentViewController
....
@end
ChildClassViewController.m
@implementation ChildViewController
- (void)doSomething
{
// XCode Warning Flag on this line
NSUInteger count = [self selectedItems];
.....
}
XCode will set Warning flag at the commented line and says that “Instance method ‘-selectedItems’ not found (return type defaults to ‘id’).
Yes I know that in ObjC there is no such thing as private methods, but using an empty category kind of gives the ability to do so. But somehow it does not get inherited by subclasses.
I usually fix it by moving the method from ParentViewController.m to ParentViewController.h. This feels weird, I loose the ability to make the method private just because I need to subclass it.
Now my question is:
-
Why does the parent subclass cannot find those methods that is declared in its category at the .m file?
-
Is there a way to remove the Warning Flag but without losing the ability to keep the method private.
Hopefully someone with more experience will be able to help explain this annoying issue.
First, note that your “empty category” isn’t a Category at all, it’s a Class Extension. Class Extensions very similar to categories but they’re new in Objective C 2.0, and they differ slightly in their use. Primarily, the compiler will warn you if you don’t implement a method in a Class Extension, but it won’t with a Category. Anyways, on to the issue at hand…
Privacy in Objective-C is all about visibility. If the compiler can’t see see the declaration of a method that’s being used, you’ll get a warning. Note that if you were to implement your subclass in the same file as your Class Extension, the compiler won’t warn you because it can see the declaration.
Therefore, If you want to use “private” methods in subclasses, you just need some way of showing the compiler that those methods exist. My favorite pattern is to declare the private methods in a Category in a separate file (like MyClass_private.h). You then only import that interface into the implementation files of the base class and any derived classes that need to see it.