Lambda表达式的一个NullPointerException

NullPointerException(简称NPE)是开发中很常见的一个异常,也比较容易处理, 但有时却也莫名其妙就NPE了, 并且错误根源也不大么容易寻找, 下面看一个关于NPE的例子, 请暂时忘记文章标题, 按常规思路来想问题.

异常描述

     Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'java.lang.Class java.lang.Object.getClass()' on a null object reference
        at com.*.HomeMinePresenterImpl.loadData(HomeMinePresenterImpl.java:26)
        at com.*.HomeMineFragment.onResume(HomeMineFragment.java:158)
        at android.support.v4.app.Fragment.performResume(Fragment.java:2401)
        at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1465)
        at android.support.v4.app.FragmentManagerImpl.moveFragmentToExpectedState(FragmentManager.java:1740)
        at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1809)

该app主页是一个MainActivity,嵌套了4个Fragment, 回到桌面启动另一个app, 再切换回来发现app崩了,出现这个异常信息. 代码如下:

    public void loadData() {
        ServiceFactory.get(HomeMineService.class)
                .getHomeMineData()
                .compose(TransformerUtil::io2Main)
                .subscribe(data -> {
                  if(success){
                    mView.onSuccess(data);
                  }else{
                    mView.onFaild(message);
                  }
                }, mView::onError);
    }

定位到的NPE是loadData()中的26行,日志中点击是定位到了compose那一行, 而关键能解决问题的异常信息是:

Attempt to invoke virtual method 'java.lang.Class java.lang.Object.getClass()' on a null object reference

这里并没有调用getClass(). 开启了调试想断进来看是哪里空了, 但是Fragment崩溃后也断不进来了. 然而基本可以确定是Fragment被回收后造成的问题.

继续研究问题, getClass通常是框架进行反射时使用的, 接着对代码进行try-catch, 仍然是上面的异常信息, 但这时候可以打断点进来了不是? 真的是mView空了! NPE是在按步执行到compose后进入了catch块, 但正常来讲也应该报类似:

Attempt to invoke virtual method 'xxx.onError(...)' on a null object reference

这样的异常信息, 怎么会跑到getClass去NPE?

反编译异常代码

将apk拖到jadx-gui中, 定位到反编译后的代码(没有任何混淆,很容易找到):

public class HomeMinePresenterImpl extends BasePresenter<HomeMineView> implements HomeMinePresenter {
    @SuppressLint({"CheckResult"})
    public void loadData() {
        Observable compose = ((HomeMineService) ServiceFactory.get(HomeMineService.class)).getHomeMineData().compose(HomeMinePresenterImpl$$Lambda$0.$instance);
        Consumer consumer = HomeMinePresenterImpl$$Lambda$1.$instance;
        HomeMineView homeMineView = (HomeMineView) this.mView;
        homeMineView.getClass();
        compose.subscribe(consumer, HomeMinePresenterImpl$$Lambda$2.get$Lambda(homeMineView));
    }
}

看到了一行homeMineView.getClass();设想此时mView是空的, 如果按反编译后的代码来执行, 那就会在compose执行完后得到NPE, 完全符合断点看到的结果. 然后看了其他类似代码的反编译, 有些调用getClass, 有些则没有, 但是有规律可循的(由于框架的缘故, 应用在subscribe中基本都会用mView来交互):

调用Observable.subscribe(Observer<? super T> observer),反编译后的class代码中都不会调用mView.getClass(), 只要使用到mView相关的方法引用(如 mView::onError)都会调到

举个栗子来验证下

public class NpeTest {
    void onError(Runnable r){
        Runnable runnable = r::run;
        runnable.run();
    }
}

编译,反汇编

javac NpeTest.java 
javap -c -p NpeTest.class 

得到

Compiled from "NpeTest.java"
public class NpeTest {
  public NpeTest();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return

  void onError(java.lang.Runnable);
    Code:
       0: aload_1
       1: dup
       2: invokevirtual #2                  // Method java/lang/Object.getClass:()Ljava/lang/Class;
       5: pop
       6: invokedynamic #3,  0              // InvokeDynamic #0:run:(Ljava/lang/Runnable;)Ljava/lang/Runnable;
      11: astore_2
      12: aload_2
      13: invokeinterface #4,  1            // InterfaceMethod java/lang/Runnable.run:()V
      18: return
}

在onError的invokevirtual这里就调用了Object.getClass.

解决问题之余心有不甘, 进行了一番折腾, 没有深入了解java对lambda的编译原理, 见笑于大方之家, 仅记录耳.

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

推荐阅读更多精彩内容

  • 我从去年开始使用 RxJava ,到现在一年多了。今年加入了 Flipboard 后,看到 Flipboard 的...
    Jason_andy阅读 5,468评论 7 62
  • // com.adobe.flash.listen settings.gradle 定义项目包含那些模块app.i...
    zeromemcpy阅读 1,611评论 0 1
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,056评论 25 707
  • 今天上午上完课后回寝室无聊刷了一下微博,被热门吓了一跳,当时我整个人都震惊了,“霍金去世”这四个字在我脑海中飞了好...
    落落青时兮阅读 484评论 2 0
  • 一个多月前,她裸辞了。 很多人都不赞同裸辞,一没保障,二不知道下一个工作要找多久。然而她找了一个多月。在一个不合适...
    萌子莫阅读 366评论 0 3