不修改源代码,动态注入Java代码的方法(转)
转自:https://blog.csdn.net/hiphoon_sun/article/details/38707927
有时,我们需要在不修改源代码的前提下往一个第三方的JAVA程序里注入自己的代码逻辑。一种情况是拿不到它的源代码,另一种情况是即使有源代码也不想修改,想让注入的代码与第三方程序代码保持相对独立。
public void run() {
System.out.println("A is running.");
}
}
public static void main(String... args) {
A a = new A();
a.run();
}
}
public void run() {
System.out.println("B is running.");
}
}
Java Instrumentation
import java.lang.instrument.ClassFileTransformer;
import java.lang.instrument.IllegalClassFormatException;
import java.security.ProtectionDomain;
import org.objectweb.asm.ClassReader;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.ClassWriter;
class InjectCodeClassWriter extends ClassWriter {
private static final String oldClass = "A";
private static final String newClass = "B";
InjectCodeClassWriter(int flags) {
super(flags);
}
@Override
public int newUTF8(final String value) {
// 将App.class中常量池(constant pool)中类A的名字的字符串改为类B的名字
if (value.equals(oldClass)) {
return super.newUTF8(newClass);
}
return super.newUTF8(value);
}
}
class InjectCodeTransformer implements ClassFileTransformer {
private static final String appClass = "App";
public byte[] transform(ClassLoader loader, String className,
Class classBeingRedefined, ProtectionDomain protectionDomain,
byte[] classfileBuffer) throws IllegalClassFormatException {
if (className.equals(appClass)) {
ClassWriter classWriter=new InjectCodeClassWriter(0);
ClassReader classReader=new ClassReader(classfileBuffer);
classReader.accept(classWriter, 0);
return classWriter.toByteArray();
} else {
return null;
}
}
}
public class InjectCodeAgent {
public static void premain(String args, Instrumentation inst) {
inst.addTransformer(new InjectCodeTransformer());
}
}
jar -cfm InjectCode.jar MANIFEST.MF InjectCodeAgent.class B.class
Class Loader
import java.io.InputStream;
import java.io.IOException;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.concurrent.ConcurrentHashMap;
import org.objectweb.asm.ClassReader;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.ClassWriter;
public class InjectCodeClassLoader extends URLClassLoader {
private static final String appClass = "App";
private static final String oldClass = "A";
private static final String newClass = "B";
private final ConcurrentHashMap<String, Object> locksMap = new ConcurrentHashMap<String, Object>();
public InjectCodeClassLoader(ClassLoader parent) {
super(((URLClassLoader) parent).getURLs(), parent);
}
private static class InjectCodeClassWriter extends ClassWriter {
InjectCodeClassWriter(int flags) {
super(flags);
}
@Override
public int newUTF8(final String value) {
if (value.equals(oldClass)) {
return super.newUTF8(newClass);
}
return super.newUTF8(value);
}
}
private Class defineClassFromClassFile(String className, byte[] classFile)
throws ClassFormatError {
return defineClass(className, classFile, 0, classFile.length);
}
private Class<?> replaceClass(String name)
throws ClassNotFoundException {
InputStream is = getResourceAsStream(name.replace('.', '/') + ".class");
if (is == null) {
throw new ClassNotFoundException();
}
ClassWriter classWriter=new InjectCodeClassWriter(0);
try {
ClassReader classReader=new ClassReader(is);
classReader.accept(classWriter, 0);
} catch (IOException e) {
throw new ClassNotFoundException();
}
Class c = defineClassFromClassFile(name, classWriter.toByteArray());
return c;
}
private Object getLock (String name) {
Object lock = new Object();
Object oldLock = locksMap.putIfAbsent(name, lock);
if (oldLock == null) {
oldLock = lock;
}
return oldLock;
}
@Override
protected Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException {
Object lock = getLock(name);
synchronized(lock) {
Class c = findLoadedClass(name);
try {
if (c == null) {
if (name.equals(appClass)) {
c = replaceClass(name);
} else {
c = findClass(name);
}
}
if (resolve) {
resolveClass(c);
}
return c;
} catch (ClassNotFoundException e) {
}
}
return super.loadClass(name, resolve);
}
}
不修改源代码,动态注入Java代码的方法(转)的更多相关文章
- 【我的Android进阶之旅】Android 源代码中的Java代码中//$NON-NLS-1$ 注释是什么意思?
1.背景 最近在负责公司基础业务和移动基础设施的开发工作,正在负责Lint代码静态检查工作.因此编写了自定义的Lint规则,在编写自定义的Lint规则前,当然是需要去把Google的关于Lint检测的 ...
- [改善Java代码]asList方法产生的List对象不可更改
上一个建议之处了asList方法在转换基本类型数组时候存在的问题,在看下asList方法返回的列表有何特殊的地方.看代码: import java.util.Arrays; import java.u ...
- [改善Java代码]注意方法中传递的参数要求(replaceAll和replace的区别)
有这样一个简单的需求:写一个方法,实现从原始字符串中删除与之匹配的所有子字符串,比如"蓝蓝的天,白云飘"中,删除"白云飘",输出"蓝蓝的天," ...
- java代码---indexOf()方法
总结:indexOf(String str,int index)方法.从参数指定位置开始,如果index值超过了字符串长度,则返回-1 package com.a.b; import java.io. ...
- java代码equals方法
package com.bc; public class Test_6 { // 我们知道java中的每个类都继承自Object类,equals是Object方法之一 String name; int ...
- java代码-----indexOf()方法--从字符串的某个字符的第一次出现的位子开始
总结:方法是indedOf()方法.this is my sister //indexOf()方法是indexOf('m')==7 .那么就是字符m第一次出现的位置是顺数第7个,就会正常显示‘t ...
- java代码----substring()方法是按索引截取字符串。。。下标0开始
总结:按照索引substring(2,5);意思是从字符串的索引为2开始(包括)到第6个字符(不包括)的位置的截取部分 package com.s.x; //substring public clas ...
- (转载)JAVA动态编译--字节代码的操纵
在一般的Java应用开发过程中,开发人员使用Java的方式比较简单.打开惯用的IDE,编写Java源代码,再利用IDE提供的功能直接运行Java 程序就可以了.这种开发模式背后的过程是:开发人员编写的 ...
- 如何向AcmeAir注入问题代码
为什么要注入问题代码? AcmeAir的常规代码是为了压测测试准备的,所以绝大部分的操作都是可以在几十毫秒中就可以正常返回的.为了向用户展示我们APM工具可以在源代码级别发现系统潜在问题,我们需要在A ...
随机推荐
- Java中实现线程同步的三种方法
实现同步的三种方法 多线程共享数据时,会发生线程不安全的情况,多线程共享数据必须同步. 实现同步的三种方法: 使用同步代码块 使用同步方法 使用互斥锁ReetrantLock(更灵活的代码控制) 代码 ...
- cnpm 安装和 command not found
安装cnpm出错 > $ npm install -g cnpm --registry=https://registry.npm.taobao.org 按照淘宝 NPM 镜像安装,cnpm -v ...
- 【leetcode】543. Diameter of Binary Tree
题目如下: 解题思路:最长的周长一定是树中某一个节点(不一定是根节点)的左右子树中的两个叶子节点之间的距离,所以最简单的办法就是把树中所有节点的左右子树中最大的两个叶子节点之间的距离求出来,最终得到最 ...
- 【leetcode】433. Minimum Genetic Mutation
题目如下: 解题思路:我的思路很简单,就是利用BFS方法搜索,找到最小值. 代码如下: class Solution(object): def canMutation(self, w, d, c, q ...
- c++ ofstream使用方法
ofstream是从内存到硬盘,ifstream是从硬盘到内存,流缓冲即是内存空间. 插入器<< : 向流输出数据. cout << "test!" &l ...
- UML-六大关系
六大关系 依赖关系:(只要是在类中使用到了对方就存在依赖关系) 泛化关系:就是继承关系,即依赖关系的特例 实现关系:依赖关系的特例 关联关系:类于类之间的关系,即依赖关系的特例(具有导航性:双向关系或 ...
- 生成100个 "20180520" 这样的时间字符串 写入txt文件
主要想记录一下 . 写NSString 到txt . 数组的去重 . 数组的截取 . 数组分割 代码如下: NSString *year = @"2018"; NSArray *m ...
- UIView的 形变属性transform
// ViewController.m // 形变属性transform // // Created by LiuWei on 2018/4/23. // Copyright © 2018年 xxx. ...
- spring-boot整合shiro实现权限管理
1.运行环境 开发工具:intellij idea JDK版本:1.8 项目管理工具:Maven 4.0.0 2.GITHUB地址 https://github.com/nbfujx/springBo ...
- 各大漏洞平台及SRC的区别和如何批量刷漏洞
批量刷漏洞: 01刷指纹->02刷原始漏洞->03刷CMS->04刷指定政府.教育->05刷众测平台->06刷SRC->07刷国内外.活动 搜索引擎: 百度.goo ...