Proguard缩小导致空指针异常

问题描述 投票:4回答:2

我的Android应用中有一个视频活动。在禁用proguard的调试版本中,所有功能都像一个超级按钮。但是在发布版本中,当启用proguard并开始视频活动时,我得到了NPE。

我的gradle文件的一部分:

buildTypes {
    debug {
        applicationIdSuffix '.dev'
        versionNameSuffix '-dev'
        minifyEnabled false
        shrinkResources false
    }
    release {
        minifyEnabled true
        shrinkResources true
        debuggable false
        proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        signingConfig signingConfigs.release
    }
}

例外:

java.lang.RuntimeException: Unable to start activity ComponentInfo{ua.com.tv24.news/ua.com.tv24.news.ui.activities.MainActivity}: java.lang.NullPointerException: Attempt to invoke virtual method 'void android.support.v7.app.ActionBar.setTitle(java.lang.CharSequence)' on a null object reference
        at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2298)
        at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2360)
        at android.app.ActivityThread.access$800(ActivityThread.java:144)
        at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1278)
        at android.os.Handler.dispatchMessage(Handler.java:102)
        at android.os.Looper.loop(Looper.java:135)
        at android.app.ActivityThread.main(ActivityThread.java:5221)
        at java.lang.reflect.Method.invoke(Native Method)
        at java.lang.reflect.Method.invoke(Method.java:372)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:899)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:694)
 Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'void android.support.v7.app.ActionBar.setTitle(java.lang.CharSequence)' on a null object reference
        at ua.com.tv24.news.ui.fragments.a.b.a(Unknown Source)
        at android.support.v4.app.Fragment.i(Unknown Source)
        at android.support.v4.app.u.a(Unknown Source)
        at android.support.v4.app.u.a(Unknown Source)
        at android.support.v4.app.u.a(Unknown Source)
        at android.support.v4.app.u.j(Unknown Source)
        at android.support.v4.app.p.onCreate(Unknown Source)
        at android.support.v7.app.ActionBarActivity.onCreate(Unknown Source)
        at ua.com.tv24.news.ui.activities.a.onCreate(Unknown Source)
        at ua.com.tv24.news.ui.activities.b.onCreate(Unknown Source)
        at ua.com.tv24.news.ui.activities.MainActivity.onCreate(Unknown Source)
        at android.app.Activity.performCreate(Activity.java:5933)
        at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1105)
        at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2251)
            at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2360)
            at android.app.ActivityThread.access$800(ActivityThread.java:144)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1278)
            at android.os.Handler.dispatchMessage(Handler.java:102)
            at android.os.Looper.loop(Looper.java:135)
            at android.app.ActivityThread.main(ActivityThread.java:5221)
            at java.lang.reflect.Method.invoke(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:372)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:899)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:694)

和我的活动:

 package ua.com.tv24.news.ui.activities;

import android.content.res.Resources;
import android.graphics.drawable.Drawable;
import android.os.Bundle;
import android.os.Parcel;
import android.os.Parcelable;
import android.support.v7.app.ActionBarActivity;
import android.view.View;
import android.widget.FrameLayout;

import com.google.android.libraries.mediaframework.exoplayerextensions.Video;
import com.google.android.libraries.mediaframework.layeredvideo.PlaybackControlLayer;
import com.google.googlemediaframework.adplayer.ImaPlayer;

import de.greenrobot.event.EventBus;
import ua.com.tv24.news.R;
import ua.com.tv24.news.events.ui.VideoPreparedEvent;

public class DashVideoActivity extends ActionBarActivity implements PlaybackControlLayer.FullscreenCallback {
    private ImaPlayer imaPlayer;
    private FrameLayout videoPlayerContainer;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        EventBus.getDefault().register(this);

        setContentView(R.layout.activity_dash_video);
        videoPlayerContainer = (FrameLayout) findViewById(R.id.video_frame);

        Bundle bundle = getIntent().getExtras();
        String videoUrl = bundle.getString(VideoActivity.VIDEO_URL_EXTRA);
        String appName = getString(R.string.app_name);

