反编译工具

jd-gui

JD-GUI,一个独立的图形实用程序,用于显示来自 CLASS 文件的 Java 源代码。

下载地址:http://java-decompiler.github.io/

image.png
如下图,Save All Sources 点击此按钮,保存所有反编译代码
image.png
缺点:经常某些代码反编译的有问题,某些新特性也不能很好的处理


CFR

cfr是一个非常强大的java反编译器,支持java5,java8的新特性,如枚举,lambda表达式等。

下载地址:https://www.benf.org/other/cfr/

反编译命令
java -jar cfr-0.151.jar java7_jar.jar --outputdir D:/Java/tmp

java -jar cfr-0.150.jar Client.class

cfr还有大量的参数可以让我们灵活的设置
java -jar cfr-0.150.jar --help

  1. --aexagg (boolean)
  2. --aexagg2 (boolean)
  3. --aggressivedoextension (boolean)
  4. --aggressivesizethreshold (int >= 0) default: 15000
  5. --allowcorrecting (boolean) default: true
  6. --analyseas (One of [DETECT, JAR, WAR, CLASS])
  7. --arrayiter (boolean) default: true if class file from version 49.0 (Java 5) or greater
  8. --caseinsensitivefs (boolean) default: true
  9. --clobber (boolean)
  10. --collectioniter (boolean) default: true if class file from version 49.0 (Java 5) or greater
  11. --commentmonitors (boolean) default: false
  12. --comments (boolean) default: true
  13. --decodeenumswitch (boolean) default: true if class file from version 49.0 (Java 5) or greater
  14. 去除switch对枚举支持的语法糖 --decodeenumswitch false
  15. --decodefinally (boolean) default: true
  16. --decodelambdas (boolean) default: true if class file from version 52.0 (Java 8) or greater
  17. 去除lambda表达式的语法糖 --decodelambdas false
  18. --decodestringswitch (boolean) default: true if class file from version 51.0 (Java 7) or greater
  19. --dumpclasspath (boolean) default: false
  20. --eclipse (boolean) default: true
  21. --elidescala (boolean) default: false
  22. --extraclasspath (string)
  23. --forbidanonymousclasses (boolean) default: false
  24. --forbidmethodscopedclasses (boolean) default: false
  25. --forceclassfilever (string, specifying either java version as 'j6', 'j1.0', or classfile as '56', '56.65535')
  26. --forcecondpropagate (boolean)
  27. --forceexceptionprune (boolean)
  28. --forcereturningifs (boolean)
  29. --forcetopsort (boolean)
  30. --forcetopsortaggress (boolean)
  31. --forcetopsortnopull (boolean)
  32. --forloopaggcapture (boolean)
  33. --hidebridgemethods (boolean) default: true
  34. --hidelangimports (boolean) default: true
  35. --hidelongstrings (boolean) default: false
  36. --hideutf (boolean) default: true
  37. --ignoreexceptions (boolean) default: false
  38. --ignoreexceptionsalways (boolean) default: false
  39. --importfilter (string)
  40. --innerclasses (boolean) default: true
  41. --instanceofpattern (boolean) default: true if class file from version 58.0 (Java 14) or greater, or experimental in 58.0 (Java 14)
  42. --j14classobj (boolean) default: false if class file from version 49.0 (Java 5) or greater
  43. --jarfilter (string)
  44. --labelledblocks (boolean) default: true
  45. --lenient (boolean) default: false
  46. --liftconstructorinit (boolean) default: true
  47. --lomem (boolean) default: false
  48. --methodname (string)
  49. --obfuscationpath (string)
  50. --outputdir (string)
  51. 结果输出目录
  52. --outputpath (string)
  53. --override (boolean) default: true if class file from version 50.0 (Java 6) or greater
  54. --previewfeatures (boolean) default: true
  55. --pullcodecase (boolean) default: false
  56. --recordtypes (boolean) default: true if class file from version 58.0 (Java 14) or greater, or experimental in 58.0 (Java 14)
  57. --recover (boolean) default: true
  58. --recovertypeclash (boolean)
  59. --recovertypehints (boolean)
  60. --relinkconststring (boolean) default: true
  61. --removebadgenerics (boolean) default: true
  62. 去除泛型的语法糖 --removebadgenerics false
  63. --removeboilerplate (boolean) default: true
  64. --removedeadconditionals (boolean)
  65. --removedeadmethods (boolean) default: true
  66. --removeinnerclasssynthetics (boolean) default: true
  67. --rename (boolean) default: false
  68. --renamedupmembers (boolean) default: Value of option 'rename'
  69. --renameenumidents (boolean) default: Value of option 'rename'
  70. --renameillegalidents (boolean) default: Value of option 'rename'
  71. --renamesmallmembers (int >= 0) default: 0
  72. --showinferrable (boolean) default: false if class file from version 51.0 (Java 7) or greater
  73. --showversion (boolean) default: true
  74. --silent (boolean) default: false
  75. --skipbatchinnerclasses (boolean) default: true
  76. --staticinitreturn (boolean) default: true
  77. --stringbuffer (boolean) default: false if class file from version 49.0 (Java 5) or greater
  78. --stringbuilder (boolean) default: true if class file from version 49.0 (Java 5) or greater
  79. --stringconcat (boolean) default: true if class file from version 53.0 (Java 9) or greater
  80. --sugarasserts (boolean) default: true
  81. --sugarboxing (boolean) default: true
  82. 去除自动装箱和拆箱的语法糖 --sugarboxing false
  83. --sugarenums (boolean) default: true if class file from version 49.0 (Java 5) or greater
  84. 去除枚举的语法糖 --sugarenums false
  85. --switchexpression (boolean) default: true if class file from version 57.0 (Java 13) or greater, or experimental in 56.0 (Java 12)
  86. --tidymonitors (boolean) default: true
  87. --tryresources (boolean) default: true if class file from version 51.0 (Java 7) or greater
  88. --usenametable (boolean) default: true
  89. --usesignatures (boolean) default: true
  90. --help (string)

