严格来讲,AppDelegate除了负责应用生命周期之外,不应该再有多余的责任。
但在iOS 实际开发过程中,很多人习惯将全局变量定义在 AppDelegate 中,因为任何项目都会访问 AppDelegate,但如此这般,AppDelegate就变得更加臃肿。
在大型项目中,想要解除耦合和模块化时,都不利于AppDelegate的维护。
所以最好的办法就是
- 将很多全局变量放在AppDelegate上,将AppDelegate作为一个依赖中心点,虽然很多模块可以访问全局变量,但会产生相互依赖,不利于模块化开发;
- 不同的模块,会根据需要在AppDelegate的不同生命周期方法中,调用各种方法。比如在程序启动时在
application:didFinishLaunchingWithOptions:
内注册一个初始化方法;
怎么解决上述两个问题:
对于Question1, 我们可以将全局变量移除,最简单的解决方式就是将类实现为单例,各个类可以提供单例来实现全局访问。
Question2: 既想解除各个模块和AppDelegate的耦合,又想要在不同生命周期完成方法的调用,最好的办法就是暴漏出一个钩子方法,这样每个模块就可以知晓不同的生命周期,然后执行不同操作。
今天看到豆瓣提供的第三方库FRDModuleManager
,有些思考,写下来与大家共享下。
FRDModuleManager的核心是利用了钩子方法,在 FRDModuleManager 被 UIApplicationDelegate 各方法内留下的钩子调用时,会调用注册的每个模块的相同的方法。这样每个模块就都知晓了应用的生命周期。
具体代码如下:
FRDModuleManager类:
- 一个单例对象
- 同时定义了从 plist 文件中添加 module 的单例方法
- 里面包含一个遵守UIApplicationDelegate的协议FRDModule
凡是遵守了FRDModule的模块都可以获取生命周期的方法,然后根据需要,在不同生命周期方法中,完成必要操作,如此即可充分的解耦。
具体的实例可以参考一下代码:
-
加载所有模块,文件从 plist中获取,将不同的模块填写到 plist 文件中。
NSString* plistPath = [[NSBundle mainBundle] pathForResource:@"ModulesRegister" ofType:@"plist"]; FRDModuleManager *manager = [FRDModuleManager sharedInstance]; [manager loadModulesWithPlistFile:plistPath];
-
在 UIApplicationDelegate 各方法中留下钩子
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { NSString* plistPath = [[NSBundle mainBundle] pathForResource:@"ModulesRegister" ofType:@"plist"]; FRDModuleManager *manager = [FRDModuleManager sharedInstance]; [manager loadModulesWithPlistFile:plistPath]; [manager application:application didFinishLaunchingWithOptions:launchOptions]; return YES; } - (void)applicationWillResignActive:(UIApplication *)application { [[FRDModuleManager sharedInstance] applicationWillResignActive:application]; } - (void)applicationDidEnterBackground:(UIApplication *)application { [[FRDModuleManager sharedInstance] applicationDidEnterBackground:application]; } - (void)applicationWillEnterForeground:(UIApplication *)application { [[FRDModuleManager sharedInstance] applicationWillEnterForeground:application]; } - (void)applicationDidBecomeActive:(UIApplication *)application { [[FRDModuleManager sharedInstance] applicationDidBecomeActive:application]; } - (void)applicationWillTerminate:(UIApplication *)application { [[FRDModuleManager sharedInstance] applicationWillTerminate:application]; }
- 在特定模块,根据需求完成注册即可,比如以下为FRDTimelineModule的模块
//这是一个遵守了FRDModule协议的模块FRDTimelineModule.h
@interface FRDTimelineModule : NSObject<FRDModule>
@end
//FRDTimelineModule.m
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
NSLog(@"%@ Timeline", NSStringFromSelector(_cmd));
return YES;
}