最易懂的回调机制说明书

这篇文章主要说明回调机制是如何运作的,也是由于前段时间在做需求时有用到这个,但是当时自己不太会,上网查了很多也没有找到合适自己看的文章或回答,在自己研究了一段时间后,结合个人实践,试图用一文的内容讲清楚回调到底是什么。

本文一共三个例子,逐渐深入,结合例子末尾处将对回调作出解释。


简单案例1

Button button = (Button) findViewById(R.id.button);
button.setOnClickListener(View.OnClickListener(){
    //do something
});

这是一个按钮,实现了View的OnClickListener接口,并重写接口里的方法OnClick,这是什么意思呢?意思是“按钮被点击时,我们会do something“,对吧?这其实就是一个最简单的回调。


简单案例2

println(1)

view.getAnimationOut().setAnimationListener(object : Animation.AnimationListener(){
    
    override fun AnimationIn(animation : Animation){
        println(2)
    }
    
    override fun AnimationEnd(animation : Animation){
        println(3)
    }
    
    override fun AnimationRepeat(animation : Animation){
        println(4)
    }
})

println(5)

这是一个用kotlin写的动画事件的监听,因为笔者正好在做需求时有用到,在回调内外都有println函数用于打印一个数,那么,这五个数的打印顺序将会是什么呢?

不了解回调机制的同学肯定会觉得是12345,为什么?因为代码是从上而下写的嘛,但是我们结合一下上面的简单案例1中的说明“按钮被点击时,我们会do Something”,放到这里来,结合动画,是不是就意味着:

”当AnimationIn,动画进来时,我们do Something“
”当AnimationEnd,动画结束时,我们do Something“
”当AnimationRepeat,动画重复时,我们do Something“

是这样对吧,那么所以现在你觉得执行顺序是什么呢?你是不是还可能觉得顺序是12345,因为动画是先开始,再结束,再重复呢?没有关系,我们先放下疑惑,接着看最后一个例子


简单案例3

class NewsItemPresenter(val view : NewestFragment) {
    private var newsItemModel = NewsItemModel()
    private var newsList : MutableList<News.DataBean> = ArrayList()


    fun refresh(){
        asyncRequest(newsItemModel.initRetrofit("http://v.juhe.cn/"))

    }

    private fun asyncRequest(completeRetrofit : HttpRequest) {
        completeRetrofit.call("guoji", 50, 30, 0, "2cb4dfb90827deb9158e0a1f4e4f1e45")
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe(object : Observer<News> {
                override fun onSubscribe(d: Disposable?) {
                    Log.d("request", "onSubscribe")

                }

                override fun onNext(news: News?) {
                    Log.d("request", "onNext")
                    var newsDataBean : List<News.DataBean>? = news?.result?.data

                    if (newsDataBean != null) {
                        for(item in newsDataBean){
                            newsList.add(item)
                        }
                    }

                }

                override fun onError(e: Throwable?) {
                    e?.printStackTrace()
                }

                override fun onComplete() {

                }
            })
    }

    fun getNewsList() : List<News.DataBean>{
         return newsList
    }
}

这里出自笔者现在正在搭的项目,基于okHttp3+Retrofit2+RxJava3+MVP模式搭建,这里简单解释一下,因为不是所有东西我们都需要关注。

开头创建了一个可变list集合newsList,refresh函数不需要关注,我们看asyncRequest函数是基于RxJava的事件流处理,在onNext方法里进行数据的处理,熟悉MVP模式的朋友肯定知道,根据类名也可以推断出来,这是MVP模式里的Presenter层,用于处理View层和Model层的代码耦合,很明显我这里的意图是onNext方法执行结束后,将所有数据add添加进newsList,然后通过最后一个方法getNewsList返回给View层进行数据展示。逻辑上似乎可以,但是真的可以这样吗?

实际上,在这里通过debug,结果是执行getNewsList方法时,newsList的size为0,也就是说,数据根本没有添加进来。


解释

综合我们上面的例子,其实回调机制简述为一句话,就是“某件事情做好了,我们再do Something”。结合一个生活中的例子,我们给朋友A打电话,想问他一个问题,但是A现在在忙自己的事情,于是就会有这样的情况出现:

我:“A,问问你,1+2等于几啊?”
A:“我现在在忙,等我有空了告诉你”
我:“好的,我先挂了”

--------------十分钟后--------------

A:”1+2=3“
我:”噢!好的谢谢你“

这里才是正常的逻辑吧?如果我们的朋友A有事情,那么我们就先挂电话去做自己的事,毕竟之后他会给我们打过来。但这个”之后“是多长时间呢?我们不知道,这个是A决定的,而不是我们。因此我们不可能一直等A,毕竟如果他正在做一件很耗时的事情的话,我们可能会一直等一直等而没有结果。所以回调机制的意义在于哪里呢?其实就在这里,即”当一件事情做好了,我再通知你,你不用一直等我“。

所以我们可以看到,从案例一开始,即使Button的OnClick函数被声明在很靠前的位置,我们可能在其之后做了很多其他UI渲染之类的操作,但是我们会等OnClick函数执行了我们再去渲染吗?不会,这个OnClick函数只是声明在那里,就像我们打电话一样,我们只是打个电话告诉我们的朋友A我有个问题想请教你,但是具体我们什么时候才能得到答案呢?对应着我们的OnClick函数什么时候才能被执行呢?我们不知道,毕竟,需要我们的朋友A回个电话,或者”Click“一下按钮我们才知道。

那么案例二的顺序就应该是1523,4不一定执行。因为即使输出234的代码在输出5的代码上面,我们仍旧不知道它什么时候会执行,我们需要一个Animation来唤醒它,所以1和5必定在234之前执行,4不一定执行是因为,万一这个动画就放一次呢?

案例三其实也是一样的,我们知道RxJava专门用于处理异步操作,里面的四个回调方法同样也是”被触发了才会执行“,我们并不知道具体时机,但总不能让主线程一直空等造成ANR吧?所以我们的主线程会先执行getNewsList,然后再去执行回调,毕竟不能让主线程被卡住。


总结

我自己也是经历了从不懂到懂的过程,这是我自认为可以最简单让大家理解什么是回调以及如何去使用回调的最简单说明,如果大家有什么意见或指正,欢迎评论告诉我。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,192评论 6 511
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,858评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,517评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,148评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,162评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,905评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,537评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,439评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,956评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,083评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,218评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,899评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,565评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,093评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,201评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,539评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,215评论 2 358

推荐阅读更多精彩内容