Proguard混淆Android项目所遇问题及总结

原创 2015年11月19日 17:39:44

Proguard混淆Android项目所遇问题及总结

最近一个老项目需要添加混淆, 因此又对混淆有了更深入的了解. 在此过程中遇到了一下问题, 记录如下: 


1. 编译打包错误

a. 类重复, 错误信息如下

:xyz:compileReleaseNdk UP-TO-DATE
:xyz:compileReleaseSources
:xyz:proguardRelease
Note: there were 1276 duplicate class definitions.
      (http://proguard.sourceforge.net/manual/troubleshooting.html#duplicateclass)
Exception while processing task 
java.io.IOException: Can't write [/Users/stone/xyz/build/intermediates/classes-proguard/release/classes.jar] (Can't read [/Users/stone/xyz/abc/build/intermediates/exploded-aar/xyz/emojicon/unspecified/jars/libs/android-support-v4.jar(;;;;;;!META-INF/MANIFEST.MF)] (Duplicate zip entry [android-support-v4.jar:android/support/v4/hardware/display/DisplayManagerCompat.class]))
at proguard.OutputWriter.writeOutput(OutputWriter.java:187)
at proguard.OutputWriter.execute(OutputWriter.java:79)
at proguard.ProGuard.writeOutput(ProGuard.java:427)
at proguard.ProGuard.execute(ProGuard.java:175)
at proguard.gradle.ProGuardTask.proguard(ProGuardTask.java:1074)
at com.android.build.gradle.tasks.AndroidProGuardTask.doMinification(AndroidProGuardTask.java:139)
at com.android.build.gradle.tasks.AndroidProGuardTask$1.run(AndroidProGuardTask.java:115)
at com.android.builder.tasks.Job.runTask(Job.java:48)
at com.android.build.gradle.tasks.SimpleWorkQueue$EmptyThreadContext.runTask(SimpleWorkQueue.java:41)
at com.android.builder.tasks.WorkQueue.run(WorkQueue.java:227)
at java.lang.Thread.run(Thread.java:745)
Caused by: java.io.IOException: Can't read [/Users/stone/xyz/build/intermediates/exploded-aar/xyz/emojicon/unspecified/jars/libs/android-support-v4.jar(;;;;;;!META-INF/MANIFEST.MF)] (Duplicate zip entry [android-support-v4.jar:android/support/v4/hardware/display/DisplayManagerCompat.class])
at proguard.InputReader.readInput(InputReader.java:188)
at proguard.InputReader.readInput(InputReader.java:158)
at proguard.OutputWriter.writeOutput(OutputWriter.java:176)
... 10 more
Caused by: java.io.IOException: Duplicate zip entry [android-support-v4.jar:android/support/v4/hardware/display/DisplayManagerCompat.class]
at proguard.io.JarWriter.getOutputStream(JarWriter.java:138)
at proguard.io.FilteredDataEntryWriter.getOutputStream(FilteredDataEntryWriter.java:105)
at proguard.io.FilteredDataEntryWriter.getOutputStream(FilteredDataEntryWriter.java:105)
at proguard.io.FilteredDataEntryWriter.getOutputStream(FilteredDataEntryWriter.java:92)
at proguard.io.ClassRewriter.read(ClassRewriter.java:68)
at proguard.io.FilteredDataEntryReader.read(FilteredDataEntryReader.java:87)
at proguard.io.FilteredDataEntryReader.read(FilteredDataEntryReader.java:87)
at proguard.io.FilteredDataEntryReader.read(FilteredDataEntryReader.java:87)
at proguard.io.JarReader.read(JarReader.java:65)
at proguard.io.DirectoryPump.readFiles(DirectoryPump.java:65)
at proguard.io.DirectoryPump.pumpDataEntries(DirectoryPump.java:53)
at proguard.InputReader.readInput(InputReader.java:184)
... 12 more
:xyz:dexRelease FAILED


出现原因: 

