基于上一篇的方法插桩,这一篇则是进一步的对每行的语句进行插桩。

对于存在分支的方法(例如if(){}else{}),对方法插桩的方法是不能够全部涉及到的。所以要对程序的每条语句进行插桩。

插入什么语句呢?可以插入包括以下的内容:

1‘  classname

2’  linenumber

此时需要用到javassist  api里的类CtMethod的方法insertAt(),对程序的每条语句进行插桩。

逻辑思想就是:读取class文件,对类的方法进行获取,读取每个类的行号范围,在每个行号前用insertAt()方法插入语句,输出行号和类名。

示例代码如下:

import java.io.IOException;
import javassist.*; public class TriangleInsertAt {
public static void main(String args[]) throws NotFoundException, CannotCompileException, IOException{
//获取class文件
CtClass clas=ClassPool.getDefault().get("Triangle");
if(clas==null){
//方法未找到
System.out.println("classname "+clas+" not found");
}else{
//isTriangle方法的行号范围
//此处对Triangle四个方法分别进行语句插桩
insertToLine(clas,"isTriangle",31,48);
insertToLine(clas,"getType",55,77);
insertToLine(clas,"diffOfBorders",83,85);
insertToLine(clas,"getBorders",90,96);
clas.writeFile();
}
}
public static void insertToLine (CtClass ccl,String method,int i,int j) throws NotFoundException, CannotCompileException{
//获取方法信息,如果方法不存在,则抛出异常
CtMethod ctMethod = ccl.getDeclaredMethod(method);
//将旧的方法名称进行重新命名
String nname = method + "$impl";
ctMethod.setName(nname);
//方法的副本
CtMethod newCtMethod = CtNewMethod.copy(ctMethod, method, ccl, null);
//用for循环语句插桩
for(;i<j;i++){
newCtMethod.insertAt(i,true,"System.out.println(\" classname "+ccl.getName()+" linenumber \"+" + i +");");
}
ccl.addMethod(newCtMethod);
System.out.println("run to here");
}
}

▲ 代码中插入输出语句的那行 

newCtMethod.insertAt(i,true,"System.out.println(\" classname "+ccl.getName()+" linenumber  \"+" + i +");");
  这里需要注意引号的不同意义,\" \"是双引号的转义字符,而加号也是有此处字符串相加的,也有System.out.println()里的输出之间的加号。
 

这只是一个基于方法插桩的语句插桩,逻辑缜密上并不完善,待要解决的存有:

