标签:
本地debug的时候,可以实时编译并更新代码,线上也可以不停服来动态更新类,即所说的java热部署。
1
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 |
public class JavaAgent {
public static final Logger logger = LoggerFactory.getLogger(JavaAgent.class); private static String classesPath; private static String jarPath; private static VirtualMachine vm; private static String pid; static { classesPath = JavaAgent.class.getClassLoader().getResource("").getPath(); logger.error("java agent:classpath:{}", classesPath); jarPath = getJarPath(); logger.error("java agent:jarPath:{}", jarPath); // 当前进程pid String name = ManagementFactory.getRuntimeMXBean().getName(); pid = name.split("@")[0]; logger.error("当前进程pid:{}", pid); } /** * 获取jar包路径 * @return */ public static String getJarPath() { // StringUtils是jar文件内容 URL url = StringUtils.class.getProtectionDomain().getCodeSource().getLocation(); String filePath = null; try { filePath = URLDecoder.decode(url.getPath(), "utf-8");// 转化为utf-8编码 } catch (Exception e) { e.printStackTrace(); } if (filePath.endsWith(".jar")) {// 可执行jar包运行的结果里包含".jar" // 截取路径中的jar包名 filePath = filePath.substring(0, filePath.lastIndexOf("/") + 1); } File file = new File(filePath); filePath = file.getAbsolutePath();//得到windows下的正确路径 return filePath; } private static void init() throws IOException, AttachNotSupportedException, AgentLoadException, AgentInitializationException { // 虚拟机加载 vm = VirtualMachine.attach(pid); vm.loadAgent(jarPath + "/javaagent.jar"); Instrumentation instrumentation = JavaDynAgent.getInstrumentation(); Preconditions.checkNotNull(instrumentation, "initInstrumentation must not be null"); } private static void destroy() throws IOException { if (vm != null) { vm.detach(); } } /** * 重新加载类 * * @param classArr * @throws Exception */ public static void javaAgent(String root, String[] classArr) throws ClassNotFoundException, IOException, UnmodifiableClassException, AttachNotSupportedException, AgentLoadException, AgentInitializationException { init(); try { // 1.整理需要重定义的类 List<ClassDefinition> classDefList = new ArrayList<ClassDefinition>(); for (String className : classArr) { Class<?> c = Class.forName(className); String classPath = (StringUtils.isNotBlank(root) ? root : classesPath) + className.replace(".", "/") + ".class"; logger.error("class redefined:" + classPath); byte[] bytesFromFile = Files.toByteArray(new File(classPath)); ClassDefinition classDefinition = new ClassDefinition(c, bytesFromFile); classDefList.add(classDefinition); } // 2.redefine JavaDynAgent.getInstrumentation().redefineClasses(classDefList.toArray(new ClassDefinition[classDefList.size()])); } finally { destroy(); } } public static void main(String[] args) throws Exception { PortUtil.test(); javaAgent(null, new String[] {"com.agileeagle.webgame.framework.util.PortUtil"}); PortUtil.test(); } } |
1
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 |
public class JavaDynAgent {
private static Instrumentation instrumentation; private static Object lockObject = new Object(); public JavaDynAgent() { } public static void agentmain(String args, Instrumentation inst) { Object var2 = lockObject; synchronized(lockObject) { if(instrumentation == null) { instrumentation = inst; System.out.println("0->" + inst); } else { System.out.println("1->" + inst); } } } public static Instrumentation getInstrumentation() { return instrumentation; } } |
标签:
原文地址:http://www.cnblogs.com/cm4j/p/hot_deploy.html