主项目引用了几个library, 而这几个library都引用了同一个jar包, library引用jar包的方式是: 将jar包放在module的libs目录下并配置如下gradle脚本

 

dependencies {
    compile fileTree(dir: 'libs', include: '*.jar')
}

 

解决过程:

以前用eclipse开发android项目遇到过这样的编译问题, 错误信息如下: 

com.android.dex.DexException: Multiple dex files define Lcom/alibaba/fastjson/JSONStreamAware;
at com.android.dx.merge.DexMerger.readSortableTypes(DexMerger.java:594)
at com.android.dx.merge.DexMerger.getSortedTypes(DexMerger.java:552)
at com.android.dx.merge.DexMerger.mergeClassDefs(DexMerger.java:533)
at com.android.dx.merge.DexMerger.mergeDexes(DexMerger.java:170)
at com.android.dx.merge.DexMerger.merge(DexMerger.java:188)
at com.android.dx.command.dexer.Main.mergeLibraryDexBuffers(Main.java:439)
at com.android.dx.command.dexer.Main.runMonoDex(Main.java:287)
at com.android.dx.command.dexer.Main.run(Main.java:230)
at com.android.dx.command.dexer.Main.main(Main.java:199)
at com.android.dx.command.Main.main(Main.java:103)

导致这个编译错误的原因如上, 当是只要jar包改成同一个文件, 就会编译通过 (由于编译出的class文件的版本不一致, 导致无法合并), 

然而在anroid studio中, 这样做并无卵用. 

我是这样做的: 将不同的library共同引用的jar包放在一个目录下(project跟目录的libs木下), 并配置如下gradle脚本

dependencies {
    compile fileTree(dir: 'libs', include: '*.jar')
    compile fileTree(dir: '../libs', include: '*.jar')
}

最终解决方案: 

把module下面的libs目录下的jar包(其他项目也引用的那个jar)删除, 并配置改module的gradle脚本(注意: 各个library引用的库的版本必须一致------------红色标出部分一致)

dependencies {
    compile fileTree(dir: 'libs', include: '*.jar')

   compile 'com.alibaba:fastjson:1.1.43'
}

注: 这样依赖的是maven仓库中的jar或则aar.


2. 运行时错误

a. 空指针

主要是由 新版的butterknife生成的类的类名规则改变 引起的, 

butterknife 7.x之前的版本生成的类的名称形如: **$$ViewBinder

butterknife 7.x即之后的版本生成的类的名称形如: **$$ViewInjector

不混淆butterknife生成的类的规则如下:

-keep class **$$ViewBinder { *; } #butterknife 7.x即之后的版本生成的类
-keep class **$$ViewInjector { *; } #butterknife 7.x 版之前生成的类

我的规则文件是从另一个项目中拷贝过来的, 所以就出现了上述问题:


b. 反射时无法取到某个属性

主要是由于 "没有保持orm框架注解过的数据库实体类" 引起的, 

这个添加规则就可以解决了, 如下: (具体规则则需要根据你的具体情况而定)

#xutils定义的实体类
-dontwarn com.lidroid.xutils.**
-keep class com.lidroid.xutils.** { *; }

#此段只能保持使用xutils注解过的字段或方法不能被混淆, db数据库的所有字段都不能被混淆, 因此需要结合下面一段
-keepclasseswithmembernames class * {
    @com.lidroid.xutils.db.annotation.* <fields>;
    @com.lidroid.xutils.db.annotation.* <methods>;
}
#使用xutils注解过的数据库实体类
-keep class a.b.c.** { *; }
-keep class a.abc.cbd.User { *; }


c. 没有CREATOR

保持CREATOR字段不被混淆的规则<sdk_home>/tools/proguard/proguard-android.txt 文件中已经定义过了

如下: 

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

当是有些人在定义CREATOR时会少些一个final, 实际上CREATOR是一个常量, 如果你少些一个final那就只是一个类变量

因此在自定义的规则文件中添加了一条规则, 如下第二条: 

