AppDelegate的解耦和瘦身

严格来讲,AppDelegate除了负责应用生命周期之外,不应该再有多余的责任。

但在iOS 实际开发过程中,很多人习惯将全局变量定义在 AppDelegate 中,因为任何项目都会访问 AppDelegate,但如此这般,AppDelegate就变得更加臃肿。
在大型项目中,想要解除耦合和模块化时,都不利于AppDelegate的维护。
所以最好的办法就是

  1. 将很多全局变量放在AppDelegate上,将AppDelegate作为一个依赖中心点,虽然很多模块可以访问全局变量,但会产生相互依赖,不利于模块化开发;
  2. 不同的模块,会根据需要在AppDelegate的不同生命周期方法中,调用各种方法。比如在程序启动时在application:didFinishLaunchingWithOptions:内注册一个初始化方法;

怎么解决上述两个问题:

对于Question1, 我们可以将全局变量移除,最简单的解决方式就是将类实现为单例,各个类可以提供单例来实现全局访问。
Question2: 既想解除各个模块和AppDelegate的耦合,又想要在不同生命周期完成方法的调用,最好的办法就是暴漏出一个钩子方法,这样每个模块就可以知晓不同的生命周期,然后执行不同操作。

今天看到豆瓣提供的第三方库FRDModuleManager,有些思考,写下来与大家共享下。
FRDModuleManager的核心是利用了钩子方法,在 FRDModuleManager 被 UIApplicationDelegate 各方法内留下的钩子调用时,会调用注册的每个模块的相同的方法。这样每个模块就都知晓了应用的生命周期。

具体代码如下:

FRDModuleManager类:

  1. 一个单例对象
  2. 同时定义了从 plist 文件中添加 module 的单例方法
  3. 里面包含一个遵守UIApplicationDelegate的协议FRDModule

凡是遵守了FRDModule的模块都可以获取生命周期的方法,然后根据需要,在不同生命周期方法中,完成必要操作,如此即可充分的解耦。

具体的实例可以参考一下代码:

  1. 加载所有模块,文件从 plist中获取,将不同的模块填写到 plist 文件中。

    NSString* plistPath = [[NSBundle mainBundle] pathForResource:@"ModulesRegister" ofType:@"plist"];
    FRDModuleManager *manager = [FRDModuleManager sharedInstance];
    [manager loadModulesWithPlistFile:plistPath];
    
  2. 在 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];
    }
    
  1. 在特定模块,根据需求完成注册即可,比如以下为FRDTimelineModule的模块
//这是一个遵守了FRDModule协议的模块FRDTimelineModule.h
@interface FRDTimelineModule : NSObject<FRDModule>

@end

//FRDTimelineModule.m
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
    {
      NSLog(@"%@  Timeline", NSStringFromSelector(_cmd));
      return YES;
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 在实际的开发过程中,AppDelegate应该除了负责应用生命周期之外,不应该再有多余的责任。但是往往在一个项目的...
    东了个尼阅读 2,350评论 0 6
  • iOS 開発の結構 画面 UI UIWebview [[UIApplication sharedApplicati...
    RencaiXiong阅读 604评论 0 0
  • 基础 1. 为什么说Objective-C是一门动态的语言? 2. 讲一下MVC和MVVM,MVP? 3. 为...
    波妞和酱豆子阅读 3,361评论 0 46
  • 上官网注册账号 首先来到环信的官网,然后登陆.没有账号先注册一个. 进去之后创建应用,如图 创建应用界面 点击确定...
    loneWolf01阅读 525评论 0 0
  • 基点教育的小伙伴们,大家好,我是梁瑞,首先非常感谢基点给我这个展示自己和实现自己价值的平台。今天能做为演讲冠军站在...
    简书jianshu1阅读 213评论 0 0