修改Jar包工具

jclasslib bytecode viewer

jclasslib bytecode viewer 是一个可以可视化已编译Java类文件和所包含的字节码的工具。 另外,它还提供一个库,可以让开发人员读写Java类文件和字节码。

下载地址:https://github.com/ingokegel/jclasslib/releases

Jar包反编译整理 - 图5

编辑

所有常量池条目都可以编辑。引用的常量池条目可以直接从属性和常量池条目的“编辑”按钮显示的上下文菜单中进行编辑。
Jar包反编译整理 - 图6
Jar包反编译整理 - 图7

Jar包签名

某些Jar包中包含代码签名,修改Jar包后需要重新签名,需要用到JDK自带工具keytool
若不重新签名,会出现JarVerifier错误。

META-INF目录下的签名文件:
image.png

原理

签名:Jar包反编译整理 - 图9

认证:

Jar包反编译整理 - 图10

签名步骤

用java的keytool生成密钥对,用java的jarsigner做签名

生成密钥对

keytool -genkey -keystore ijvmkeys.keystore -keyalg RSA -validity 10000 -alias friend.keystore
Jar包反编译整理 - 图11

查看密钥文件

keytool -list -v -keystore ijvmkeys.keystore
Jar包反编译整理 - 图12

签名Jar包

jarsigner -verbose -keystore ijvmkeys.keystore friend.jar friend.keystore
Jar包反编译整理 - 图13
Jar包反编译整理 - 图14
FRIEND_K.SF文件:

  1. Signature-Version: 1.0
  2. SHA1-Digest-Manifest-Main-Attributes: QHukAYw2MtCop4vlrhjJDDro1fQ=
  3. Created-By: 1.6.0_12 (Sun Microsystems Inc.)
  4. SHA1-Digest-Manifest: YePdyFc1+FVdY1PIcj6WVuTJAFE=
  5. Name: com/yfq/test/friend/Friend$1.class
  6. SHA1-Digest: mj79V3+YKsRAzxGHpyFGhOdY4dU=
  7. Name: com/yfq/test/friend/Friend.class
  8. SHA1-Digest: tqPfF2lz4Ol8eJ3tQ2IBvvtduj0=

它包含了签名的版本,签名者,还有被签名的类名,以及这个类的hash摘要,第四行是整个本文件的摘要,用于jar包的校验

Jar验证

jarsigner -verify friend.jar和jarsigner -verify stranger.jar
Jar包反编译整理 - 图15

参考: https://www.cnblogs.com/Gandy/p/7290069.html https://blog.csdn.net/yangxt/article/details/1796965 https://www.cnblogs.com/jixp/articles/10731726.html

JCE(加密)供应商的相关Jar包

是JCE供应商(provider)的相关Jar,有可能报JCE cannot authenticate the provider BC
这是由于JCE供应商相关Jar包,必须签名两次,第一次必须是Oracle,第二次是CA。

必须由JDK环境认证证书的签名,即必须由Oracle 进行JCE提供商认证签名,需要申请。

官方说明:https://www.oracle.com/java/technologies/javase/getcodesigningcertificate.html

参考: https://stackoverflow.com/questions/13721579/jce-cannot-authenticate-the-provider-bc-in-java-swing-application https://stackoverflow.com/questions/1756801/how-to-sign-a-custom-jce-security-provider https://stackoverflow.com/questions/20998124/jar-file-manifest-does-not-contain-permission-attribute https://www.oracle.com/java/technologies/javase/getcodesigningcertificate.html https://docs.oracle.com/javase/6/docs/technotes/guides/security/crypto/HowToImplAProvider.html#Step6

绕过Oracle签名

使用OpenJDK,没有Oracle JDK导致的相关签名问题,经验证,修改的Jar包可以直接使用运行。

安装OpenJDK

下载地址:https://adoptopenjdk.net/?variant=openjdk8&jvmVariant=hotspot

需要本地设置环境变量JAVA_HOME
即可在该环境下使用

IDEA配置:
系统配置:Maven Importing -> JDK for importer 修改为openjdk
工程配置:project -> project SDK 改为openjdk
modules -> Dependencies -> Module SDK 改为openjdk

集成到java工程

JDK使用openjdk

工程下目录/lib 增加demo.jar

pom依赖配置:

  1. <dependency>
  2. <groupId>demo</groupId>
  3. <artifactId>demo</artifactId>
  4. <version>1.0.0</version>
  5. <scope>system</scope>
  6. <systemPath>${project.basedir}/lib/demo.jar</systemPath>
  7. </dependency>

构建插件:

  1. <plugin>
  2. <groupId>org.springframework.boot</groupId>
  3. <artifactId>spring-boot-maven-plugin</artifactId>
  4. <configuration>
  5. <includeSystemScope>true</includeSystemScope>
  6. </configuration>
  7. </plugin>