加入收藏 | 设为首页 |

安博电竞APP下载ios-java怎么完成不损坏程序逻辑的前提下,刺进一些探针

海外新闻 时间: 浏览:231 次

上节说了javaagent和javassist,其实javassist也是根据ASM完成的。一般人不懂得JVM指令的话,底子ASM搞不起来,也用到了访问者的规划方式,看起来跟我们写代码不是一个套路,学习本钱比较高,所以有了javassist。

源码:https://github.com/limingios/netFuture/tree/master/源码/『互联网架构』插桩处理埋点(113)/



(一)插桩

  • 修正上节源码,进行插桩

IdigAgentTest

package com.idig8;
import com.idig8.agent.test.UserServiceImpl;
import java.lang.instrument.Instrumentation;
public class IdigAgentTest {
public static void main(String[] args) {
System.out.println("hello world idig8!");
UserServiceImpl userService = new UserServiceImpl();
userService.hello();
}
}



UserServiceImpl 增加hello办法,打印办法里边的内容

package com.idig8.agent.test;
public class UserServiceImpl {
public UserServiceImpl(){
System.out.println("hello world!");
}
public void hello(){
String p1 ="100";
System.out.print("p1 = "+p1);
}
}



IdigAgent 增加插桩的办法

package com.idig8.agent.test;
import javassist.*;
import java.io.IOException;
import java.lang.instrument.ClassFileTransformer;
import java.lang.instrument.IllegalClassFormatException;
import java.lang.instrument.Instrumentation;
import java.security.ProtectionDomain;
public class IdigAgent {
public static void premain(String args,安博电竞APP下载ios-java怎么完成不损坏程序逻辑的前提下,刺进一些探针 Instrumentation instrumentation) {
System.out.println("premain:" + args);
instrumentation.addTransformer(new ClassFileTransformer() {
@Override
public byte[] transform(ClassLoader loader, String className, Class
ProtectionDomain protectionDomain,
byte[] classfileBuffer) throws IllegalClassFormatException {
if (!"com/idig8/agent/test/UserServiceImpl".equals(className)) {
return null;
}
// javassist 东西 改造
try {
ClassPool pool = new ClassPool();
pool.insertClassPath(new LoaderClassPath(loader));
CtClass ctclass = pool.get("com.idig8.agent.test安博电竞APP下载ios-java怎么完成不损坏程序逻辑的前提下,刺进一些探针.UserServiceImpl");
CtMethod method = ctclass.getDeclaredM安博电竞APP下载ios-java怎么完成不损坏程序逻辑的前提下,刺进一些探针ethod("hello");
method.insertBefore(" System.out.println(System.currentTimeMillis());");
// method.insertBefore("long begin = System.currentTimeMillis();"
// +" System.out.println(begin);");
//
// method.insertAfter(" long end = System.currentTimeMillis();\n" +
// " System.out.println(end - begin);");
return ctclass.toBytecode();
} catch (NotFoundException e) {
e.printStackTrace();
} catch (CannotCompileException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
});
}
}



pom中增加javassist的jar包

 
org.javassist
javassist
3.18.1-GA



  • 代码的履行流程

IdigAgentTest履行这个办法

IdigAgent ->IdigAgentTest->UserServiceImpl的类->UserServiceImpl被插桩发动开端的hello-> 打印hello办法体

premain:abc //IdigAgent 
hello world idig8! //IdigAgentTest
hello world! //UserServiceImpl的类
1562479947074 //UserServiceImpl被插桩发动开端的hello
p1 = 100 //打印hello办法体



  • 留意代码

参加insertBefore中的bgin 和 insertAfter 的 end 经过end-begin 可是后台报错了



javassist.CannotCompileException: [source error] no such field: begin

为什么呢,由于插桩的时分都是以代码快的方式,局部变量。



在实践开发中不必修正原有的办法,而是会新写一个办法,在新办法进行出来,调用安博电竞APP下载ios-java怎么完成不损坏程序逻辑的前提下,刺进一些探针要插桩的办法。新办法的参数炀怎么读和要插桩的保持一致,包含注解,参数。其实有点相似动态署理。

 public void hello$agent()
{
{
long begin = System.currentTimeMillis();
try{
hello();
}finally {
long end = System.currentTimeMillis();
System.out.println(end - begin);
}
}
}



看到这儿,点了重视吧!