MJRefresh中異步更改UI為Refreshing狀態(tài)導(dǎo)致內(nèi)部狀態(tài)和UI狀態(tài)不一致的問題

1.前言

項目使用MJRefresh作為下拉刷新控件山孔。在手動觸發(fā)下拉刷新時候遇到了一個bug钾军,看了一下MJRefresh的源碼,發(fā)現(xiàn)MJRefresh的實現(xiàn)有點瑕疵展鸡,總結(jié)在此姨丈。

2.問題描述

如果我們這樣使用MJRefresh畅卓,最后MJRefresh Header將會保持下拉刷新的狀態(tài),而不能恢復(fù)到Idle的狀態(tài)构挤。

    MJRefreshNormalHeader *header = [MJRefreshNormalHeader headerWithRefreshingBlock:^{
        @strongify(self);
        [[self.viewModel reloadData] subscribeNext:^(id x) {
            @strongify(self);
            [self endRefreshing];
        } error:^(NSError *error) {
            @strongify(self);
            [self endRefreshing];
        }];
    }];
    self.tableView.mj_header = header;
    ...
    [self.tableView.mj_header endRefreshing];
    [self.tableView.mj_header beginRefreshing];

以上代碼中調(diào)用beginRefreshing是為了觸發(fā)下拉刷新髓介。調(diào)用endRefreshing是不必要的,但是這條語句會導(dǎo)致MJRefresh表現(xiàn)不正確(即不能恢復(fù)到Idle狀態(tài))筋现,作為組件應(yīng)該更加健壯一些唐础,說明MJRefresh實現(xiàn)上有些瑕疵箱歧。下面具體分析一下。

3.問題原因

問題核心原因是:
在MJRefreshHeader類setState方法中“更改UI為refreshing狀態(tài)”的操作是異步的一膨。也就是說呀邢,設(shè)置Refreshing狀態(tài)時,設(shè)置內(nèi)部狀態(tài)和設(shè)置UI狀態(tài)被分離開了豹绪,如果在中間插入了設(shè)置內(nèi)部狀態(tài)(比如Idle)的操作可能會導(dǎo)致內(nèi)部狀態(tài)和UI狀態(tài)不一致的問題价淌。另外,MJRefreshendRefreshing方法中“設(shè)置狀態(tài)為Idle”操作是異步的瞒津。
出現(xiàn)問題的原因就是兩次異步蝉衣,由于執(zhí)行順序的原因,導(dǎo)致內(nèi)部狀態(tài)和UI狀態(tài)不一致巷蚪。

源碼如下:

- (void)beginRefreshing
{
    ...
    self.state = MJRefreshStateRefreshing;
    ...
}

---

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

- (void)setState:(MJRefreshState)state
{
    // 根據(jù)狀態(tài)做事情
    if (state == MJRefreshStateIdle) {
        ...
        // 恢復(fù)inset和offset
        [UIView animateWithDuration:MJRefreshSlowAnimationDuration animations:^{
            self.scrollView.mj_insetT += self.insetTDelta;
            
            // 自動調(diào)整透明度
            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;
                // 增加滾動區(qū)域top
                self.scrollView.mj_insetT = top;
                // 設(shè)置滾動位置
                [self.scrollView setContentOffset:CGPointMake(0, -top) animated:NO];
            } completion:^(BOOL finished) {
                [self executeRefreshingCallback];
            }];
         });
    }
}

按照我們在問題描述中的調(diào)用方式病毡,最后執(zhí)行順序如下:

  1. dispatch set state idle operation
  2. set state refreshing
  3. dispatch set ui refreshing operation
  4. set state idle
  5. set ui refreshing
    至此,內(nèi)部狀態(tài)為idle屁柏,UI狀態(tài)為refreshing啦膜。
    內(nèi)部狀態(tài)為Idle狀態(tài),之后的endRefreshing將不會生效(發(fā)現(xiàn)newState與oldState一致就直接返回了)淌喻,UI無法恢復(fù)為Idle狀態(tài)僧家。

4.問題解決

最好的解決辦法是把setState中“更改UI為refreshing狀態(tài)”的操作變成同步的。避免設(shè)置內(nèi)部狀態(tài)和設(shè)置UI狀態(tài)的分離裸删,因為兩者分離之后八拱,如果中間執(zhí)行了“設(shè)置狀態(tài)為Idle”,那么將導(dǎo)致最終內(nèi)部狀態(tài)為Idle烁落、UI狀態(tài)為Refreshing的問題乘粒,也就是標(biāo)題所說的內(nèi)部狀態(tài)和UI狀態(tài)不一致的問題。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末伤塌,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子轧铁,更是在濱河造成了極大的恐慌每聪,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件齿风,死亡現(xiàn)場離奇詭異药薯,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)救斑,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進(jìn)店門童本,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人脸候,你說我怎么就攤上這事穷娱“竽瑁” “怎么了?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵泵额,是天一觀的道長配深。 經(jīng)常有香客問我,道長嫁盲,這世上最難降的妖魔是什么篓叶? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮羞秤,結(jié)果婚禮上缸托,老公的妹妹穿的比我還像新娘。我一直安慰自己瘾蛋,他們只是感情好俐镐,可當(dāng)我...
    茶點故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著瘦黑,像睡著了一般京革。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上幸斥,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天匹摇,我揣著相機(jī)與錄音,去河邊找鬼甲葬。 笑死廊勃,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的经窖。 我是一名探鬼主播坡垫,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼画侣!你這毒婦竟也來了冰悠?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤配乱,失蹤者是張志新(化名)和其女友劉穎溉卓,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體搬泥,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡桑寨,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了忿檩。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片尉尾。...
    茶點故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖燥透,靈堂內(nèi)的尸體忽然破棺而出沙咏,到底是詐尸還是另有隱情辨图,我是刑警寧澤,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布芭碍,位于F島的核電站徒役,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏窖壕。R本人自食惡果不足惜忧勿,卻給世界環(huán)境...
    茶點故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望瞻讽。 院中可真熱鬧鸳吸,春花似錦、人聲如沸速勇。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽烦磁。三九已至养匈,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間都伪,已是汗流浹背呕乎。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留陨晶,地道東北人猬仁。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像先誉,于是被迫代替她去往敵國和親湿刽。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,916評論 2 344

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,527評論 25 707
  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫褐耳、插件诈闺、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,033評論 4 62
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)铃芦,斷路器买雾,智...
    卡卡羅2017閱讀 134,601評論 18 139
  • 1.雖然用戶界面做得風(fēng)格獨特、很簡潔杨帽,但能不能加個新手引導(dǎo)!`途注盈! 2.文章編輯后怎么發(fā)到專題啊~ 能不能保持一下傳...
    wang351311閱讀 544評論 3 1
  • 自從接觸簡書,通過它了解到一些好書叙赚、好的文章老客、有用的干貨等等僚饭,這些或受益匪淺或有所啟發(fā),但是最大的感受應(yīng)該是貧窮胧砰。...
    夏簡之閱讀 4,956評論 20 14