启用Proguard时解组时找不到类

时间:2015-06-17 08:53:51

标签: android proguard parcelable

我一直在处理似乎是Proguard配置问题。我的应用程序在调试时工作正常,但发布的apk在启动第二个活动时给了我一个例外。当尝试从Intent中检索Parcelable extras时似乎会发生这种情况。

这是stacktrace:

E/Parcel﹕ Class not found when unmarshalling: com.mycompany.model.Task
    java.lang.ClassNotFoundException: com.mycompany.model.Task
            at java.lang.Class.classForName(Native Method)
            at java.lang.Class.forName(Class.java:251)
            at java.lang.Class.forName(Class.java:216)
            at android.os.Parcel.readParcelableCreator(Parcel.java:2133)
            at android.os.Parcel.readParcelable(Parcel.java:2097)
            at android.os.Parcel.readValue(Parcel.java:2013)
            at android.os.Parcel.readArrayMapInternal(Parcel.java:2314)
            at android.os.Bundle.unparcel(Bundle.java:249)
            at android.os.Bundle.getString(Bundle.java:1118)
            at android.content.Intent.getStringExtra(Intent.java:5128)
            at com.android.server.am.ActivityStackSupervisor.startActivityLocked(ActivityStackSupervisor.java:1447)
            at com.android.server.am.ActivityStackSupervisor.startActivityMayWait(ActivityStackSupervisor.java:1043)
            at com.android.server.am.ActivityManagerService.startActivityAsUser(ActivityManagerService.java:4054)
            at com.android.server.am.ActivityManagerService.startActivity(ActivityManagerService.java:3948)
            at android.app.ActivityManagerNative.onTransact(ActivityManagerNative.java:159)
            at com.android.server.am.ActivityManagerService.onTransact(ActivityManagerService.java:2629)
            at android.os.Binder.execTransact(Binder.java:404)
            at dalvik.system.NativeStart.run(Native Method)
Caused by: java.lang.NoClassDefFoundError: com/mycompany/model/Task
            at java.lang.Class.classForName(Native Method)
            at java.lang.Class.forName(Class.java:251)
            at java.lang.Class.forName(Class.java:216)
            at android.os.Parcel.readParcelableCreator(Parcel.java:2133)
            at android.os.Parcel.readParcelable(Parcel.java:2097)
            at android.os.Parcel.readValue(Parcel.java:2013)
            at android.os.Parcel.readArrayMapInternal(Parcel.java:2314)
            at android.os.Bundle.unparcel(Bundle.java:249)
            at android.os.Bundle.getString(Bundle.java:1118)
            at android.content.Intent.getStringExtra(Intent.java:5128)
            at com.android.server.am.ActivityStackSupervisor.startActivityLocked(ActivityStackSupervisor.java:1447)
            at com.android.server.am.ActivityStackSupervisor.startActivityMayWait(ActivityStackSupervisor.java:1043)
            at com.android.server.am.ActivityManagerService.startActivityAsUser(ActivityManagerService.java:4054)
            at com.android.server.am.ActivityManagerService.startActivity(ActivityManagerService.java:3948)
            at android.app.ActivityManagerNative.onTransact(ActivityManagerNative.java:159)
            at com.android.server.am.ActivityManagerService.onTransact(ActivityManagerService.java:2629)
            at android.os.Binder.execTransact(Binder.java:404)
            at dalvik.system.NativeStart.run(Native Method)
