美文网首页iOS小记iOS 奔跑吧 iOS
AppDelegate的解耦和瘦身

AppDelegate的解耦和瘦身

作者: CoderDancer | 来源:发表于2016-11-08 18:19 被阅读1459次

    严格来讲,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;
        }
    

    相关文章

      网友评论

      • 清蒸鱼跃龙门:全局变量是给项目里所以类都可以用的,那这跟单例有什么关系 没搞懂
      • Link913:一直讲钩子的我到现在其实还是不是很懂...
        kirito_song:@已删号这名字都有人用 iOS开发里、hook属于是脱离初级的第一步吧、普遍第一次接触runtime就在这
        kirito_song:@已删号这名字都有人用 钩子hook?搜一搜swizzle。想再深入搜一搜越狱开发
        Josscii:@SkyHarute 在我的理解, 钩子方法就像回调一样, 作用是让客户端在某个方法被调用时收到消息,从而能够"勾住"来做一些事情.

      本文标题: AppDelegate的解耦和瘦身

      本文链接:https://www.haomeiwen.com/subject/kltzuttx.html