Understanding the iOS Delegate Pattern: A Simple Guide
The delegate pattern is one of the most common design patterns you’ll encounter in iOS development. If you’ve used UIKit, you’ve already interacted with it, perhaps without fully understanding how it works. In this post, I’ll break down the delegate pattern in simple terms and show you how to implement it in your own code.
What is the Delegate Pattern?
At its core, the delegate pattern is a communication technique that allows one object to send messages to another object when specific events occur.
Think of it like this: imagine you’re at a restaurant. You (the customer) tell the waiter what you want to eat. The waiter doesn’t cook the food themselves - they take your order to the kitchen. Later, when your food is ready, the waiter brings it back to you.
In this analogy:
- The kitchen is the object that does the work (the delegating object)
- You are the delegate (the object that responds to events)
- The waiter is the delegate protocol (the communication channel)
Why Use the Delegate Pattern?
The delegate pattern offers several benefits:
- Separation of concerns: Objects can focus on their core responsibilities and delegate other tasks
- Loose coupling: Objects don’t need to know details about each other, just the protocol they agree on
- One-to-one communication: Clear, direct communication between two objects
- Reusability: The same object can serve as a delegate for multiple objects
How to Implement the Delegate Pattern
Implementing the delegate pattern involves three main components:
- Protocol: Define the methods that the delegate must implement
- Delegating Object: Contains a reference to the delegate and calls its methods
- Delegate: Implements the protocol methods to respond to events
Step-by-Step Implementation Example
Let’s examine a real-world example of the delegate pattern using an Indian stock tracking app.
1. Define the Protocol
First, we define a protocol that outlines what the delegate needs to implement:
protocol StockManagerDelegate: AnyObject {
func didUpdatePrice(stockManager: StockManager)
}
Notice that:
- We use
AnyObject
to ensure that only classes (not structs) can adopt this protocol - We define a method that will be called when stock prices are updated
- The method includes a reference to the sender (
stockManager
) so the delegate knows where the event came from
2. Create the Delegating Object
Next, we create the object that will delegate tasks:
class StockManager {
weak var delegate: StockManagerDelegate?
var stocks = [
Stock(symbol: "RELIANCE", name: "Reliance Industries Ltd.", price: 2450.75),
// other stocks...
]
func refreshPrices() {
for i in 0..<stocks.count {
let randomChange = Double.random(in: -5...5)
stocks[i].price += randomChange
}
delegate?.didUpdatePrice(stockManager: self)
}
}
Key points:
- We declare a
delegate
property asweak
to avoid memory leaks (more on this later) - We use optional chaining (
delegate?
) when calling the delegate method, as the delegate might not be set - We pass
self
to the delegate method so it knows which manager is sending the update
3. Implement the Delegate
Finally, we implement the delegate protocol in our view controller:
class StockTrackerViewController: UITableViewController {
let stockManager = StockManager()
override func viewDidLoad() {
super.viewDidLoad()
setupUI()
stockManager.delegate = self
}
// ...other methods...
}
extension StockTrackerViewController: StockManagerDelegate {
func didUpdatePrice(stockManager: StockManager) {
DispatchQueue.main.async {
self.tableView.reloadData()
}
}
}
Notice how:
- We set
self
(the view controller) as thestockManager
’s delegate - We implement the required protocol method to reload the table view when prices update
- We use
DispatchQueue.main.async
to ensure UI updates happen on the main thread
Memory Management with Delegates
An important aspect of the delegate pattern is proper memory management. Always declare your delegate properties as weak
to avoid strong reference cycles (also called retain cycles).
weak var delegate: SomeProtocol?
This is crucial because:
- The delegating object (e.g.,
StockManager
) holds a reference to its delegate - The delegate (e.g.,
StockTrackerViewController
) often holds a reference to the delegating object - Without using
weak
, these objects would keep each other in memory forever, causing a memory leak
This is why delegate protocols typically require the AnyObject
conformance - because only class types can be marked as weak
.
Real-World Examples in UIKit
UIKit uses the delegate pattern extensively:
UITableView
hasUITableViewDelegate
andUITableViewDataSource
UITextField
hasUITextFieldDelegate
UIScrollView
hasUIScrollViewDelegate
For example, when you implement tableView(_:didSelectRowAt:)
, you’re using the delegate pattern to respond to a table view’s row selection event.
When to Use the Delegate Pattern
The delegate pattern is most appropriate when:
- You need one-to-one communication between objects
- An object needs to notify another when specific events occur
- You want to customize the behavior of a reusable component
- You need to communicate back from a child view controller to a parent
Alternatives to Delegates
While delegates are powerful, they’re not the only communication pattern in iOS:
- Callbacks/Closures: For simpler scenarios where you don’t need a formal protocol
- Notification Center: For one-to-many communication
- Key-Value Observing (KVO): For observing property changes
- Combine: Apple’s reactive framework for handling asynchronous events
Conclusion
The delegate pattern is a fundamental communication technique in iOS development that enables loose coupling between objects. By understanding and implementing this pattern, you’ll write more maintainable, flexible code that follows Apple’s design guidelines.
In our stock tracker example, we used delegates to notify the view controller when stock prices changed, allowing it to update the UI accordingly. This separation of concerns keeps our code organized and makes each component more reusable.