使用javassist修改.class文件,並重新打包
Javassist是一款字節碼編輯工具,可以直接編輯和生成Java生成的字節碼,以達到對.class文件進行動態修改的效果。熟練使用這套工具,可以讓Java編程更接近與動態語言編程。
下面實現如何修改jar包里的方法
Javassist下載地址 查看
1.准備jar包
my/Solution.java

package my.tools; public class Solution{ public int lengthOfLongestSubstring(String s) { int longSub = 0; if(s.isEmpty()) return 0; if(s.length()==1) return 1; char[] arr = s.toCharArray(); String str = String.valueOf(arr[0]); longSub = 1; for(int i=1;i<s.length();i++){ int pos = str.indexOf(arr[i]); if (pos !=-1) { str = str.substring(pos+1)+arr[i]; }else{ str+=arr[i]; } if(str.length()>longSub) longSub = str.length(); } return longSub; } public int strStr(String haystack, String needle) { if(needle.isEmpty()||needle.equals(haystack)) return 0; int l=needle.length(); int r = haystack.length()-l; for(int i=0;i<r+1;i++){ String tempStr=haystack.substring(i,l+i); if(tempStr.equals(needle)) return i; } return -1; } }
my/Encrypt.java

package my.tools; import javax.crypto.Cipher; import javax.crypto.SecretKey; import javax.crypto.SecretKeyFactory; import javax.crypto.spec.DESKeySpec; import javax.crypto.spec.IvParameterSpec; import java.math.BigInteger; import java.net.URLDecoder; import java.security.MessageDigest; import java.security.NoSuchAlgorithmException; public class Encrypt { public static String md5(String plainText) { byte[] secretBytes = null; try { MessageDigest md = MessageDigest.getInstance("MD5"); md.update(plainText.getBytes()); secretBytes = md.digest(); } catch (NoSuchAlgorithmException e) { throw new RuntimeException("dont have algorithm"); } String md5code = new BigInteger(1, secretBytes).toString(16); for (int i = 0; i < 32 - md5code.length(); i++) { md5code = "0" + md5code; } return md5code; } public static String decrypt(String message, String key) throws Exception { byte[] bytesrc = convertHexString(message); Cipher cipher = Cipher.getInstance("DES/CBC/PKCS5Padding"); DESKeySpec desKeySpec = new DESKeySpec(key.getBytes("UTF-8")); SecretKeyFactory keyFactory = SecretKeyFactory.getInstance("DES"); SecretKey secretKey = keyFactory.generateSecret(desKeySpec); IvParameterSpec iv = new IvParameterSpec(key.getBytes("UTF-8")); cipher.init(2, secretKey, iv); byte[] retByte = cipher.doFinal(bytesrc); return new String(retByte); } public static byte[] convertHexString(String ss) { byte[] digest = new byte[ss.length() / 2]; for (int i = 0; i < digest.length; i++) { String byteString = ss.substring(2 * i, 2 * i + 2); int byteValue = Integer.parseInt(byteString, 16); digest[i] = ((byte) byteValue); } return digest; } public static String decode(String value) { try { return URLDecoder.decode(decrypt(value, "Yst@2_BI"), "utf-8"); } catch (Exception e) { e.printStackTrace(); } return null; } }
新建resource
編譯 javac -sourcepath my my\tools\*.java -d resource
META-INF/MANIFEST.MF
Manifest-Version: 1.0
Created-By: 1.8.0_151 (Oracle Corporation)
進入resource打包 jar -cvfm mytools.jar META-INF\MANIFEST.MF *
使用
import my.tools.Solution; public class Main { public static void main(String[] args) { System.out.println(new Solution().strStr("aacabseew", "ab")); } }
輸出:3
實現的查找第二個字符串出現的位置
2.修改編譯后的class
導入下載的Javassist里的javassist.jar
import javassist.ClassPool; import javassist.CtClass; import javassist.CtMethod; public class Main { public static void main(String[] args) { try { ClassPool.getDefault().insertClassPath("data/mytools.jar"); CtClass c2 = ClassPool.getDefault().getCtClass("my.tools.Solution"); CtMethod[] ms = c2.getDeclaredMethods(); for (CtMethod c : ms) { System.out.println(c.getName()); CtClass[] ps = c.getParameterTypes(); for (CtClass cx : ps) { System.out.println("\t" + cx.getName()); } if (c.getName().equals("strStr") && ps.length == 2 && ps[0].getName().equals("java.lang.String") && ps[1].getName().equals("java.lang.String")) { c.setBody("{System.out.println($1);return 0;}"); } } c2.writeFile(); } catch (Exception e) { e.printStackTrace(); } } }
運行后會在項目根目錄下生成修改后的class文件
3.生成新的jar
將以前的jar包修改mytools.zip
解壓mytools.zip為mytools1,將剛才生成的my/tools/Solution.class替換掉mytools1里的Solution.class,
壓縮mytools1為mytools1.zip
修改文件名為mytools1.jar
導入新的jar包
測試
import my.tools.Solution; public class Main { public static void main(String[] args) { System.out.println(new Solution().strStr("aacabseew","ab")); } }
輸出
aacabseew
0
完成