一种无源代码文件的Java程序修改方法

开发 前端
JavaAssist又叫编译时的类,是Jboss开源的分析、编辑和创建Java字节码的类库,它能够直接用java编码的形式,而不需要了解虚拟机指令,就能动态改变类的结构,或者动态生成类。

一、前言

公司有个老旧项目忽然报错,追踪代码发现逻辑有问题,可又由于公司代码管理不当,导致源码丢失,当前只有可运行的jar包;如果要修复这个问题,只能通过修改字节码文件的方式,然后重新打包部署。

二、准备工作

①:需要反编译的xxx.jar包;

②:反编译工具:JD-JUI.exe;

③:代码编辑工具(IDEA);

三、两种解决方案:

方案一:

第一步,在IDEA中新建一个maven项目第二步,把xxx.jar导入到该项目中第三步,定位要修改的xxx.class文件,在src–>main–>java里面创建一个同路径的package,并新建xxx.java,然后在xxx.class文件的内容复制到当前xxx.java中。注意:当前文件可能除了依赖第三方库依赖,还依赖其它文件,需要同时copy出来,复制的时候注意保持包名一致。

图片

第四步,找到xxx.jar包下的pom.xml文件复制到当前项目的pom.xml文件中,解决依赖第三方库的问题。

图片

第五步,修改新创建的java源码,修改完成后右键重新编译该文件。

图片

第六步,编译完成以后,在target文件下找到新生成的xxx.class文件第七步,使用压缩包工具打开原始xxx.jar包,找到xxx.class文件,使用新生成的xxx.class文件替换覆盖掉即可。

图片

优点:如果修改文件依赖少,操作简单快捷缺点:如果修改文件依赖比较多,除了考虑依赖的第三方包,也要粘贴复制其它文件,这样费时繁琐,本来只需要更改一个文件,但是却需要其他文件支持,产生依赖爆炸的问题。

方案二:

JavaAssist简单介绍:JavaAssist又叫编译时的类,是Jboss开源的分析、编辑和创建Java字节码的类库,它能够直接用java编码的形式,而不需要了解虚拟机指令,就能动态改变类的结构,或者动态生成类。

案例1:重新生成字节码文件

public static void main(String[] args) throws Exception{
//CtClass对象容器
ClassPool classPool=ClassPool.getDefault();
//CtClass对象容器中创建一个public的JATest类
CtClass jATestClazz=classPool.makeClass("com.tyun.javaassist.MyTest");

//***属性操作
//MyTest类中添加private int id
CtField ctIdField=new CtField(classPool.getCtClass("int"),"id",jATestClazz);
ctIdField.setModifiers(Modifier.PRIVATE);
jATestClazz.addField(ctIdField);

//MyTest类中添加private String username
CtField ctUserNameField=new CtField(classPool.getCtClass("java.lang.String"),"username",jATestClazz);
ctUserNameField.setModifiers(Modifier.PRIVATE);
jATestClazz.addField(ctUserNameField);

//添加getter,setter方法
jATestClazz.addMethod(CtNewMethod.getter("getId",ctIdField));
jATestClazz.addMethod(CtNewMethod.getter("setId",ctIdField));
jATestClazz.addMethod(CtNewMethod.getter("getUsername",ctUserNameField));
jATestClazz.addMethod(CtNewMethod.getter("setUsername",ctUserNameField));

//添加构造函数
CtConstructor ctConstructor=new CtConstructor(new CtClass[]{},jATestClazz);
//添加构造函数方法体
StringBuffer sb = new StringBuffer();
sb.append("{\n").append("this.id = 27;\n").append("this.username=\"卓耿\";\n}");
ctConstructor.setBody(sb.toString());
jATestClazz.addConstructor(ctConstructor);

// 添加自定义方法
CtMethod method = new CtMethod(CtClass.voidType, "sayHello", new CtClass[]{}, jATestClazz);
method.setModifiers(Modifier.PUBLIC);
StringBuffer printSb = new StringBuffer();
printSb.append("{\nSystem.out.println(\"begin!\");\n")
.append("System.out.println(id);\n")
.append("System.out.println(username);\n")
.append("System.out.println(\"end!\");\n")
.append("}");
method.setBody(printSb.toString());
jATestClazz.addMethod(method);

//生成一个Class对象
Class<?> clazz=jATestClazz.toClass();
Object object=clazz.newInstance();

//反射执行方法
clazz.getMethod("sayHello",new Class[]{}).invoke(object,new Object[]{});

//将生成的class写入文件中
FileOutputStream fileOutputStream=new FileOutputStream(new File("JATest.class"));
fileOutputStream.write(jATestClazz.toBytecode());
fileOutputStream.close();
}

运行代码,生成MyTest.class文件;

图片

图片

案例二:修改字节码文件文件中的指定方法

未修改前源代码。

public class TyunTest {
public static void main(String[] args) {
sayHello();
}
public static void sayHello(){
System.out.println("你好,世界");

}
}

将源文件打成jar;

图片

使用使用JavaAssist读取jar包修改字节码文件;

ClassPool classPool=ClassPool.getDefault();
// 设置jar包路径
classPool.insertClassPath("/Users/wyw_yong/Desktop/tyun/Tiyun.jar");
// 获取修改的类
CtClass ctClazz = classPool.getCtClass("TyunTest");

// 获取类中的方法
CtMethod sayHelloMethod = ctClazz.getDeclaredMethod("sayHello");
// 修改类中的方法内容
sayHelloMethod.setBody("System.out.println(\"hello javaAssist\");");

Class newTestJarClass = ctClazz.toClass();
// 使用修改过的类创建对象
Object newTestJar = newTestJarClass.newInstance();
Method newPrintTestMethod = newTestJarClass.getDeclaredMethod("sayHello");
newPrintTestMethod.invoke(newTestJar);
// 解除代码锁定,恢复可编辑状态
ctClazz.defrost();
// 写出到外存中
ctClazz.writeFile();

执行代码,在文件路径下查看字节码文件;

图片

​可以看到方法中的输出打印"你好,世界"变成了"hello javaAssist"。

四、结尾

以上就是丢失源码的情况下,只能通过修改字节码文件的两种方法。​

责任编辑:武晓燕 来源: 新钛云服
相关推荐

2019-01-13 15:16:35

2009-12-15 19:18:39

Ruby源代码

2017-11-20 22:28:43

程序员源代码编程

2010-07-21 16:23:09

运行telnet程序

2011-02-23 09:35:25

Eclipse远程调试

2018-01-31 09:23:57

恶意软件代码方法

2018-12-14 14:30:12

安全检测布式系测试

2011-07-04 17:53:48

快速测试

2010-03-26 13:34:47

CentOS安装

2021-06-11 00:11:23

GPS数据协议

2012-08-16 09:39:04

2019-11-13 08:52:19

Linux监控日志

2018-02-08 08:11:41

2022-01-26 11:03:22

代码C语言标志位

2023-11-27 09:35:59

开源许可证

2022-11-27 08:08:42

2020-12-23 10:10:23

Pythonweb代码

2022-07-07 10:33:27

Python姿势代码

2022-06-22 09:44:41

Python文件代码

2017-12-11 10:40:14

点赞
收藏

51CTO技术栈公众号