AppDelegate的解耦和瘦身

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