在Swift中使用Storyboard和Segue時的依賴注入
Demo下載
我們都知道在使用Storyboard時,實現依賴注入總是有點不優雅,讓我們先來看看在Objective-C時如何使用:
Objective-C
//In RJDemoViewController.m
- (void)setDependenciesViewModel:(RJDemoViewModel *)viewModel delegate:(id <RJDemoViewControllerDelegate>)delegate
{
self.viewModel = viewModel;
self.delegate = delegate;
}
- (void)assertDependencies
{
// 依賴的對象
NSParameterAssert(self.viewModel &&self.delegate);
}
- (void)viewDidLoad
{
[super viewDidLoad];
[self assertDependencies];
}
然后就可以在prepareForSegue
中設置依賴:
- (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender
{
if ([segue.identifier isEqualToString:showDemoViewController]) {
RJDemoViewController *viewController = segue.destinationViewController;
NSParameterAssert([viewController isKindOfClass:[RJDemoViewController class]]);
[viewController setDependenciesViewModel:[self viewModelForSelectedThing] delegate:self];
}
}
至此,當所有的依賴一旦沒有被注入,軟件就會立刻奔潰,錯誤定位也變得十分地方便。
然后在Swift
中有沒有存在更Swift的方式來實現依賴注入吶?
Swift
首先定義一個協議(Protocol):
protocol Injectable {
associatedtype InjectObject
func inject(_: InjectObject)
func assertDependencies()
}
注意func inject(_: InjectObject)
由于ViewController依賴的對象有可能是Number,String
等等各種類型,為了可讀性,所以忽略了inject
方法的參數名,由遵循該協議的ViewController
去自由定義:
class RJDemoViewController: UIViewController, Injectable {
@IBOutlet weak private var mainLabel: UILabel!
private var mainText: String!
override func viewDidLoad() {
super.viewDidLoad()
//由于mainText是String!類型,若沒賦值,這里就會Crash
assertDependencies()
// 大膽地使用mainText
mainLabel.text = mainText
}
//注: 這里參數名定義為text,提高了可讀性
func inject(text: String) {
mainText = text
}
func assertDependencies() {
assert(mainText != nil)
}
}
至此,在prepareForSegue
中,便可實現注入
override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) {
if segue.identifier == "showDemoViewController" {
let vc = segue.destinationViewController as! RJDemoViewController
vc.inject("myMainText")
}
}