Caused by: java.lang.ClassNotFoundException: Didn't find class "com.mycompany.model.Task" on path: DexPathList[[directory "."],nativeLibraryDirectories=[/vendor/lib, /system/lib]]
            at dalvik.system.BaseDexClassLoader.findClass(BaseDexClassLoader.java:67)
            at java.lang.ClassLoader.loadClass(ClassLoader.java:497)
            at java.lang.ClassLoader.loadClass(ClassLoader.java:457)
            at java.lang.Class.classForName(Native Method)
            at java.lang.Class.forName(Class.java:251)
            at java.lang.Class.forName(Class.java:216)
            at android.os.Parcel.readParcelableCreator(Parcel.java:2133)
            at android.os.Parcel.readParcelable(Parcel.java:2097)
            at android.os.Parcel.readValue(Parcel.java:2013)
            at android.os.Parcel.readArrayMapInternal(Parcel.java:2314)
            at android.os.Bundle.unparcel(Bundle.java:249)
            at android.os.Bundle.getString(Bundle.java:1118)
            at android.content.Intent.getStringExtra(Intent.java:5128)
            at com.android.server.am.ActivityStackSupervisor.startActivityLocked(ActivityStackSupervisor.java:1447)
            at com.android.server.am.ActivityStackSupervisor.startActivityMayWait(ActivityStackSupervisor.java:1043)
            at com.android.server.am.ActivityManagerService.startActivityAsUser(ActivityManagerService.java:4054)
            at com.android.server.am.ActivityManagerService.startActivity(ActivityManagerService.java:3948)
            at android.app.ActivityManagerNative.onTransact(ActivityManagerNative.java:159)
            at com.android.server.am.ActivityManagerService.onTransact(ActivityManagerService.java:2629)
            at android.os.Binder.execTransact(Binder.java:404)
            at dalvik.system.NativeStart.run(Native Method)

有Parcelable Task模型:

public class Task implements Parcelable {

    public enum Mode {MODE1, MODE2}

    public enum Type {TYPE1, TYPE2, TYPE3}

    private final String id;
    private final Mode mode;
    private final Type type;
    private ArrayList<Character> charactersList;

    public Task(String id, ArrayList<Character> characters, Mode mode, Type type) {
        this.id = id;
        this.charactersList = characters;
        this.mode = mode;
        this.type = type;
    }

    public Mode getMode() {
        return mode;
    }

    public Type getType() {
        return type;
    }

    public String getId() {
        return id;
    }

    public ArrayList<Character> getCharactersList() {
        return charactersList;
    }

    private Task(Parcel in) {
        id = in.readString();

        Parcelable[] parcelablecharactersArray = in.readParcelableArray(Character.class.getClassLoader());
        charactersList = new ArrayList<>();
        for (Parcelable p : parcelablecharactersArray) {
            charactersList.add((Character) p);
        }

        mode = Mode.valueOf(in.readString());
        type = Type.valueOf(in.readString());

    }

    public static final Parcelable.Creator<Task> CREATOR = new Parcelable.Creator<Task>() {

        public Task createFromParcel(Parcel in) {
            return new Task(in);
        }

        public Task[] newArray(int size) {
            return new Task[size];
        }
    };

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

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeString(id);
        dest.writeParcelableArray(charactersList.toArray(new Parcelable[charactersList.size()]), 0);
        dest.writeString(mode.name());
        dest.writeString(type.name());

    }

}

并且有活动启动并接收意图:

public class TaskActivity {

    private Task mTask;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        getWindow().setBackgroundDrawable(null);

        Bundle extras = getIntent().getExtras();
        mTask = extras.getParcelable(Constants.TASK_KEY);
    }

    private void launchNextTask(Task nextTask) {
        if (nextTask != null) {
            Intent intent = new Intent(TaskActivity.this, TaskActivity.class);
            intent.putExtra(Constants.TASK_KEY, nextTask);
            startActivity(intent);
        }
    }
}

和proguard规则:

-keep class android.support.v4.app.** { *; }
-keep interface android.support.v4.app.** { *; }

-keep class * implements android.os.Parcelable {
  public static final android.os.Parcelable$Creator *;
}

-keep public class com.mycompany.model.Task { *; }
-keep public class com.mycompany.model.Character { *; }
-keep public class com.mycompany.Constants { *; }

你知道问题可能是什么吗?我的第一个猜测是,当尝试在Parcelable构造函数中创建Character ArrayList时,类加载器存在问题(顺便说一下,Character也实现了Parcelable)。第二个猜测是枚举。

2 个答案:

答案 0 :(得分:4)

我认为问题来自Parcelable Task扩展。

# could be refined, but if it works you got the idea
-keepnames class * implements android.os.Parcelable { *; }
-keepclassmembers class * implements android.os.Parcelable { *; }
-keep public class com.mycompany.model.Task { *; }

答案 1 :(得分:0)

找到答案。它是Proguard配置。但它不是关于Parcelable或Task,而是关于另一个模块中的其他类,它们被混淆了,不应该是。

添加

-keep public class com.othermodule.OtherClass { *; }

做了这个伎俩。

因此,堆栈跟踪有点误导,或者至少不够明确。

无论如何,谢谢@shkschneider的帮助。