It was not all that long ago when I decided I wanted to switch things up and move from 5+ years of web development to native mobile, specifically iOS. When Swift was announced, my desire to jump into iOS increased! I would be able to learn a language at the same time as everyone else (without being years behind). While that was all fine and good, there came a time I started thinking about how much I really ought to learn Objective-C. ðĪ Yeah sure… I could have probably got by knowing just Swift, but there’s a whole other world out there! ð When it comes to understanding iOS engineering at a deeper level, knowing Objective-C is a must. And… since I don’t come from a CS background… I really don’t have all too much experience with C. Learning this crazy bracket mayhem known as Objective-C will be good for me… it will… it will… ð
What to Expect From This Post
I want to write about my recent learnings so I have a place to reflect (I have a horrible memory, so I often need to come back to everything I learn). I likely (hopefully?) won’t be writing a ton of Objective-C in the near future, so cementing my understanding and having something to reference is key. Additionally, I found there aren’t too many Objective-C learning resources geared at experienced iOS developers (with Swift knowledge only)… maybe because there aren’t too many going from Swift to Objective-C? But… there is at least one (me ð), and there are likely others, too. If you’re one of those people, hopefully this will help! Additionally, if you are an Objective-C master and for some reason still reading, please feel free to correct me in the comments! ð
One way of ensuring I hit on everything I learned is construction of a simple app, a list of US Presidents in a table view. Most of what’s about to follow is included in that app. Skip down to the bottom of this article for a link.
The Basics
First, let’s start with some of the basic differences between Swift and Objective-C. I’ve found once I jump that hurdle of the minor syntactical differences, all is once again good in the world.
Basic Syntax
Ok, so you’re probably aware Objective-C loves these ð []
. Take a minute now to introduce your fingers to these keys you rarely use anywhere else in life. So let’s call some methods in Objective-C and Swift to see the differences.
Swift
var someClass = SomeClass()
someClass.someStringProperty = "ðĪš"
someClass.doSomething(with: "ð")
Objective-C
// note: alternatively can by initialized with [SomeClass new]
SomeClass *someClass = [[SomeClass alloc] init];
someClass.someStringProperty = @"ðĪš";
[someClass doSomethingWith:@"ð"]
Alright, so there’s a few more things going on here beyond the brackets.
- What’s the
@
before quotes? There is a simple answer for this, it has to do with identifying differences between Objective-C and C/C++ (more information here for those who are curious), but you always use it when defining a string or an array literal. - Why
alloc
andinit
? This goes back to initializing classes in C. First the class needs memory allocation, then you call its initializer. The shorthand for this is just callingnew
(more reading on differences here). - Semicolons… Oh, forgot to mention. Use them. You have to. ð
- Why an asterisk when defining a property? This is also around from C. It means we’re storing a reference to an object and not the object itself (also known as a pointer). This syntax is used for all non-primitive types, so strings, custom classes, arrays, etc. (e.g. a string would be initialized as
NSString *someString = "hi ð";
). - But… dot syntax for
someStringProperty
? ðŦ Yeah, that was confusing for me when first looking at Objective-C. The easy answer is you should use dot syntax when getting or setting a property on an instance of a class. It actually has to do with using a property accessor vs an instance variable, details on that below.
Classes
Header files
One of the biggest differences between Swift and Objective-C, and likely one you’ve already seen if you’ve seen or written any Objective-C code, is multiple files for each class - a header file (.h) and an implementation file (.m). As much as I prefer Swift in almost every way, header files are in fact quite nice. It gives you an easy way to see all public methods at a quick glance. It also removes some of the confusing keywords (e.g. private
, fileprivate
, … ðĩ). So how do they work? Let’s take a look at a simple class in both Objective-C and Swift.
Swift (Car.swift)
class Car {
let numberOfWheels = 4
var color: String?
func changeColor(to color: String) {
self.color = color
}
}
Objective-C (DSCar.h)
@interface DSCar : NSObject
@property NSString *color;
- (void)changeColorTo:(NSString *)color;
@end
Objective-C (DSCar.m)
@interface DSCar ()
@property int numberOfWheels;
@end
@implementation DSCar
- (instancetype)init {
// Don't worry about this syntax, it's just the standard way to setup initializers.
// We're just setting a default value for `numberOfWheels`, which needs to be done
// in an initializer.
if (self = [super init]) {
self.numberOfWheels = 4;
}
return self;
}
- (void)changeColorTo:(NSString *)color {
self.color = color;
}
@end
Few things to note about all of this:
DSCar
vsCar
… huh? Objective-C doesn’t allow namespacing so it’s good practice to namespace your classes yourself to avoid collisions. This is why you have things likeNSNotificationCenter
andUIButton
. ð Note: As pointed about by @tewha on the comments, this is really only important when building frameworks and not necessary in app code.- What’s private and what’s not? ðĩ At first glance it may seem a bit confusing with
@interface
s and@implementation
s scattered all over the place (at least it did for me). We’ll talk about those keywords a bit later, but for now the easiest way to know what’s public is only what’s defined in the header file. So, this class only allows you to see itscolor
property andchangeColorTo(_:)
method. See the categories section further down for details on hownumberOfWheels
is private (via a class extension). - Why is there an initializer in the Objective-C version? This is necessary to set the default value for
numberOfWheels
. Don’t worry too much about the weird syntax around it, this is generally how all initializers look in Objective-C. If you’re curious as to why, read this.
So much more code to define a simple class! I know. ð
Methods
The syntax for methods is a bit tricky and took me a bit to get used to. No, I lie, I’m still not used to it. Let me try to break it down a bit. Let’s look at a simple method in both Swift and Objective-C.
Swift
func doSomethingCool(with thisThing: String, numberOfTimes: Int) -> Bool {
return true;
}
Objective-C
- (bool)doSomethingCoolWith:(NSString *)thisThing
numberOfTimes:(int)numberOfTimes {
return YES;
}
First, the -
identifier. This just means we have an instance method. For class methods, it changes to a +
. Next, the method name itself includes the name of the first parameter. All other parameters follow (usually on newlines, auto indentation will take care of this for you) with the internal and external parameter names. The return type is at the beginning of the method. If there isn’t a return, this becomes void
. Calling this method would then look something like…
Swift
something.doSomething(with: "SomeString", numberOfTimes: 5)
Objective-C
[something doSomethingCoolWith:"SomeString" numberOfTimes:5];
Compiler directives are all the keywords with an @
at the beginning. This one threw me for a bit of a loop when first looking at Objective-C code. Why does @interface
go in both the header and implementation file? Why are there no curly braces and what’s with @end
? ðĩ (hmm… lots of dizzy faces in this post… ðĪ). So here’s a list of them all and what they do.
@interface
…@end
: This is what surrounds the properties and methods you define in your header (.h) file. It can also be used for categories (Swift extensions, see below) or class extensions (also more below under categories section), which is the only time you’ll see it in an implementation (.m) file.@implementation
…@end
: This is what surrounds most of your code in your implementation file, which the only exception being@interface
mentioned above. Essentially this is what you would put in your swift file when defining a class.@property
: Instructs the compiler to write accessor methods (get and set) for a property. More about this under the property accessors section, but just know this is how you define properties. Public properties are defined in the header file, and private properties in the implementation file (using@interface
…@end
). Oh, and one more thing (see what I did there? ð)… with property definitions, come a list of parenthetical options which instruct the compiler how to auto-generate these accessor methods. You have probably seen this, it looks something like@property (readonly nonatomic weak)
. Here are some examples:readonly
/rewrite
(default:rewrite
): Use a getter, no setter.nonatomic
/atomic
(default:atomic
): Nonatomic removes lock which prevents access from multiple threads. Should always be removed because of cost and checked at a higher level in program.weak
/strong
/copy
/assign
(default:strong
):weak
is the same as in Swift.copy
will create a strong reference, but the instance will be immutable (e.g.NSMutableString
passed as an immutableNSString
). Good habit to use when mutable subclass exists. (thanks for clarifying the differences here, @tewha… more in comments)
@synthesize
: Tells the compiler that an instance variable is backing an accessor (e.g._numberOfWheels
backsnumberOfWheels
).@class
: Tells the complier there’s a class, used as a promise in the header file. Can also be used for compiler efficiency so header file does not need to be recompiled when dependent class is modified. Note: Class still needs to imported in the implementation file.
Check out this NSHipster post for even more on compiler directives.
Inheritance and Protocols
Inheritance and conforming to protocols are pretty straightforward and similar to Swift. Conforming to a protocol is maybe even a bit more clear syntactically in Objective-C. Here’s an example of a protocol and conforming class in both Swift and Objective-C.
Swift
protocol SomeProtocol {
func someMethod(with someParameter: String)
}
class SomeClass: BaseClass, SomeProtocol {
...
}
Objective-C (.h)
@protocol SomeProtocol
- (void)someMethodWithSomeParameter:(NSString *)someParameter;
@end
@interface SomeClass : BaseClass <SomeProtocol>
...
@end
As we discussed before (under Compiler Directives), @property
gives us property accessors for instance variables. Instance variables are where the value/reference is actually being stored and @property
just gives us setters and getters. We could instead write these methods ourselves, which would look something like this:
- (void)setNumberOfWheels:(int)numberOfWheels {
_numberOfWheels = numberOfWheels;
}
- (int)numberOfWheels {
return _numberOfWheels;
}
It’s important to note, property accessors call a method, which you don’t get from accessing an instance variable directly. Why is that important? There may be important side effects in the setter (or getter) that will not be called by accessing the instance variable directly.
Categories
Categories are pretty much the same as extensions in Swift, it’s basically just a difference in syntax.
Swift
extension Array {
func someNewArrayMethod {
...
}
}
Objective-C
@interface NSArray (SomeCategoryName)
- (void)someNewArrayMethod;
@end
Class extensions (which btw are generated by Xcode with all new view controllers) are categories used to define private properties in your implementation file. This is often used for private properties.
@interface ViewController ()
@property UITableView *tableView;
@property NSArray<DataType *> *data;
@end
Blocks
Last but not least, blocks. These are pretty much the exact same as closures in Swift, but with much weirder and confusing syntax (cue dizzy face: ðĩ). The best way to keep track of this is probably not memorization, but instead bookmarking this gem of a website: http://fuckingblocksyntax.com. Let’s look at an example:
Swift
let someClosure = { (param1, param2) in
...
}
someClosure("hello", "world")
Objective-C
void (^someBlock)(NSString *, NSString *) = ^void(NSString *param1, NSString *param2) {
...
};
someBlock(@"hello", @"world");
At least with blocks, the syntax is pretty much the same once you get around the weird ^
s all over the place. Honestly, Swift’s closure syntax isn’t all that intuitive either… ðĪ·ââïļ
Last few things to note about blocks:
- Use
typedef
(liketypealias
) to name blocks and make them less confusing to the reader. - Variables captured by a block are constant within the block, using
__block
keyword when declaring the external variable (yeah, annoying ðĄ). - Like in Swift, use weak references to prevent strong reference cycles. This is done using the
__weak
keyword prior to the block definition (looks something like:__weak UIViewController *weakSelf = self
).
Swift/Objective-C Interoperability
There’s quite a bit to talk about here, especially when talking about how to write Swift code compatible for Objective-C (meh, no one does that, right? ð). Maybe I’ll follow-up with another post specific to this topic. For now, one thing I want to mention important for writing Objective-C code to be used in Swift. Remember the list of options used when defining a property? Well here’s a new one to use to account for Swift’s optionals: nullable
/nonnull
.
* Property annotation used to indicate whether property should be an optional in Swift.
* Use NS_ASSUME_NONNULL_BEGIN/END
to wrap a bunch of properties as nonnull
at a time (reverse not possible).
Example Project
I tried to create an extremely simple Objective-C project to help understand the concepts when starting a new project from scratch. It simply threads a JSON file of US presidents and displays them in a table view. Take a look and play around with it to help better your understanding of some of these concepts.
That’s it! For now… maybe a second part will follow at some point in the future as I continue my journey. Oh, and SUPER HUGE thank you to Mikey Ward Mikey Ward and Big Nerd Ranch for providing me most of this knowledge in a recent Objective-C training course. ð Again, please feel free to comment with anything I may have missed, you’d like to see in a follow-up post, or is just plain wrong. Thanks for reading!
comments powered by Disqus