一種無源代碼文件的Java程序修改方法
一、前言
公司有個(gè)老舊項(xiàng)目忽然報(bào)錯(cuò),追蹤代碼發(fā)現(xiàn)邏輯有問題,可又由于公司代碼管理不當(dāng),導(dǎo)致源碼丟失,當(dāng)前只有可運(yùn)行的jar包;如果要修復(fù)這個(gè)問題,只能通過修改字節(jié)碼文件的方式,然后重新打包部署。
二、準(zhǔn)備工作
①:需要反編譯的xxx.jar包;
②:反編譯工具:JD-JUI.exe;
③:代碼編輯工具(IDEA);
三、兩種解決方案:
方案一:
第一步,在IDEA中新建一個(gè)maven項(xiàng)目第二步,把xxx.jar導(dǎo)入到該項(xiàng)目中第三步,定位要修改的xxx.class文件,在src–>main–>java里面創(chuàng)建一個(gè)同路徑的package,并新建xxx.java,然后在xxx.class文件的內(nèi)容復(fù)制到當(dāng)前xxx.java中。注意:當(dāng)前文件可能除了依賴第三方庫依賴,還依賴其它文件,需要同時(shí)copy出來,復(fù)制的時(shí)候注意保持包名一致。
第四步,找到xxx.jar包下的pom.xml文件復(fù)制到當(dāng)前項(xiàng)目的pom.xml文件中,解決依賴第三方庫的問題。
第五步,修改新創(chuàng)建的java源碼,修改完成后右鍵重新編譯該文件。
第六步,編譯完成以后,在target文件下找到新生成的xxx.class文件第七步,使用壓縮包工具打開原始xxx.jar包,找到xxx.class文件,使用新生成的xxx.class文件替換覆蓋掉即可。
優(yōu)點(diǎn):如果修改文件依賴少,操作簡單快捷缺點(diǎn):如果修改文件依賴比較多,除了考慮依賴的第三方包,也要粘貼復(fù)制其它文件,這樣費(fèi)時(shí)繁瑣,本來只需要更改一個(gè)文件,但是卻需要其他文件支持,產(chǎn)生依賴爆炸的問題。
方案二:
JavaAssist簡單介紹:JavaAssist又叫編譯時(shí)的類,是Jboss開源的分析、編輯和創(chuàng)建Java字節(jié)碼的類庫,它能夠直接用java編碼的形式,而不需要了解虛擬機(jī)指令,就能動(dòng)態(tài)改變類的結(jié)構(gòu),或者動(dòng)態(tài)生成類。
案例1:重新生成字節(jié)碼文件
public static void main(String[] args) throws Exception{
//CtClass對(duì)象容器
ClassPool classPool=ClassPool.getDefault();
//CtClass對(duì)象容器中創(chuàng)建一個(gè)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));
//添加構(gòu)造函數(shù)
CtConstructor ctConstructor=new CtConstructor(new CtClass[]{},jATestClazz);
//添加構(gòu)造函數(shù)方法體
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);
//生成一個(gè)Class對(duì)象
Class<?> clazz=jATestClazz.toClass();
Object object=clazz.newInstance();
//反射執(zhí)行方法
clazz.getMethod("sayHello",new Class[]{}).invoke(object,new Object[]{});
//將生成的class寫入文件中
FileOutputStream fileOutputStream=new FileOutputStream(new File("JATest.class"));
fileOutputStream.write(jATestClazz.toBytecode());
fileOutputStream.close();
}
運(yùn)行代碼,生成MyTest.class文件;
案例二:修改字節(jié)碼文件文件中的指定方法
未修改前源代碼。
public class TyunTest {
public static void main(String[] args) {
sayHello();
}
public static void sayHello(){
System.out.println("你好,世界");
}
}
將源文件打成jar;
使用使用JavaAssist讀取jar包修改字節(jié)碼文件;
ClassPool classPool=ClassPool.getDefault();
// 設(shè)置jar包路徑
classPool.insertClassPath("/Users/wyw_yong/Desktop/tyun/Tiyun.jar");
// 獲取修改的類
CtClass ctClazz = classPool.getCtClass("TyunTest");
// 獲取類中的方法
CtMethod sayHelloMethod = ctClazz.getDeclaredMethod("sayHello");
// 修改類中的方法內(nèi)容
sayHelloMethod.setBody("System.out.println(\"hello javaAssist\");");
Class newTestJarClass = ctClazz.toClass();
// 使用修改過的類創(chuàng)建對(duì)象
Object newTestJar = newTestJarClass.newInstance();
Method newPrintTestMethod = newTestJarClass.getDeclaredMethod("sayHello");
newPrintTestMethod.invoke(newTestJar);
// 解除代碼鎖定,恢復(fù)可編輯狀態(tài)
ctClazz.defrost();
// 寫出到外存中
ctClazz.writeFile();
執(zhí)行代碼,在文件路徑下查看字節(jié)碼文件;
?可以看到方法中的輸出打印"你好,世界"變成了"hello javaAssist"。
四、結(jié)尾
以上就是丟失源碼的情況下,只能通過修改字節(jié)碼文件的兩種方法。?