ProGuard Introduction

半城伤御伤魂 提交于 2019-11-30 00:46:50

介绍

ProGuard是一个Java类文件压缩器、优化器、混淆器和预校验器。在压缩步骤会发现并移除无用的类、字段、方法和属性。 在优化步骤会分析和优化Method(方法)的字节码。在混淆步骤会使用短且没有意义的名字来重命名剩余的类、字段和方法。 这些前面的步骤会使代码更小、更高效、更难反向工程。最后的预校验步骤会给这些类添加Java Micro版本需要的预校验信息或为Java 6减少启动时间的预校验信息。

这些步骤每一个都是可选的。例如,ProGuard也可以在一个程序中仅仅用来列出无效代码; 或用于在Java 6中预校验类文件提高效率。








ProGuard 通常读取input jars(或 wars, ears, zips,或 目录)。然后压缩、优化、混淆和预校验它们。 压缩步骤之后,可以执行多个优化过程。ProGuard把这些处理过的结果生成一个或多个output jars (或 wars, ears, zips, 或 目录)。 输入可能包含资源文件,他们的名字和内容可以可选地被更新成反射混淆之后的类的名字。

ProGuard 要求input jars中的library jars (or wars, ears, zips, or directories) 被指明。这些在本质上是你编译代码要用到的库。ProGuard用它们来重构类的依赖,并在必要的时候做适当的处理。 这些library jars始终未被改变,你应该一直把它们放在你最终程序的类路径中。

入口点

为了决定哪些代码需要被维持原样和哪些代码可以被移除或混淆,你必须在你的代码中指定一个或多个 入口点。 这些入口点通常是含有 main 方法的类、applets、midlets等等。
  • 压缩步骤, ProGuard 从这些入口点启动并递归决定哪些类和类方法是有用的。而剩余的所有的其它类和类方法会被废弃。
  • 优化步骤, ProGuard 进一步优化代码。不是入口点的类和方法可以被设置为private、static或final,无用的参数可能被移除,并且一些方法可能会被内联。
  • 混淆步骤, ProGuard 重命名不是入口点的类和方法的名字。整个过程中保持这些入口点以确保他们还能通过他们的原始名字被访问到。
  • 预校验步骤 是唯一不需要知道入口点的步骤

使用 章节 描述了必要的-keep选项示例 章节 则提供了大量的示例.

反射

反射和反省同样存在所有自动代码处理所带来的特殊问题。对ProGuard而言, 在你代码中被创建或被动态执行的类和类成员也必须要当作入口点指定。例如:Class.forName()构造在运行时可能会引用任何类。 通常难以预测哪些类必须被保留(保留它们的原始名字),例如类名称可能会从配置文件中读取。因此你必须在你的ProGuard配置中指定它们,使用简单的-keep选项。

不过,ProGuard已经为你指出并处理以下方案:

  • Class.forName("SomeClass")
  • SomeClass.class
  • SomeClass.class.getField("someField")
  • SomeClass.class.getDeclaredField("someField")
  • SomeClass.class.getMethod("someMethod", new Class[] {})
  • SomeClass.class.getMethod("someMethod", new Class[] { A.class })
  • SomeClass.class.getMethod("someMethod", new Class[] { A.class, B.class })
  • SomeClass.class.getDeclaredMethod("someMethod", new Class[] {})
  • SomeClass.class.getDeclaredMethod("someMethod", new Class[] { A.class })
  • SomeClass.class.getDeclaredMethod("someMethod", new Class[] { A.class, B.class })
  • AtomicIntegerFieldUpdater.newUpdater(SomeClass.class, "someField")
  • AtomicLongFieldUpdater.newUpdater(SomeClass.class, "someField")
  • AtomicReferenceFieldUpdater.newUpdater(SomeClass.class, SomeType.class, "someField")
类和类成员名称当然可以不同,但是构造函数应该每个字符都相同,为了ProGuard能够认出它们。在压缩阶段被引用的类和类成员会被 保留,而混淆阶段字符串参数会被适当地更新。

此外,如果有必要保持某些类或类成员,ProGuard会提供一些建议。例如:ProGuard会提示类似"(SomeClass)Class.forName(variable).newInstance()"。 这可能表示类或接口SomeClass的实现可能需要被保留。你随后可以相应地修改你的配置。

为了产生合适的结果,你至少应该对你正在处理的代码有所了解。代码混淆进行了大量的反射,可能会产生失败和错误,尤其是没有内部代码的必要信息。


Copyright © 2002-2011 Eric Lafortune 翻译: 大坤.
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!