美文网首页iOS开发精进iOS开发笔记iOS Dev
iOS开发细节 | 通知怎么写?

iOS开发细节 | 通知怎么写?

作者: Lol刀妹 | 来源:发表于2017-11-23 14:06 被阅读804次
    iu

    前言

    工作的这一年多时间里我见识了各种通知的写法,今天总结一下。

    第一种:

    最纯粹的,如:

    [[NSNotificationCenter defaultCenter] postNotificationName:@"aNotification" object:nil];
    

    第二种:

    在PCH文件中写宏,如:

    #define aNotification @"aNotification"
    

    第三种:

    写在.h文件中,如:

    static NSString *const aNotification = @"aNotification";
    

    第四种:

    在.m中定义并且在.h中暴露出来,如:

    .h文件:

    extern NSString * const aNotification;
    

    .m文件:

    NSString * const aNotification = @"aNotification";
    

    思考

    除了第一种low到爆之外其余三种貌似都可以,但是最好的只有一种,哪种最好呢?

    《禅与Objective-C编程艺术》里面提到:

    当你定义你自己的 NSNotification 的时候你应该把你的通知的名字定义为一个字符串常量,就像你暴露给其他类的其他字符串常量一样。你应该在公开的接口文件中将其声明为 extern 的, 并且在对应的实现文件里面定义。
    因为你在头文件中暴露了符号,所以你应该按照统一的命名空间前缀法则,用类名前缀作为这个通知名字的前缀。
    同时,用一个 Did/Will 这样的动词以及用 "Notifications" 后缀来命名这个通知也是一个好的实践。

    如:

    // Foo.h
    extern NSString * const ZOCFooDidBecomeBarNotification
    
     // Foo.m
    NSString * const ZOCFooDidBecomeBarNotification = @"ZOCFooDidBecomeBarNotification";
    

    抛开命名规范来说,这与上面提到的第四种写法一样,那么这样写的依据是什么?

    官方文档推荐的是:

    Notifications are identified by global NSString objects whose names are composed in this way:

    [Name of associated class] + [Did | Will] + [UniquePartOfName] + Notification
    

    For example:

    NSApplicationDidBecomeActiveNotification
    NSWindowDidMiniaturizeNotification
    NSTextViewDidChangeSelectionNotification
    NSColorPanelColorDidChangeNotification
    

    官方文档明确的告诉了我们通知如何命名,但是并没有告诉我们通知写到哪,既然如此就看看苹果的API吧。

    苹果自己的API:

    UIKIT_EXTERN NSNotificationName const UITextFieldTextDidBeginEditingNotification;
    UIKIT_EXTERN NSNotificationName const UITextFieldTextDidEndEditingNotification;
    UIKIT_EXTERN NSNotificationName const UITextFieldTextDidChangeNotification;
    

    这是UITextField.h文件里的内容,是不是一看就觉得逼格很高?和《禅与Objective-C编程艺术》里的写法一样,在.m中定义然后在.h中暴露出去,但是它用到了UIKIT_EXTERNNSNotificationName这两个我并没用用到的单词(姑且叫单词吧。。。)

    按理说我们遵循官方文档再仿照苹果API就可以写出完美的通知了,但是,UIKIT_EXTERNNSNotificationName这两个单词我们能随便用吗?

    先看看UIKIT_EXTERN是什么:

    #ifdef __cplusplus
    #define UIKIT_EXTERN        extern "C" __attribute__((visibility ("default")))
    #else
    #define UIKIT_EXTERN            extern __attribute__((visibility ("default")))
    #endif
    

    是个宏定义,看不懂。


    看不懂可以百度,百度之后大概了解了:

    UIKIT_EXTERN简单来说,就是将函数修饰为兼容以往C编译方式的、具有extern属性(文件外可见性)、public修饰的方法或变量库外仍可见的属性。

    UIKIT_EXTERN只是在extern的基础上增加了兼容性(不知道这样描述可以不),说明我们是可以使用的。

    再看看NSNotificationName

    typedef NSString *NSNotificationName NS_EXTENSIBLE_STRING_ENUM;
    

    其实就是NSString *,我们当然可以用。

    NSNotificationName代替NSString *的好处是什么?直观。别人一看到NSNotificationName就知道你这是在定义一个通知。

    拿两个相同的通知对比一下就清楚了:

    UIKIT_EXTERN NSString * const ZOCCacheControllerDidClearCacheNotification;
    UIKIT_EXTERN NSNotificationName const ZOCCacheControllerDidClearCacheNotification;
    

    总结

    结合《禅与Objective-C编程艺术》+官方文档+苹果API可以得出:

    通知应该这样写:

    1. 命名符合通知专用格式
    [Name of associated class] + [Did | Will] + [UniquePartOfName] + Notification
    
    1. 在.m中定义,并且在.h中暴露。

    这是苹果API里的一个通知,是模板,以后就照着模板写吧:

    UIKIT_EXTERN NSNotificationName const UITextFieldTextDidChangeNotification;
    

    最后提供一个完整模板

    // Foo.h
    UIKIT_EXTERN NSNotificationName const ZOCFooDidBecomeBarNotification
    
     // Foo.m
    NSNotificationName const ZOCFooDidBecomeBarNotification = @"ZOCFooDidBecomeBarNotification";
    

    相关文章

      网友评论

      本文标题:iOS开发细节 | 通知怎么写?

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