Effective Objective-C 2.0 初读小结

  • 1.对于OC中的对象声明例如NSObject *obj1 = [NSObject new];, obj1这个指针变量是分配在栈上的, 他指向的是这一个分配在堆上面的实例对象, 如果进行下面的赋值操作NSObject *obj2 = obj1;,那么并没有新生成一个实例对象, 只是在栈上分配了一个新的指针变量obj2, 而obj2和obj1指向的实例对象是同一个.
  • 2.关于文件头文件的引入问题, 一般情况下不建议在A.h文件中引入其他的B.h文件, 因为在别人引入A.h的时候, 同时也引入了B.h文件, 增加不必要的文件耦合和编译时间, 一般在.h文件中使用前向声明@class B, 而在.m文件中才真的引入头文件, 当然对于protocol不能使用前向声明, 如果将protocol放在了另一个.h文件中, 那么就必须要引入这个头文件了.
  • 3.尽量使用字面量语法来初始化字符串, 数组, 字典等, 因为字面量语法其实是一种语法糖, 使用它可以让代码可读性更高, 当然对于一些必须要使用到初始化方法的时候字面量语法就不好用了.例如:
  • NSString *str = @"string";
    NSArray *arr = @[obj1, obj2];
    arr[1]// 读取使用下标而尽量不使用对应的函数...
    [array setObject:(nonnull id) atIndexedSubscript:(NSUInteger)]
  • 4.少用#define来定义常量, 因为宏定义只是简单的代码替换, 并没有类型判断, 不便于我们阅读判断, 同时宏定义可以被覆盖, 当别人引入了我们的头文件的时候, 可能会覆盖我们里面定义的宏, 带来很麻烦的调试, 我们应该使用C语言风格的const,static, extern相结合来定义常量
    /// 使用static 和const 定义文件内部的常量 一般使用k开头命名
    static float const kAnimationTime = 2.0f;
    /// 使用const定义全局的常量, 在其他文件中可以通过 extern float const kAnimationTime引入使用, 一般不用k开头命名, 而使用class名字
    float const CustomAnimationTime = 2.0f;
  • 5.用好枚举, 使用枚举来表示选项, 状态码, 可以让代码更清晰, 这个在系统的API中也经常看到, 比如按钮的状态, autoresizing... , 例如如果你需要用一些状态码来表示网络请求的结果: 你可能会有两种方法
      1. 定义一个整形变量, 然后说明, 不同的整数代表不同的状态, 那么这样对于开发就很不方便, 必须得很清楚并且很正确的输入对应的整数才能表示相应的状态, 那么就很容易出错, 和不便于维护
      int statusCode;
      if (statusCode == 200) { }/// 请求成功
      else if () ....
      2. 使用枚举, 对不同的状态定义不同的名字, 这样就很清晰方便了, 当然定义的时候使用NS_ENUM比使用enum要`好`
    typedef NS_ENUM(NSInteger, ErrorCode) {
      ErrorCodeNotFind,
      ErrorCodeLostConnection,
      ErrorCodeUnknow
    };
    显然上面你应该选用枚举, 同时还有一种情况就是, 定义多选项, 这个你是会把他们都放进一个数组中么?? 当然不要这样做, 这个时候也应该使用枚举来定义, 不过会有一点的小技巧, Apple对这种进行了一个包装, 使用NS_OPTIONS而不是enum
    typedef NS_OPTIONS(NSInteger, ErrorOptions) {
      ErrorOptionsNone = 0,
      ErrorOptionsOne = 1 << 0, ///左移操作    --- 1 --- 0001
      ErrorOptionsTwo = 1 << 1,               --- 2 --- 0010
      ErrorOptionsThree = 1 << 2              --- 4 --- 0100
    };
    因为上面定义的枚举值都为2的整数次幂值, 所以后面就可以使用位操作符 与(&)和或(|)来进行选项的筛选
    ErrorOptions options = ErrorOptionsOne | ErrorOptionsTwo; //--- 0011
      if (options & ErrorOptionsOne) {// ErrorOptionsOne
          // 结束判断后面
      }
      else if (options & ErrorOptionsTwo) {// ErrorOptionsTwo
          // ...
      }
      else {
          // ...
      }
  • 6.需要遍历操作的时候, 尽量不要用C语言风格的for遍历, 而是采用OC的 for-in方式的快速枚举, 当然使用block的方式来遍历未必不是更好的一种方式, 尤其是遍历字典的时候.
  • 7.需要缓存的时候使用NSCache而不要使用NSArray或者NSDictionary, 因为使用NSCache来进行缓存当内存不足的时候系统会自动清理缓存, 并且会首先清理缓存时间较长的东西, 如果使用NSArray或者NSDictionary就没有这个福利了
  • 8.不要在load方法里面执行耗时的操作, 因为这个时候会阻塞当前的线程, 如果是主线程被阻塞, 那么...就不能接受用户的响应, 同时不要在load方法里面使用其他的类和调用函数, 因为这个时候程序是脆弱的, 有可能使用的class还没有被加载到系统中来, 当然使用Foundation里面的NSString...这些是没有问题的
  • 9.initialize这个方法在文档中写明了是在第一次使用这个类的时候才会调用一次(懒加载), 但是需要注意的是, 如果父类中实现了initialize这个方法, 而子类中没有实现这个方法, 当初始化子类的时候, 父类的这个initialize方法是会被调用多次的(消息转发机制), 比如
        ZJChildClass类里面没有重写initialize方法, 但是他的父类重写了, 所以在初始化ZJChildClass的时候, 父类的initialize会被调用两次, 即会打印两条
      @interface ZJBaseClass : NSObject
      @end
      @implementation ZJBaseClass
      + (void)initialize {
          NSLog(@"加载一次-----");
      }
      @end
      @interface ZJChildClass : ZJBaseClass
      @end
    所以一般都是这样来重写initialize方法的, 保证只会像我们期望的那样调用一次
      + (void)initialize {
          if (self == [ZJBaseClass class]) { /// 不能用 [self class]
              NSLog(@"加载一次-----");
         }
      }
  • 10.对只需要执行一次的代码使用dispatch_once, 这样可以保证线程安全, 并且只执行一次, 最常见的是用来实现单例
    + (instancetype)sharedInstance {
      static Object *sharedInstance = nil;
      static dispatch_once_t onceToken;
      dispatch_once(&onceToken, ^{
          sharedInstance = [self new];
      });
      return sharedInstance;
    }
  • 11.多用GCD少用NSObject的一些performSelector等方法, 因为使用performSelector这种方式可能会造成内存泄漏, 一般情况下使用GCD都可以完成, 比如dispatch_after来实现延时后执行
  • 12.使用NSTimer的时候要特别注意内存泄漏的问题, 因为NSTimer会持有目标对象, 很容易造成循环引用的问题, 也许你会想到在这个目标对象的dealloc里面让NSTimer失效(调用 invalidation 并且置为nil), 但是这根本就没有用, 因为循环引用的原因, 根本就不会调用dealloc方法, 所以在里面销毁是没有用的, 需要在对象被销毁之前手动销毁计时器
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

yusirxiaer

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值