首页 > 解决方案 > 奇怪的 NPE:实例方法内的原始字段 null

问题描述

我有一个带有以下堆栈跟踪的奇怪 NPE:

Fatal Exception: java.lang.NullPointerException: Attempt to read from field 'int com.foo.gvpclient.x.c' on a null object reference
       at com.foo.gvpclient.GVPConfiguration.getDeviceTypeAsString(GVPConfiguration.java:507)
       at com.foo.gvpclient.request.uniapi.authentication.LoginAnonymousRequest.<init>(LoginAnonymousRequest.java:39)
       at com.foo.bar.analytics.managers.AnalyticsManager.sendEventsBatchAsAnonymous(AnalyticsManager.java:489)
       at com.foo.bar.analytics.managers.AnalyticsManager.addEvent(AnalyticsManager.java:324)
       at com.foo.bar.analytics.managers.AnalyticsManager.onBrowseEvent(AnalyticsManager.java:266)
       at com.foo.bar.analytics.managers.AnalyticsManager.onBrowseEvent(AnalyticsManager.java:231)
       at com.foo.bar.analytics.managers.AnalyticsManager.onBrowseEvent(AnalyticsManager.java:711)
       at com.foo.bar.login.activities.LoginActivity.register(LoginActivity.java:469)
       at com.foo.bar.login.activities.LoginActivity.access$register(LoginActivity.java:51)
       at com.foo.bar.login.activities.LoginActivity$onCreate$$inlined$apply$lambda$1.onClick(LoginActivity.java:123)
       at android.view.View.performClick(View.java:7339)
       at android.widget.TextView.performClick(TextView.java:14221)
       at android.view.View.performClickInternal(View.java:7305)
       at android.view.View.access$3200(View.java:846)
       at android.view.View$PerformClick.run(View.java:27787)
       at android.os.Handler.handleCallback(Handler.java:873)
       at android.os.Handler.dispatchMessage(Handler.java:99)
       at android.os.Looper.loop(Looper.java:214)
       at android.app.ActivityThread.main(ActivityThread.java:7076)
       at java.lang.reflect.Method.invoke(Method.java)
       at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:493)
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:965)

这是getDeviceTypeAsString()方法:

public String getDeviceTypeAsString() {
    return String.valueOf(this.mDeviceType);
}

wheremDeviceType被声明为

private int mDeviceType;

如果我错了,请纠正我,但这里没有什么可以为空的。

为了完整起见,从这里调用该方法:

public LoginAnonymousRequest(GVPConfiguration gvpConfig) {
    super(gvpConfig);
    // More unrelated stuff
    this.mQueryStringParams.put("deviceType", this.mGvpConfig.getDeviceTypeAsString());
}

同样,如果我错了,请纠正我,但如果gvpConfig在上面的构造函数中为 null,那么 NPE 将被抛出,this.mGvpConfig.getDeviceTypeAsString()而不是在方法本身内。

那么是什么导致了这个 NPE 呢?可能与代码混淆有关吗?

标签: javaandroid

解决方案


这不是混淆,而是由 proguard 或 r8 等相同工具完成的优化。

基本上,简单的访问器在调用者中被优化为内联,实际上GVPConfiguration调用站点的对象为空。


推荐阅读