        //TODO get url and name from extras
        VideoItem video = new VideoItem(appName, new Video(videoUrl, Video.VideoType.DASH, ""),
                null);

        createImaPlayer(video);
    }

    @Override
    protected void onResume() {
        super.onResume();
        if (imaPlayer != null) {
            imaPlayer.play();
        }
    }

    @Override
    protected void onPause() {
        if (imaPlayer != null) {
            imaPlayer.pause();
        }
        super.onPause();
    }

    @Override
    protected void onDestroy() {
        if (imaPlayer != null) {
            imaPlayer.release();
        }
        EventBus.getDefault().unregister(this);
        super.onDestroy();
    }

    public void createImaPlayer(VideoItem videoListItem) {
        if (imaPlayer != null) {
            imaPlayer.release();
        }

        // If there was previously a video player in the container, remove it.
        videoPlayerContainer.removeAllViews();

        String adTagUrl = videoListItem.adUrl;
        String videoTitle = videoListItem.title;

        imaPlayer = new ImaPlayer(this,
                videoPlayerContainer,
                videoListItem.video,
                videoTitle,
                adTagUrl);
        imaPlayer.setFullscreenCallback(this);

        Resources res = getResources();

        Drawable logo = res.getDrawable(R.drawable.launcher);
        imaPlayer.setLogoImage(logo);

        imaPlayer.play();
    }

    @Override
    public void onGoToFullscreen() {
    }

    @Override
    public void onReturnFromFullscreen() {
    }

    public static class VideoItem implements Parcelable {

        /**
         * The title of the video.
         */
        public final String title;

        /**
         * The actual content video (contains its URL, media type - either DASH or mp4,
         * and an optional media type).
         */
        public final Video video;

        /**
         * The URL of the VAST document which represents the ad.
         */
        public final String adUrl;

        /**
         * @param title The title of the video.
         * @param video The actual content video (contains its URL, media type - either DASH or mp4,
         *              and an optional media type).
         * @param adUrl The URL of the VAST document which represents the ad.
         */
        public VideoItem(String title, Video video, String adUrl) {
            this.title = title;
            this.video = video;
            this.adUrl = adUrl;
        }

        @Override
        public int describeContents() {
            return 0;
        }

        @Override
        public void writeToParcel(Parcel dest, int flags) {

        }
    }

    public void onEventMainThread(VideoPreparedEvent event) {
        View progress = findViewById(R.id.videoLoaderBar);
        if (progress != null) {
            progress.setVisibility(View.INVISIBLE);
        }
    }
}

还有我的监护人规则:

-dontwarn com.google.android.gms.**

# Picasso 
-dontwarn com.squareup.okhttp.**

-keep class * extends java.util.ListResourceBundle {
    protected Object[][] getContents();
}

-keep public class com.google.android.gms.common.internal.safeparcel.SafeParcelable {
    public static final *** NULL;
}

-keepnames @com.google.android.gms.common.annotation.KeepName class *
-keepclassmembernames class * {
    @com.google.android.gms.common.annotation.KeepName *;
}

-keepnames class * implements android.os.Parcelable {
    public static final ** CREATOR;
}

#okhttp
-keepnames class com.levelup.http.okhttp.** { *; }
-keepnames interface com.levelup.http.okhttp.** { *; }
-keepnames class com.squareup.okhttp.** { *; }
-keepnames interface com.squareup.okhttp.** { *; }

-dontwarn com.squareup.okhttp.internal.http.*
-dontwarn org.codehaus.mojo.animal_sniffer.IgnoreJRERequirement

#retrofit
-keepattributes Signature
-keepattributes *Annotation*

-keep class com.google.gson.** { *; }
-keep class com.google.inject.** { *; }
-keep class org.apache.http.* { *; }
-keep class org.apache.james.mime4j.* { *; }
-keep class javax.inject.** { *; }
-dontwarn rx.*
-keep class com.example.testobfuscation.** { *; }
-keepattributes Signature
-keep class sun.misc.Unsafe { *; }