1‘  获取行号,需要借助于文件复制,在每循环一次readLine方法,就对linenumber执行+1,在此需要做出判断,该行是否为方法体,构造函数,main方法,注释,空行(此处的空行为代码里的空行,并不意味着null而是换行的“\n”),创建一个数组,对特殊行号进行标记,我目前的思想是构造二维数组,记录行号和该行的属性。(此处需用到IO流的文件读取和写入的方法,在判断时会用到contains()和startwith()方法等;

2’  根据上述的代码生成一个新的class文件,借助工具查看插桩后的java代码,会发现输出的行号并不是按顺序,而且有的行没有,有的行有两次输出。此为编译后生成的class文件的显示。

对此,需要进行缜密的插入判断,借助于第一条的建议进行判断性的插入。

3‘  整个程序中需要语句插桩的有main方法,构造函数,方法,此代码对前两者并没有插入,问题出在还借用了方法插桩的方法,对方法进行重命名,插入输出语句,然后将新的方法也写入到源文件中。而语句插桩并不需要重写方法,而是直接经过判断将输出语句写入到每行符合条件的代码前。输出的结果应该并没有整个程序的行数那么多,而且行数需递增型输出。

总结:

java基础重中之重,仅是一个简单的语句插桩就涉及到了文件复制,读取判断,提取字符串,迭代,Javassist API和JDK API都需要运用到很多方法。

将语句插桩进一步完善的思路:首先文件复制时,创建数组记录行数和出现方法和构造函数的特殊行数;其次读取class文件时,用到CtMethod类的insertAt方法进行选择性的插桩,注释,构造方法的那一行,以及类似“\n”," }","{"之类的字符串都要进行排除;之后运行需要进一步完善不足之处。

一个功能只能经过不断地完善才能尽量的减少bug和提升性能,没有十全十美之时。

Javsssist用InsertAt()方法对语句插桩的更多相关文章

  1. Javassist进行方法插桩

    javassist官网 http://jboss-javassist.github.io/javassist/ javassist API网 http://jboss-javassist.github ...

  2. zorka源码解读之通过beanshell进行插桩的流程

    zorka中插桩流程概述 1.在SpyDefinition中配置插桩属性,将SpyDefinition实例提交给插桩引擎.2.SpyDefinition实例中包含了插桩探针probes,probe插入 ...

  3. APK修改神器:插桩工具 DexInjector

    本文介绍了一个针对Dex进行插桩的工具,讲解了一下直接修改Dalvik字节码和Dex文件时遇到的问题和解决方法 作者:字节跳动终端技术-- 李言 背景 线下场景中,我们经常需要在APK中插入一些检测代 ...

  4. 插桩 inline hook 动态二进制插桩的原理和基本实现过程

    插桩测试 https://source.android.google.cn/compatibility/tests/development/instrumentation https://zhuanl ...

  5. 开发 IDEA Plugin 引入探针,基于字节码插桩获取执行SQL

    作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 片面了! 一月三舟,托尔斯泰说:"多么伟大的作家,也不过就是在书写自己的片 ...

  6. Java Instrumentation插桩技术学习

    Instrumentation基础 openrasp中用到了Instrumentation技术,它的最大作用,就是类的动态改变和操作. 使用Instrumentation实际上也可以可以开发一个代理来 ...

  7. 手淘架构组最新实践 | iOS基于静态库插桩的⼆进制重排启动优化 抖音研发实践:基于二进制文件重排的解决方案 APP启动速度提升超15% 编译期插桩

    抖音研发实践:基于二进制文件重排的解决方案 APP启动速度提升超15% 原创 Leo 字节跳动技术团队 2019-08-09 https://mp.weixin.qq.com/s/Drmmx5JtjG ...

  8. 方案设计:基于IDEA插件开发和字节码插桩技术,实现研发交付质量自动分析

    作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 如何保证代码质量? 业务提需求,产品定方案,研发做实现,测试验流程.四种角色的相互配 ...

  9. Flymeos插桩适配教程

    插桩适配前提,安装Ubuntu或者其他linux系统. 安装JDK7 sudo apt--jdk Ubuntu 16.04与基于它的版本,需要添加源 sudo add-apt-repository p ...

随机推荐

  1. fiddler导致页面确定按钮无法使用(测试遇到的问题经验)

    这几天在测试的是遇到几个问题,就是在删除或者保存有些提示信息的时候 比如下面这种: 点击确定的时候,一直无响应,换了几台电脑其他电脑都是正常的,本机清楚缓存.关闭浏览器重新打开.重启电脑都试过了了就是 ...

  2. Unity C# GetSaveFileName()的应用

    本文原创,转载请注明出处:http://www.cnblogs.com/AdvancePikachu/p/6944870.html 唉哟,这次厉害咯,网上搜罗了好久,终于被我找到汉化的保存对话框了,根 ...

  3. {网络编程}和{多线程}应用:基于TCP协议【实现多个客户端发送文件给一个服务器端】--练习

    要求: 实现多个客户端发送文件给一个服务器端 提示:多个人创建客户端发送文件,服务端循环接收socket,从socket中获取文件 说明:这里我们只要建立一个服务端就可以了,然后让多台电脑使用客户端给 ...

  4. EJB基础知识

    本人博客文章网址:https://www.peretang.com/basic-knowledge-of-ejb/ 什么是EJB 可移植的, 可重用的, 可伸缩的业务应用程序的平台 为什么选择EJB ...

  5. vue-cli创建自己的项目

    vue-cli 是一个官方发布 vue.js 项目脚手架,使用 vue-cli 可以快速创建 vue 项目,GitHub地址是:https://github.com/vuejs/vue-cli 一. ...

  6. 没有main方法真的不能执行代码了吗?

    今天看北大慕课遇到一段代码,于是下载下来跑了一下,奇葩的是,没有main方法既没报错,还出了结果. 下面贴出代码: class InitialTest { public static void mai ...

  7. ui-router 父子state共享数据

    如果ui-view是嵌套的,那么在子view中,可以直接引用 $scope.$parent 即可.

  8. CC2530入门教程-02】CC2530的通用I/O端口输入和输出控制

    第2课  CC2530的通用I/O端口输入和输出控制 广东职业技术学院  欧浩源 一.CC2530的引脚概述 CC2530微控制器采用QFN40封装,有40 个引脚.其中,有21个数字I/O端口,其中 ...

  9. mysql时间戳与日期格式的相互转换

    1.UNIX时间戳转换为日期用函数: FROM_UNIXTIME()[sql] view plain copyselect FROM_UNIXTIME(1156219870); 输出:2006-08- ...

  10. 【Android Developers Training】 96. 运行一个同步适配器

    注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...