Form the References listed at bottom of file.
Q1: On a UITableViewCell constructor:
Objective-C
- (id)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier
Swift
init(style: UITableViewCellStyle, reuseIdentifier: String?)
What is the reuseIdentifier
used for?
A1:
The reuseIdentifier
is used to indicate that a cell can be re-used in a UITableView
. For example when the cell looks the same, but has different content. The UITableView
will maintain an internal cache of UITableViewCell
’s with the reuseIdentifier
and allow them to be re-used when dequeueReusableCellWithIdentifier:
is called. By re-using table cell’s the scroll performance of the tableview is better because new views do not need to be created.
Q2: Explain the difference between atomic and nonatomic synthesized properties?
A2: Atomic and non-atomic refers to whether the setters/getters for a property will atomically read and write values to the property. When the atomic keyword is used on a property, any access to it will be “synchronized”. Therefore a call to the getter will be guaranteed to return a valid value, however this does come with a small performance penalty. Hence in some situations nonatomic is used to provide faster access to a property, but there is a chance of a race condition causing the property to be nil under rare circumstances (when a value is being set from another thread and the old value was released from memory but the new value hasn’t yet been fully assigned to the location in memory for the property).
Q3: Explain the difference between copy and retain?
A3: Retaining an object means the retain count increases by one. This means the instance of the object will be kept in memory until it’s retain count drops to zero. The property will store a reference to this instance and will share the same instance with anyone else who retained it too. Copy means the object will be cloned with duplicate values. It is not shared with any one else.
Q4: What is method swizzling in Objective C and why would you use it?
A4: Method swizzling allows the implementation of an existing selector to be switched at runtime for a different implementation in a classes dispatch table. Swizzling allows you to write code that can be executed before and/or after the original method. For example perhaps to track the time method execution took, or to insert log statements
#import "UIViewController+Log.h"
@implementation UIViewController (Log)
+ (void)load {
static dispatch_once_t once_token;
dispatch_once(&once_token, ^{
SEL viewWillAppearSelector = @selector(viewDidAppear:);
SEL viewWillAppearLoggerSelector = @selector(log_viewDidAppear:);
Method originalMethod = class_getInstanceMethod(self, viewWillAppearSelector);
Method extendedMethod = class_getInstanceMethod(self, viewWillAppearLoggerSelector);
method_exchangeImplementations(originalMethod, extendedMethod);
});
}
- (void) log_viewDidAppear:(BOOL)animated {
[self log_viewDidAppear:animated];
NSLog(@"viewDidAppear executed for %@", [self class]);
}
@end
Q5: What’s the difference between not-running, inactive, active, background and suspended execution states?
OR
What are the different app states?
A5:
-
Not running: The app has not been launched or was running but was terminated by the system.
-
Inactive: The app is running in the foreground but is currently not receiving events. (It may be executing other code though.) An app usually stays in this state only briefly as it transitions to a different state.
-
Active: The app is running in the foreground and is receiving events. This is the normal mode for foreground apps.
-
Background: The app is in the background and executing code. Most apps enter this state briefly on their way to being suspended. However, an app that requests extra execution time may remain in this state for a period of time. In addition, an app being launched directly into the background enters this state instead of the inactive state.
-
Suspended: The app is in the background but is not executing code. The system moves apps to this state automatically and does not notify them before doing so. While suspended, an app remains in memory but does not execute any code. When a low-memory condition occurs, the system may purge suspended apps without notice to make more space for the foreground app.
Q6: What is a category and when is it used?
A6:
A category is a way of adding additional methods to a class without extending it. It is often used to add a collection of related methods. A common use case is to add additional methods to built in classes in the Cocoa frameworks. For example adding async download methods to the UIImage
class.
Q7: Can you spot the bug in the following code and suggest how to fix it?
@interface MyCustomController : UIViewController
@property (strong, nonatomic) UILabel *alert;
@end
@implementation MyCustomController
- (void)viewDidLoad {
CGRect frame = CGRectMake(100, 100, 100, 50);
self.alert = [[UILabel alloc] initWithFrame:frame];
self.alert.text = @"Please wait...";
[self.view addSubview:self.alert];
dispatch_async(
dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0),
^{
sleep(10);
self.alert.text = @"Waiting over";
}
);
}
@end
A7: All UI updates must be done on the main thread. In the code above the update to the alert text may or may not happen on the main thread, since the global dispatch queue makes no guarantees . Therefore the code should be modified to always run the UI update on the main thread
dispatch_async(
dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0),
^{
sleep(10);
dispatch_async(dispatch_get_main_queue(), ^{
self.alert.text = @"Waiting over";
});
});
Q8:
What is the difference between viewDidLoad
and viewDidAppear
? Which should you use to load data from a remote server to display in the view?
A8:
viewDidLoad
is called when the view is loaded, whether from a Xib file, storyboard or programmatically created in loadView
. viewDidAppear
is called every time the view is presented on the device. Which to use depends on the use case for your data. If the data is fairly static and not likely to change then it can be loaded in viewDidLoad
and cached. However if the data changes regularly then using viewDidAppear
to load it is better. In both situations, the data should be loaded asynchronously on a background thread to avoid blocking the UI.
Q9:
What considerations do you need when writing a UITableViewController
which shows images downloaded from a remote server?
A9: This is a very common task in iOS and a good answer here can cover a whole host of knowledge. The important piece of information in the question is that the images are hosted remotely and they may take time to download, therefore when it asks for “considerations”, you should be talking about:
- Only download the image when the cell is scrolled into view, i.e. when
cellForRowAtIndexPath
is called. - Downloading the image asynchronously on a background thread so as not to block the UI so the user can keep scrolling.
- When the image has downloaded for a cell we need to check if that cell is still in the view or whether it has been re-used by another piece of data. If it’s been re-used then we should discard the image, otherwise we need to switch back to the main thread to change the image on the cell.
Other good answers will go on to talk about offline caching of the images, using placeholder images while the images are being downloaded.
Q10: What is a protocol, how do you define your own and when is it used?
A10: A protocol is similar to an interface from Java. It defines a list of required and optional methods that a class must/can implement if it adopts the protocol. Any class can implement a protocol and other classes can then send messages to that class based on the protocol methods without it knowing the type of the class.
@protocol MyCustomDataSource
- (NSUInteger)numberOfRecords;
- (NSDictionary *)recordAtIndex:(NSUInteger)index;
@optional
- (NSString *)titleForRecordAtIndex:(NSUInteger)index;
@end
A common use case is providing a DataSource for UITableView
or UICollectionView
.
Q11: What is KVC and KVO? Give an example of using KVC to set a value.
A11: KVC stands for Key-Value Coding. It's a mechanism by which an object's properties can be accessed using string's at runtime rather than having to statically know the property names at development time. KVO stands for Key-Value Observing and allows a controller or class to observe changes to a property value.
Let's say there is a property name on a class:
@property (nonatomic, copy) NSString *name;
We can access it using KVC:
NSString *n = [object valueForKey:@"name"]
And we can modify it's value by sending it the message:
[object setValue:@"Mary" forKey:@"name"]
Q12: What are blocks and how are they used?
A12:
Blocks are a way of defining a single task or unit of behavior without having to write an entire Objective-C class. Under the covers Blocks are still Objective C objects. They are a language level feature that allow programming techniques like lambdas and closures to be supported in Objective-C. Creating a block is done using the ^ { }
syntax:
myBlock = ^{
NSLog(@"This is a block");
}
It can be invoked like so: myBlock();
It is essentially a function pointer which also has a signature that can be used to enforce type safety at compile and runtime. For example you can pass a block with a specific signature to a method like so:
- (void)callMyBlock:(void (^)(void))callbackBlock;
If you wanted the block to be given some data you can change the signature to include them:
- (void)callMyBlock:(void (^)(double, double))block {
...
block(3.0, 2.0);
}
Q13: What mechanisms does iOS provide to support multi-threading?
A13:
NSThread
creates a new low-level thread which can be started by calling thestart
method.
NSThread* myThread = [[NSThread alloc] initWithTarget:self
selector:@selector(myThreadMainMethod:)
object:nil];
[myThread start];
NSOperationQueue
allows a pool of threads to be created and used to executeNSOperations
in parallel.NSOperations
can also be run on the main thread by askingNSOperationQueue
for themainQueue
.
NSOperationQueue* myQueue = [[NSOperationQueue alloc] init];
[myQueue addOperation:anOperation];
[myQueue addOperationWithBlock:^{
/* Do something. */
}];
- GCD or Grand Central Dispatch is a modern feature of Objective-C that provides a rich set of methods and API's to use in order to support common multi-threading tasks. GCD provides a way to queue tasks for dispatch on either the main thread, a concurrent queue (tasks are run in parallel) or a serial queue (tasks are run in FIFO order).
dispatch_queue_t myQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_async(myQueue, ^{
printf("Do some work here.\n");
});
Q14: What is the Responder Chain?
A14:
When an event happens in a view, for example a touch event, the view will fire the event to a chain of UIResponder
objects associated with the UIView
. The first UIResponder
is the UIView
itself, if it does not handle the event then it continues up the chain to until UIResponder
handles the event. The chain will include UIViewController
's, parent UIView
's and their associated UIViewController
's, if none of those handle the event then the UIWindow
is asked if it can handle it and finally if that doesn't handle the event then the UIApplicationDelegate
is asked.
If you get the opportunity to draw this one out, it's worth doing to impress the interviewer:
Q15: What's the difference between using a delegate and notification?
A15:
Both are used for sending values and messages to interested parties. A delegate is for one-to-one communication and is a pattern promoted by Apple. In delegation the class raising events will have a property for the delegate and will typically expect it to implement some protocol
. The delegating class can then call the _delegate_s protocol methods.
Notification allows a class to broadcast events across the entire application to any interested parties. The broadcasting class doesn't need to know anything about the listeners for this event, therefore notification is very useful in helping to decouple components in an application.
[NSNotificationCenter defaultCenter]
postNotificationName:@"TestNotification"
object:self];
Q16:
What's your preference when writing UI's? Xib files, Storyboards or programmatic UIView
?
A16: There's no right or wrong answer to this, but it's great way of seeing if you understand the benefits and challenges with each approach. Here's the common answers I hear:
-
Storyboard's and Xib's are great for quickly producing UI's that match a design spec. They are also really easy for product managers to visually see how far along a screen is.
-
Storyboard's are also great at representing a flow through an application and allowing a high-level visualization of an entire application.
-
Storyboard's drawbacks are that in a team environment they are difficult to work on collaboratively because they're a single file and merge's become difficult to manage.
-
Storyboards and Xib files can also suffer from duplication and become difficult to update. For example if all button's need to look identical and suddenly need a color change, then it can be a long/difficult process to do this across storyboards and xibs.
-
Programmatically constructing
UIView
's can be verbose and tedious, but it can allow for greater control and also easier separation and sharing of code. They can also be more easily unit tested.
Most developers will propose a combination of all 3 where it makes sense to share code, then re-usable UIView
's or Xib
files.
Q17: How would you securely store private user data offline on a device? What other security best practices should be taken?
A17: Again there is no right answer to this, but it's a great way to see how much a person has dug into iOS security. If you're interviewing with a bank I'd almost definitely expect someone to know something about it, but all companies need to take security seriously, so here's the ideal list of topics I'd expect to hear in an answer:
-
If the data is extremely sensitive then it should never be stored offline on the device because all devices are crackable.
-
The keychain is one option for storing data securely. However it's encryption is based on the pin code of the device. User's are not forced to set a pin, so in some situations the data may not even be encrypted. In addition the users pin code may be easily hacked.
-
A better solution is to use something like SQLCipher which is a fully encrypted SQLite database. The encryption key can be enforced by the application and separate from the user's pin code.
Other security best practices are:
-
Only communicate with remote servers over SSL/HTTPS.
-
If possible implement certificate pinning in the application to prevent man-in-the-middle attacks on public WiFi.
-
Clear sensitive data out of memory by overwriting it.
-
Ensure all validation of data being submitted is also run on the server side.
Q18: What is MVC? How is it implemented in iOS? What are some pitfalls you've experienced with it? Are there any alternatives to MVC?
A18:
MVC stands for Model, View, Controller. It is a design pattern that defines how to separate out logic when implementing user interfaces. In iOS, Apple provides UIView
as a base class for all View's, UIViewController
is provided to support the Controller which can listen to events in a View and update the View when data changes. The Model represents data in an application and can be implemented using any NSObject
, including data collections like NSArray
and NSDictionary
.
Some of the pitfalls that people hit are bloated UIViewController
and not separating out code into classes beyond the MVC format. I'd highly recommend reading up on some solutions to this:
- https://www.objc.io/issues/1-view-controllers/lighter-view-controllers/
- https://speakerdeck.com/trianglecocoa/unburdened-viewcontrollers-by-jay-thrash
- https://programmers.stackexchange.com/questions/177668/how-to-avoid-big-and-clumsy-uitableviewcontroller-on-ios
In terms of alternatives, this is pretty open ended. The most common alternative is MVVM using ReactiveCocoa, but others include VIPER and using Functional Reactive code.
Q19: A product manager in your company reports that the application is crashing. What do you do?
A19: This is a great question in any programming language and is really designed to see how you problem solve. You're not given much information, but some interviews will slip you more details of the issue as you go along. Start simple:
- Get the exact steps to reproduce it.
- Find out the device, iOS version.
- Do they have the latest version?
- Get device logs if possible.
Once you can reproduce it or have more information then start using tooling. Let's say it crashes because of a memory leak, I'd expect to see someone suggest using Instruments leak tool. A really impressive candidate would start talking about writing a unit test that reproduces the issue and debugging through it.
Other variations of this question include slow UI or the application freezing. Again the idea is to see how you problem solve, what tools do you know about that would help and do you know how to use them correctly.
Q20: What is AutoLayout? What does it mean when a constraint is "broken" by iOS?
A20:
AutoLayout is way of laying out UIView
's using a set of constraints that specify the location and size based relative to other views or based on explicit values. AutoLayout makes it easier to design screens that resize and layout out their components better based on the size and orientation of a screen. Constraint's include:
- Setting the horizontal/vertical distance between 2 views
- Setting the height/width to be a ratio relative to a different view
- A width/height/spacing can be an explicit static value
Sometimes constraints conflict with each other. For example imagine a UIView
which has 2 height constraints: one says make the UIView
200px high, and the second says make the height twice the height of a button. If the iOS runtime can not satisfy both of these constraints then it has to pick only one. The other is then reported as being "broken" by iOS.