MJRefresh源码知识点解析

一、公共类

MJRefresh源码知识点解析_第1张图片

这里主要来说说UIScrollView+MJRefresh
(1) 交换函数exchangeInstanceMethod1:(SEL)method1 method2:(SEL)method2

@implementation UITableView (MJRefresh)
+ (void)load
{
[self exchangeInstanceMethod1:@selector(reloadData)  method2:@selector(mj_reloadData)];
}
- (void)mj_reloadData
{
[self mj_reloadData];
[self executeReloadDataBlock];
}
@end

此方法是在load方法中调用。 load方法是在程序一加载就会最先调用的,之后走main函数,再然后才是我们熟悉的didFinishLaunchingWithOptionsload方法只会走一次,一般跟hook 机制相关的方法都会在load方法中执行。 hook(钩子),可以理解为把方法钩住,然后做些自己想做的事情。

比如 mj_reloadData里面的[self executeReloadDataBlock]。有些人可能有这样的疑问,在mj_reloadData中调用[self mj_reloadData]不会产生死循环吗? 当然不会的!别忘了此时mj_reloadData已经与reloadData交换了,所以[self mj_reloadData]实际上执行的是reloadData方法~ 事实证明把[self mj_reloadData]改为[self reloadData]才会出现死循环呢。

说到load方法,就不得不提一提initialize方法。两个方法全局都只执行一次。只不过initialize是在类即将初始化的时候执行的。

(2)分类中重写set和get方法

static const char MJRefreshHeaderKey = '\0';
- (void)setMj_header:(MJRefreshHeader *)mj_header
{
    if (mj_header != self.mj_header) {
        // 删除旧的,添加新的
       [self.mj_header removeFromSuperview];
       [self insertSubview:mj_header atIndex:0];
       // 存储新的
       [self willChangeValueForKey:@"mj_header"]; // KVO
       objc_setAssociatedObject(self, &MJRefreshHeaderKey,
       mj_header, OBJC_ASSOCIATION_ASSIGN);
      [self didChangeValueForKey:@"mj_header"]; // KVO
    }
}
- (MJRefreshHeader *)mj_header
{
    return objc_getAssociatedObject(self, &MJRefreshHeaderKey);
}

分类中是不能直接增加属性的,即使我们在.h文件中写了@property (strong, nonatomic) MJRefreshHeader *mj_header;,实际上也是不生成set和get方法的。分类中不能访问成员变量。

但是,我们可以通过关联的方法来重写set和get方法,也就可以给分类增加属性啦。有一点值得注意,就是这两个KVOwillChangeValueForKeydidChangeValueForKey。如果是这样的代码_name = name;,系统会自动给name增加KVO,但由于分类中不能访问成员变量,也只能手动添加咯

二、MJRefresh层级

MJRefresh源码知识点解析_第2张图片

里面的代码逻辑小码哥注释的很详细,相信认真看的童鞋都杠杠的! 我就从中总结几点我特别受益的地方。
(1)高内聚低耦合。 全篇看下来没有一行的废代码,一层一层的很清晰。
(2)让外界可以一行代码轻松调用,比如+ (instancetype)headerWithRefreshingBlock:(MJRefreshComponentRefreshingBlock)refreshingBlock。 从此以后我自己定义的控件也都使用这样的方法了
(3)命名很规范。
(4)一些方面要提前想到哪些种情况不允许,要提前return掉。防止外界进行一些不规范的操作导致代码崩溃或执行不必要的操作。
(5)通过数据的改变来改变UI。比如endRefreshing方法

 - (void)endRefreshing
{
    dispatch_async(dispatch_get_main_queue(), ^{
    self.state = MJRefreshStateIdle;
    });
}

并没有在endRefreshing方法中直接恢复inset和offset,而是将状态(数据)先改变,并在状态中做事情

- (void)setState:(MJRefreshState)state
{
    MJRefreshCheckState
    // 根据状态做事情
    if (state == MJRefreshStateIdle) {
        if (oldState != MJRefreshStateRefreshing) return;
    // 保存刷新时间
    [[NSUserDefaults standardUserDefaults] setObject:[NSDate date] forKey:self.lastUpdatedTimeKey];
    [[NSUserDefaults standardUserDefaults] synchronize];
    // 恢复inset和offset
    [UIView animateWithDuration:MJRefreshSlowAnimationDuration animations:^{
        self.scrollView.mj_insetT += self.insetTDelta;
        // 自动调整透明度
        if (self.isAutomaticallyChangeAlpha) self.alpha = 0.0;
    } completion:^(BOOL finished) {
        self.pullingPercent = 0.0;
        if (self.endRefreshingCompletionBlock) {
        self.endRefreshingCompletionBlock();
        }
    }];
    } else if (state == MJRefreshStateRefreshing) {
           dispatch_async(dispatch_get_main_queue(), ^{
           [UIView animateWithDuration:MJRefreshFastAnimationDuration animations:^{
           CGFloat top = self.scrollViewOriginalInset.top + self.mj_h;
          // 增加滚动区域top
         self.scrollView.mj_insetT = top;
          // 设置滚动位置
          [self.scrollView setContentOffset:CGPointMake(0, -top) animated:NO];
        } completion:^(BOOL finished) {
             [self executeRefreshingCallback];
       }];
   });
 }
}

你可能感兴趣的:(MJRefresh源码知识点解析)