读书笔记(三)

本人有若干成套学习视频, 可试看! 可试看! 可试看, 重要的事情说三遍 包含Java, 数据结构与算法, iOS, 安卓, python, flutter等等, 如有需要, 联系微信tsaievan.

这本书在第三章"接口与API设计"中提到了一些知识点, 有必要拿出来说说:
(一)首先来说说description方法.

在自定义的一些对象中, 如果打印对象指针的话, 只会打印类名和指针地址, 并不会将一些细节打印出来, 这个时候,我们往往会去重写description方法. 但是我们往往忽略了debugDescription方法. 那么, 这两种方法有什么不同呢?

如果是重写description方法, 打印的时候并不会走debugDescription方法, 但是如果你打了断点, 那么在你po对象的时候, 就会把debugDescription方法里需要打印的内容打印在控制台上.

比如我新建一个YFPerson类, 重写其description方法和debugDescription方法:


#import "YFPerson.h"

@interface YFPerson ()
@property (nonatomic, copy) NSString *firstName;
@property (nonatomic, copy) NSString *lastName;
@end

@implementation YFPerson
- (instancetype)initWithFirstName:(NSString *)firstName lastName:(NSString *)lastName {
    if (self = [super init]) {
        _firstName = [firstName copy];
        _lastName = [lastName copy];
    }
    return self;
}

- (NSString *)description {
    return [NSString stringWithFormat:@"对象的类为: %@, 对象的地址为: %p", [self class], self];
}

- (NSString *)debugDescription {
    return [NSString stringWithFormat:@"对象的类为: %@, 对象的地址为: %p, 姓为: %@, 名为: %@", [self class], self, _lastName, _firstName];
}

@end

那么我直接打印person对象时, 其结果是:

<EffectiveObjective-C2.0编写高质量iOS与OS X代码的52个有效方法>读书笔记(三)_第1张图片
`description`方法

但是如果我在23行打上断点, 用po的方法去打印person对象, 那么就会打印更多信息, 这些信息是我在重写debugDescription方法的时候返回的:

<EffectiveObjective-C2.0编写高质量iOS与OS X代码的52个有效方法>读书笔记(三)_第2张图片
`debugDescription`方法
(二)然后说说"尽量使用不可变对象"

书中提到了一个例子:

比如现在有一个YFPerson类, 然后有一个朋友列表功能, 你需要添加朋友和删除朋友, 这时候你会想到用一个属性, 比如NSMutableSet, 但是书中建议将NSMutableSet写成一个私有的成员变量:

@implementation YFPerson {
    NSMutableSet *_friendList;
}

然后暴露两个接口:

- (void)addFriend:(YFPerson *)newFriend;

- (void)removeFriend:(YFPerson *)oldFriend;

这两个接口给外界修改朋友列表.

书中不建议的做法是直接暴露集合, 然后直接修改集合, 这样可能在YFPerson不知情的情况下直接修改了集合, 造成数据混乱

所以书中说: 尽量使用不可变对象.
  • 我的理解是尽量让对象暴露给外界的接口越少越好, 除非是不得已的时候, 才暴露接口对外, 如果对外暴露属性的话, 最好是只读属性, 特别是对那些从服务器拉下来的属性, 这些属性值都是服务器给的, 客户端修改不了, 所以都可以写成只读属性.

  • 还有一些集合类, 数组, 字典, 集合等, 尽量不要将可变的数组, 字典, 集合暴露给外界, 而是封装一套API供外界使用, 而不是让外界直接操纵可变数组, 字典, 集合.

以上的做法提高了对象的稳定性!

(三)为私有方法名加前缀

这个想必大家都知道并且注意到了, 在很多源码里, 大家也都是这么写的, 但是有一条很多源码也没有注意到:

★ 不要单用一条下划线做私有方法的前缀, 因为这种做法是预留给苹果公司用的.
(四)理解OC错误模型

主要介绍了NSError的使用, 书中不建议频繁使用抛出异常:

只有发生了可使整个应用程序崩溃的严重错误时, 才应使用异常

书中也提供了两种方式来处理 :

  • 指派"代理方法" (delegate method) 来处理错误
  • 把错误信息放在NSError对象里, 经由"输出参数"返回给调用者.
一种是拿到用户传入的NSError对象的指针地址, 然后根据指针地址, 修改NSError对象的值:

比如我添加好友, 如果lastName为空的话, 就报错, 就可以这么写:

- (void)addFriend:(YFPerson *)newFriend withError:(NSError *__autoreleasing *)error {
    if (error) {
        if ([newFriend.lastName isEqualToString:@""] || newFriend.lastName == nil) {
            *error = [NSError errorWithDomain:NSOSStatusErrorDomain code:101 userInfo:@{
                                                                                        NSLocalizedDescriptionKey : @"添加的好友没有姓"
                                                                                        }];
            return;
        }
    }
    [_friendList addObject:newFriend];
}

外界在调用时, 将error所在的地址作为参数传过去, 然后运行结束之后, 如果发生错误的话, error的值会被修改:

    NSError *error = nil;
    [person addFriend:[YFPerson new] withError:&error];
    if (error) {
        NSLog(@"添加好友失败: %@", error);
    }else {
        NSLog(@"添加好友成功");
    }

本例中, 我传入了一个空的YFPerson对象, 打印结果如下:

添加好友失败错误提示
另外一种是依靠代理来做错误处理:

比如我删除好友, 如果传入的参数为空的话, 我也做一个错误处理:
先写一个代理方法:

@class YFPerson;

@protocol YFPersonDelegate 

- (void)removeFriend:(YFPerson *)oldFriend withError:(NSError *)error;

@end

然后在删除好友时做一个判断:

- (void)removeFriend:(YFPerson *)oldFriend {
    if (!oldFriend) {
        NSError *error = [NSError errorWithDomain:NSOSStatusErrorDomain code:102 userInfo:@{
                                                                                            NSLocalizedDescriptionKey : @"删除好友不能为空"
                                                                                            }];
        if ([self.delegate respondsToSelector:@selector(removeFriend:withError:)]) {
            [self.delegate removeFriend:oldFriend withError:error];
        }
        return;
    }
    [_friendList removeObject:oldFriend];
}

如果好友为空, 则调用代理方法, 把错误对象传进去, 这样, 外界在传入参数为空时, 就会调用代理方法, 我在外界就可以获取到错误信息:

[person removeFriend:nil];
- (void)removeFriend:(YFPerson *)oldFriend withError:(NSError *)error {
    NSLog(@"删除好友失败 : %@", error);
}

控制台打印的错误信息为:

删除好友失败错误信息

以上就是利用NSError的两种基本方法!

(五)理解NSCopying协议

又遇到蛋疼的深拷贝浅拷贝的问题了, 在网上找到了一篇很好的文章, 觉得比自己总结得好, 套用文中的一句话:

概念性的东西,没有必要纠结于此。只要知道进行拷贝操作时,被拷贝的是指针还是内容即可
文章地址贴在下面了, 需要的童鞋可以看看:
iOS 集合的深复制与浅复制

PS. 本人有若干成套学习视频, 包含Java, 数据结构与算法, iOS, 安卓, python, flutter等等, 如有需要, 联系微信tsaievan.

你可能感兴趣的:(读书笔记(三))