本文仅作学习交流目的,提倡软件正版,反对盗版!
工欲善其事必先利其器
- Java Decompiler 1.4.0,简称JD
- InteliJ IDEA 2017,简称IDEA
- dirtyJOE
- Oracle JDK 1.8
先掌握大局
- 将已签名包转化为未签名包;
- 定位关键代码位置;
- 修改class文件字节码;
- 替换class文件重新打包;
付诸行动
1.将已签名包转化为未签名包
如果你用到的Jar文件使用了签名,它会保证里面的每个class文件不能被修改,所以即使你成功修改了class文件中的字节码,得到的Jar也是无法运行的。这些经过签名的Jar包的META-INF文件夹中一般包含了*.SF
和相应的*.RSA
文件。这些文件记录Jar包中每个文件的签名信息,以保证代码不被篡改。
使用下面的方法可以重新生成一个未签名Jar包。参考自stackoverflow作者Houtman的回答
1 | arduino复制代码// 使用JDK编译代码 |
附上源代码,免得回答被删
1 | Java复制代码// JarUnsigner.java |
- 定位代码关键位置
举一个需要输入序列号才能试用的库文件的例子,但是为了保护Jar包作者权益,对包名进行打码了。通过Jar包的说明书,可以知道如何使用它,就知道是什么地方输入序列号啦,要不然是个正常人也没法用对吧。例如
1 | Java复制代码// 文档说这样子可以验证序列号 |
详细步骤
- 使用JD反编译Jar包,然后
Save All Sources
,具体步骤这里省略; - 从IDEA新建一个工程,把反编译的源代码放到源目录,Jar包可能是经过混淆过的,不过这个关系不大,我们字节码都能改,还怕看不懂混淆?继续往下走
- 右击
authentication
这个类文件,选择Find Usages
,快捷键一般是Ctrl+G
,看到有个叫做p.java的文件用到,做了一些判断操作Blabla,看屏幕截图1:
- 点开这个p.java类搜索到的地方,可以看到,它在比较一个返回结果,然后给出不同的错误提示,这个结果是由一个t.a(三个参数)的方法计算得到,并且看得出来当计算结果的a属性值为0时就是
Licence OK!
,看屏幕截图2: - 好,我们再来看t.java这个文件的a方法又有什么东东(按住
Ctrl
键,鼠标点击那个t.a),我们只要保证他得到的结果的a属性等于0就好了,可以看到只有一种情况下a属性才会等于0,其他时候都是等于-1的,到这一步IDEA的使命就完成了,看屏幕截图3:
结果
通过上面5步(取决于不同的包,不一定都是5步),我们知道了需要改动t
类里面的a
方法,让它里面操作属性a
时,值为-1
的都改为0
就能成功。
- 修改class文件字节码
接下来就要使用dirtyJOE软件来定位并修改字节码了,作者也试过Class Editor, Java Bytecode Editor都因年久失修,没改成功。通过查阅JVM文档我们知道给整数赋值有几种指令,这里就说两种:
iconst_<i>
i可以为m1,0,1,2,3,4,5
,分别设置的值为-1,0,1,2,3,4,5
,指令16进制字节码分别是02
设置-1,03
设置0,04
设置1,以此类推bipush <i>
i范围可以是0-255,指令16进制字节码是10
,比如代码里面有的21就是bipush 21
,16字节码表示为10 15
let’s go
- 使用dirtyJOE打开
t.class
文件,切换到 Methods 页上,如下图所示:
- 双击我们需要修改的方法(同名时通过比对方法签名来区分),进入编辑界面,图中的两个指令的组合就是将值
-1
赋值给t$a.a
属性,双击图中的iconst_ml
,字节码是02
,改为03
,然后回车即可,可以使用Ctrl+F
查找多处进行修改:
- 关闭编辑窗口,保存修改。
warning
要保证字节码的数量不增多,也不减少,因为类和类之间代码调用跟字节位置关系密切。不然会导致修改的class文件无法使用。原来一行是1个字节的,继续用1个字节,2个的就继续两个,3个的继续保持三个。
- 替换class文件重新打包
这是最简单的一步,将修改好的class文件,使用zip压缩工具替换掉即可。如果是你Windows用户不推荐使用WinRaR,可以将jar文件(第一步生成的未签名包)重命名为zip文件,然后选择用Windows资源管理器
打开,将修改的class文件复制粘贴进去。
本文转载自: 掘金