#不混淆CREATOR常量, 有人定义CREATOR时不写final关键字 -_-!!
-keep class * implements android.os.Parcelable {
    public static final android.os.Parcelable$Creator *;
    public static android.os.Parcelable$Creator *;
}



3. 总结: 

1. 混淆的原理

把源代码中的 "包名/类名/变量名/方法名" 改成无意义的字符串, 以达到保护源码的目的.


2. 容易出问题的地方

a. 反射 (包名类名, 方法名, 变量名)

如果你把类名混淆了, 下面的语句会出什么问题? !

if(obj.getClass().getName().equals("a.b.c.SomeClass")) {
    //do something
}

变量名和方法名同理!!

b. xml文件, 如下标注的部分如果混淆了, 会发生什么问题?!!



当然xml文件还有AndroidManifest.xml, 这个文件会用到四大组件, 因此有以下避免混淆的规则保持四大组件的类名(外加自定义的Application): 

-keep public class * extends android.app.Activity
-keep public class * extends android.app.Application
-keep public class * extends android.app.Service
-keep public class * extends android.content.BroadcastReceiver
-keep public class * extends android.content.ContentProvider


3. 一个总原则

所有硬编码的名称都不能混淆. 

所谓硬编码: 

a. xml中的东西, 如上.

b. 字符串中写死的类名, 变量名, 方法名, 包名........

c. 反射 ------ 这个其实在上一条中有所体现, 如放射去方法或变量时就是用了硬编码 ------- 字符串!

如下: 

Class a = getSomeClass(2);
a.getMethod("methodName", int.class);
a.getField("aField");

当然只是列举了所谓"硬编码"中的一部分, 更多的混淆问题和"硬编码" 有待读者发现


proguard及混淆规则===> 点击这里

错误之处再所难免, 希望你能帮我更正! 大笑














相关文章推荐

android代码混淆报错总结:java.io.IOException: Please correct the above

http://www.cnblogs.com/zfrr/archive/2012/06/01/2530744.html   android代码混淆报错总结:java.io.IOExceptio...

Android 代码混淆异常 transformClassesAndResourcesWithProguardForRelease FAILED

前几天混淆代码,出现了 这个异常  Warning:Exception while processing task java.io.IOException: Please correct th...

Android studio混淆打包的时候一直失败,提示Warning:Exception while processing task java.io.FileNotFoundException: D

至于具体的文件混淆规则这里不说了,网上一堆,接下来就是混淆中遇到的一个问题: Android studio混淆打包的时候一直失败,提示Warning:Exception whil...

The same input jar is specified twice 解决办法

The same input jar is specified twice 解决办法 Android studio打包中遇到了这样的问题 Warning:Exception while proces...
  • YLBF_DEV
  • YLBF_DEV
  • 2016年01月02日 16:44
  • 10171

java.io.IOException: The same input jar [E:\Jre\lib\rt.jar] is specified twice.

执行Maven Install打包的时候,出现以下错误信息:[INFO] proguard jar: E:\maven\mvnRespo\net\sf\proguard\proguard-base\5...

Android Studio 问题记(一)

1,Error Loading Project: Cannot load module walk 解决:close project , import project 2,unregistered ...
  • gogler
  • gogler
  • 2016年06月21日 11:57
  • 3534

proguard.ParseException: Expecting keyword 'class', 'interface', or 'enum' before '-libraryjars' in

执行Maven Install打包的时候,出现以下错误信息: [proguard] proguard.ParseException: Expecting keyword 'class', 'inter...

Android用Proguard混淆代码遇到的…

http://my.oschina.net/u/936286/blog/175062 第一种问题: Proguard returned with error code 1. See conso...

生成带混淆配置的aar库

aar ProGuard

android杂记-三方依赖(清单合并失败,aar)

一、与依赖清单合并失败(Studio导入有依赖关系的Eclipse项目时出现的问题) 1. 问题:Error:Execution failed for task ':smartfridgenation...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Proguard混淆Android项目所遇问题及总结
举报原因:
原因补充:

(最多只允许输入30个字)