Google Groups no longer supports new Usenet posts or subscriptions. Historical content remains viewable.
Dismiss

Obj-C - Categories on Protocols

2 views
Skip to first unread message

Derek Fawcus

unread,
Jun 20, 2020, 5:00:08 PM6/20/20
to discuss...@gnu.org
I recently read the ACM article on the history of Obj-C [1],
and noticed it indicated that the intended ability to create
a category on a protocol as never having shipped as a feature,
but that it did ship as a thing in Swift [2].

The references at the end of the ACM paper state:

"Extensions on Protocols allow programmers to add methods and their
implementations to a protocol which all conforming classes will
automatically gain. We refer specifically to the section of the
website describing Extensions on Protocols in Swift to document
that this functionality originally proposed for Objective-C in 1995
but never shipped finally became available in the Swift successor
language in 2014."


So I wonder, does this mean that the Apple Obj-C runtime happens to
include this mechanism, despite it not being exposed to the language,
or is it only provided by the Swift runtime?

DF

[1] https://dl.acm.org/doi/10.1145/3386332
(The origins of Objective-C at PPI/Stepstone and its evolution at NeXT)

[2] https://docs.swift.org/swift-book/LanguageGuide/Protocols.html#ID521

David Chisnall

unread,
Jun 21, 2020, 5:51:32 AM6/21/20
to Derek Fawcus, discuss...@gnu.org
Hi,

You don’t actually need anything new from the runtime to support this. I implemented something equivalent in the EtoileFoundation framework around 2007 and GNUstep also has something similar in -Base Additions.

Categories are somewhat problematic in Objective-C because there is no defined composition order. A lot of other languages have solved this in various ways. In Verona, we’re addressing it by allowing default methods on interfaces not allowing subclassing and getting most the benefits (and almost none of the overhead) that Smalltalk and Objective-C derive from this dynamic behaviour from a structural and algebraic type system.

Categories, in the runtime, are just a little bit of metadata with a list of methods and a class to apply them to. You could make this entirely declarative by defining a protocol like GSConcreteProtocol (containing no methods) and then define a class that implements that protocol and another protocol with optional methods, so you’d write something like this:

@protocol Example
@required
- (void)mustBeImplemented;
@optional
- (void)hasDefaultImplementation;
@end

@interface ExampleConcreteProtocol : NSObject<Example, GSConcreteProtocol> @end
@implementation ExampleConcreteProtocol
- (void)mustBeImplemented { /* Keep the compiler happy */ }
- (void)hasDefaultImplementation
{
// Some code goes here
}
@end


You’d then use the runtime’s introspection to get the list of all classes and first find all classes that conform to the GSConcreteProtocol protocol. Then you’d probably do some sanity checking (e.g. that they inherit directly from NSObject and have no other ivars) and then, for each of these classes, you’d find the protocols that they conformed to.

Next, you’d find all other classes that conformed to that protocol. You’d then iterate over all of the optional methods in the protocol and find ones that are not implemented by the class, then copy the method from the concrete protocol class into the target.

The end result is that you provide a class that implements the optional methods in a protocol and then any class that adopts the protocol but does not define the methods gets the default implementation.

This requires that the runtime support protocol introspection to find optional methods. This should all be working with both v1 and v2 GNUstep ABIs.

David
0 new messages