天天看點

一種無源代碼檔案的Java程式修改方法

作者:老誠不bug

一、前言

公司有個老舊項目忽然報錯,追蹤代碼發現邏輯有問題,可又由于公司代碼管理不當,導緻源碼丢失,目前隻有可運作的jar包;如果要修複這個問題,隻能通過修改位元組碼檔案的方式,然後重新打包部署。

二、準備工作

①:需要反編譯的xxx.jar包;

②:反編譯工具:JD-JUI.exe;

③:代碼編輯工具(IDEA);

三、兩種解決方案:

方案一:

第一步,在IDEA中建立一個maven項目第二步,把xxx.jar導入到該項目中第三步,定位要修改的xxx.class檔案,在src–>main–>java裡面建立一個同路徑的package,并建立xxx.java,然後在xxx.class檔案的内容複制到目前xxx.java中。注意:目前檔案可能除了依賴第三方庫依賴,還依賴其它檔案,需要同時copy出來,複制的時候注意保持包名一緻。

一種無源代碼檔案的Java程式修改方法

第四步,找到xxx.jar包下的pom.xml檔案複制到目前項目的pom.xml檔案中,解決依賴第三方庫的問題。

一種無源代碼檔案的Java程式修改方法

第五步,修改新建立的java源碼,修改完成後右鍵重新編譯該檔案。

一種無源代碼檔案的Java程式修改方法

第六步,編譯完成以後,在target檔案下找到新生成的xxx.class檔案第七步,使用壓縮包工具打開原始xxx.jar包,找到xxx.class檔案,使用新生成的xxx.class檔案替換覆寫掉即可。

一種無源代碼檔案的Java程式修改方法

優點:如果修改檔案依賴少,操作簡單快捷缺點:如果修改檔案依賴比較多,除了考慮依賴的第三方包,也要粘貼複制其它檔案,這樣費時繁瑣,本來隻需要更改一個檔案,但是卻需要其他檔案支援,産生依賴爆炸的問題。

方案二:

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檔案;

一種無源代碼檔案的Java程式修改方法
一種無源代碼檔案的Java程式修改方法

案例二:修改位元組碼檔案檔案中的指定方法

未修改前源代碼。

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

  }
}           

将源檔案打成jar;

一種無源代碼檔案的Java程式修改方法

使用使用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();           

執行代碼,在檔案路徑下檢視位元組碼檔案;

一種無源代碼檔案的Java程式修改方法

可以看到方法中的輸出列印"你好,世界"變成了"hello javaAssist"。

四、結尾

以上就是丢失源碼的情況下,隻能通過修改位元組碼檔案的兩種方法。​

來源:https://www.51cto.com/article/719349.html

繼續閱讀