-keep class retrofit.** { *; }
-keep interface retrofit.** { *; }
-keepclasseswithmembers class * {
    @retrofit.http.* <methods>;
}
-keepclasseswithmembers class * {
    @ua.* <methods>;
}

-dontwarn retrofit.appengine.**

-keep class ua.com.tv24.news.models.** { *; }
-keepattributes InnerClasses

#xml parser
-keep interface org.simpleframework.xml.** { *; }
-keep class org.simpleframework.xml.** { *; }

#eventBus
-keepclassmembers class ** {
    public void onEvent(**);
}
-keepclassmembers class ** {
    public void onEventMainThread(**);
}

#support v7
-keep class android.support.v7.** { *; }

#unusage classes
-dontwarn retrofit.appengine.UrlFetchClient
-dontwarn retrofit.RxSupport$1
-dontwarn com.google.ads.a.a
-dontwarn com.google.ads.a.c
-dontwarn com.google.ads.a.d
-dontwarn com.google.ads.a.f
-dontwarn com.google.android.exoplayer.hls.HlsMediaPlaylistParser
-dontwarn okio.DeflaterSink
-dontwarn org.simpleframework.xml.stream.StreamProvider
-dontwarn okio.Okio
-dontwarn org.simpleframework.xml.stream.StreamReader
-dontwarn org.simpleframework.xml.stream.StreamReader$Entry
-dontwarn org.simpleframework.xml.stream.StreamReader$Start
-dontwarn org.simpleframework.xml.stream.StreamReader$Text
-dontwarn retrofit.RestMethodInfo$RxSupport
-dontwarn retrofit.RxSupport
-dontwarn retrofit.RxSupport$2
android proguard
2个回答
7
投票

请注意,您从ProGuard处理中排除了类和方法:

# Hide warnings about references to newer platforms in the library
-dontwarn android.support.v7.**
# don't process support library
-keep class android.support.v7.** { *; }
-keep interface android.support.v7.** { *; }

如果涉及Reflection,通常会发生此错误,因为在这种情况下Proguard无法解决依赖关系。

我只是看到您错过了ProGuard文件中的接口(根据例外):

Caused by: java.lang.NullPointerException: Attempt to invoke virtual method ...

0
投票

此问题在proguard的优化过程中发生,但我的问题有点不同,崩溃日志为:

java.lang.RuntimeException: Unable to start activity ComponentInfo{com.cmcm.gamemoney_sdk_style2/com.cmcm.gamemoney.Main3Activity}: android.view.InflateException: Binary XML file line #37: Binary XML file line #37: Error inflating class android.support.v7.widget.Toolbar
       at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2957)
       at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3032)
       at android.app.ActivityThread.-wrap11(Unknown Source:0)
       at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1696)
       at android.os.Handler.dispatchMessage(Handler.java:105)
       at android.os.Looper.loop(Looper.java:164)
       at android.app.ActivityThread.main(ActivityThread.java:6942)
       at java.lang.reflect.Method.invoke(Native Method)
       at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:327)
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1374)
Caused by: android.view.InflateException: Binary XML file line #37: Binary XML file line #37: Error inflating class android.support.v7.widget.Toolbar
Caused by: android.view.InflateException: Binary XML file line #37: Error inflating class android.support.v7.widget.Toolbar
Caused by: java.lang.reflect.InvocationTargetException
       at java.lang.reflect.Constructor.newInstance0(Native Method)
       at java.lang.reflect.Constructor.newInstance(Constructor.java:334)
       at android.view.LayoutInflater.createView(LayoutInflater.java:647)
       at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:790)
       at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:730)
       at android.view.LayoutInflater.rInflate(LayoutInflater.java:863)
       at android.view.LayoutInflater.rInflateChildren(LayoutInflater.java:824)
       at android.view.LayoutInflater.rInflate(LayoutInflater.java:866)
       at android.view.LayoutInflater.rInflateChildren(LayoutInflater.java:824)
       at android.view.LayoutInflater.inflate(LayoutInflater.java:515)
       at android.view.LayoutInflater.inflate(LayoutInflater.java:423)
       at android.view.LayoutInflater.inflate(LayoutInflater.java:374)
       at android.support.v7.app.l.s(AppCompatDelegateImpl.java:607)
       at android.support.v7.app.l.r(AppCompatDelegateImpl.java:518)
       at android.support.v7.app.l.b(AppCompatDelegateImpl.java:466)
       at android.support.v7.app.i.setContentView(AppCompatActivity.java:140)
       at com.cmcm.gamemoney_sdk.Main3Activity.onCreate(Main3Activity.java:41)
       at android.app.Activity.performCreate(Activity.java:7183)
       at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1221)
       at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2910)
       at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3032)
       at android.app.ActivityThread.-wrap11(Unknown Source:0)
       at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1696)
       at android.os.Handler.dispatchMessage(Handler.java:105)
       at android.os.Looper.loop(Looper.java:164)
       at android.app.ActivityThread.main(ActivityThread.java:6942)
       at java.lang.reflect.Method.invoke(Native Method)
       at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:327)
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1374)
Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'void android.support.v7.widget.l.a()' on a null object reference
       at android.support.v7.widget.s.setBackgroundDrawable(AppCompatImageButton.java:124)
       at android.view.View.setBackground(View.java:21620)
       at android.view.View.<init>(View.java:5553)
       at android.widget.ImageView.<init>(ImageView.java:176)
       at android.widget.ImageButton.<init>(ImageButton.java:96)
       at android.widget.ImageButton.<init>(ImageButton.java:92)
       at android.support.v7.widget.s.<init>(AppCompatImageButton.java:73)
       at android.support.v7.widget.Toolbar.y(Toolbar.java:1362)
       at android.support.v7.widget.Toolbar.c(Toolbar.java:918)
       at android.support.v7.widget.Toolbar.<init>(Toolbar.java:322)
       at android.support.v7.widget.Toolbar.<init>(Toolbar.java:229)
       ... 29 more

<< [AppCompatImageButton#setBackgroundDrawable()的关键点,所以我反编译了apk的代码并找到了这样的混淆方法代码:public final void setBackgroundDrawable(Drawable var1) { super.setBackgroundDrawable(var1); this.a.a(); }

与proguad的代码之前的比较:

public void setImageDrawable(@Nullable Drawable drawable) { super.setImageDrawable(drawable); if (this.mImageHelper != null) { this.mImageHelper.applySupportImageTint(); } }

这意味着proguard进程错误地删除了null check,为什么?检查AppCompatImageButton的源代码后,我发现

mImageHelper

被定义为final,也意味着要在构造函数中初始化:private final AppCompatImageHelper mImageHelper; public AppCompatImageButton(Context context, AttributeSet attrs, int defStyleAttr) { super(TintContextWrapper.wrap(context), attrs, defStyleAttr); this.mBackgroundTintHelper = new AppCompatBackgroundHelper(this); this.mBackgroundTintHelper.loadFromAttributes(attrs, defStyleAttr); this.mImageHelper = new AppCompatImageHelper(this); this.mImageHelper.loadFromAttributes(attrs, defStyleAttr); }
问题是在初始化之前,super是构造函数的第一行,并且最终在初始化之前调用

setImageDrawable()

,这导致崩溃。
正如我所说,我不鼓励使用-keep class android.support.v7.** { *; }解决此问题,这是可以解决的问题。

第一个解决方案在proguard配置中声明,以标记不优化:

-dontoptimize

但是将禁用所有优化,不利用优化过程。

所以我能找到的最佳解决方案是:

-optimizations !field/propagation/value

它标记不会优化固定值的调用。 

顺便说一下,在跟踪真正问题所在的位置时,我从Windows 10切换到macOS,发现问题仅发生在Windows 10中,我检查差异仅发现jvm版本不等于,macOS为“ 1.8.0_73”但Windows 10为“ 1.8.0_221”,否则,例如proguard版本均为6.0.3,并且proguard-base-6.0.3.jar的md5都相同。

© www.soinside.com 2019 - 2024. All rights reserved.