我的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
请注意,您从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 ...
此问题在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进程错误地删除了被定义为final,也意味着要在构造函数中初始化:null check
,为什么?检查AppCompatImageButton的源代码后,我发现mImageHelper
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都相同。