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

javassist API网  http://jboss-javassist.github.io/javassist/html/index.html

javassist参考博客  https://www.ibm.com/developerworks/cn/java/j-dyn0916/

 Ⅰ插桩

自动用例生成(使用Randoop)

评价(对用例筛选冗余)>功能覆盖、语句覆盖(一般用后者)

           >插桩  (插入语句)

              用Javassist实现自动插入语句,即插桩

方法层次的插桩:

  在方法前插入进入方法的语句,方法后插入退出方法的语句

Ⅱ用Javassist对方法插桩

1‘  首先在官网下Javassist的jar包,放置在web项目的lib下,在eclipse里右击项目选择build path的最后一个,导入jar包。

2’  新建一个相对要插桩类a.java的被插桩程序aJssistTiming.java

弄明白这个对方法的插桩,还是得看程序,我对测试第二个博客里的Triangle.java代码进行方法插桩,在方法前后都插入一个输出语句,TriangleJssistTiming.java代码示例如下:

 import java.io.IOException;
import javassist.*; public class TriangleJssistTiming {
public static void main(String argv[]){
try {
String classname="Triangle";
//获取class文件
CtClass clas =ClassPool.getDefault().get(classname); if(clas==null){
System.out.println("classname"+clas+" not found.");
}else{
//调用方法
addTiming(clas,"isTriangle");
addTiming(clas,"getType");
addTiming(clas,"diffOfBorders");
addTiming(clas,"getBorders");
clas.writeFile();
}
} catch (CannotCompileException ex) {
ex.printStackTrace();
} catch (NotFoundException ex) {
ex.printStackTrace();
} catch (IOException ex) {
ex.printStackTrace();
} } private static void addTiming(CtClass cct,String method) throws NotFoundException, CannotCompileException{
//获取方法信息,如果方法不存在,则抛出异常
CtMethod ctMethod = cct.getDeclaredMethod(method);
//将旧的方法名称进行重新命名
String nname = method + "$impl";
ctMethod.setName(nname);
//方法的副本,采用过滤器的方式
CtMethod newCtMethod = CtNewMethod.copy(ctMethod, method, cct, null); //为该方法添加时间过滤器来计算时间,并判断获取时间的方法是否有返回值
String type = ctMethod.getReturnType().getName();
StringBuffer body = new StringBuffer();
body.append("{\n long start = System.currentTimeMillis();\n"); body.append("System.out.println(\" "+ nname +"_in_11\");");
//返回值类型不同
if(!"void".equals(type)) {
body.append(type + " result = ");
}
//可以通过$$将传递给拦截器的参数,传递给原来的方法
body.append(nname + "($$);\n");
//输出方法运行时间差
// body.append("System.out.println(\"Call to method " + nname + " took \" + \n (System.currentTimeMillis()-start) + " + "\" ms.\");\n");
body.append("System.out.println(\" "+ nname +"_out_11\");");
if(!"void".equals(type)) {
body.append("return result;\n");
} body.append("}");
//替换拦截器方法的主体内容,并将该方法添加到class之中
newCtMethod.setBody(body.toString());
cct.addMethod(newCtMethod); //输出拦截器的代码块
System.out.println("拦截器方法的主体:");
System.out.println(body.toString());
}
}

  ▲1'  body.append()插入字符串,里面若是输出语句需要注意用转义字符  “ “ 改成/“    /"

   2’  构造一个addtiming(0方法,方法里构造拦截器时使用一个 java.lang.StringBuffer 来累积正文文本(这显示了处理 String的构造的正确方法。这种变化取决于原来的方法是否有返回值。如果它 有返回值,那么构造的代码就将这个值保存在局部变量中,这样在拦截器方法结束时就可以返回它。如果原来的方法类型为 void ,那么就什么也不需要保存,也不用在拦截器方法中返回任何内容。

   3‘  body.append(nname + "($$);\n") 中 nname 是原来方法修改后的名字。在调用中使用的 $$ 标识符是 Javassist 表示正在构造的方法的一系列参数的方式。通过在对原来方法的调用中使用这个标识符,在调用拦截器方法时提供的参数就可以传递给原来的方法。

3‘  运行

  在eclipse里运行插桩程序和被插桩程序,然后看当地目录下又生成一个Triangle.class文件,大小比原来的大一点。

  在cmd代码目录下运行该文件  java   Triangle

▲原Triangle里没有main方法,需自己插入main方法并创建一个对象引用第一个方法isTriangle()

比较插桩前后Triangle.class,在当地目录找到Triangle.class(后生成的,不在bin目录里,较大的文件),然后用,导入该文件,可以查看原java代码,会发现多了四个类似的方法:

 import java.io.PrintStream;

 public class Triangle
{
public static void main(String[] args)
{
Triangle tr = new Triangle(2L, 3L, 4L);
tr.isTriangle(tr);
} protected long lborderA = 0L;
protected long lborderB = 0L;
protected long lborderC = 0L; public Triangle(long lborderA, long lborderB, long lborderC)
{
this.lborderA = lborderA; this.lborderB = lborderB; this.lborderC = lborderC;
} public boolean isTriangle$impl(Triangle triangle)
{
boolean isTriangle = false;
if ((triangle.lborderA > 0L) && (triangle.lborderA <= 9223372036854775807L) &&
(triangle.lborderB > 0L) && (triangle.lborderB <= 9223372036854775807L) &&
(triangle.lborderC > 0L) && (triangle.lborderC <= 9223372036854775807L)) {
if ((diffOfBorders(triangle.lborderA, triangle.lborderB) < triangle.lborderC) &&
(diffOfBorders(triangle.lborderB, triangle.lborderC) < triangle.lborderA) &&
(diffOfBorders(triangle.lborderC, triangle.lborderA) < triangle.lborderB)) {
isTriangle = true;
}
}
return isTriangle;
} public String getType$impl(Triangle triangle)
{
String strType = "Illegal";
if (isTriangle(triangle)) {
if ((triangle.lborderA == triangle.lborderB) &&
(triangle.lborderB == triangle.lborderC)) {
strType = "Regular";
} else if ((triangle.lborderA != triangle.lborderB) &&
(triangle.lborderB != triangle.lborderC) &&
(triangle.lborderA != triangle.lborderC)) {
strType = "Scalene";
} else {
strType = "Isosceles";
}
}
return strType;
} public long diffOfBorders$impl(long a, long b)
{
return a > b ? a - b : b - a;
} public long[] getBorders$impl()
{
long[] borders = new long[3];
borders[0] = this.lborderA;
borders[1] = this.lborderB;
borders[2] = this.lborderC;
return borders;
} public boolean isTriangle(Triangle paramTriangle)
{
long l = System.currentTimeMillis();
System.out.println(" isTriangle$impl_in_11");
boolean bool = isTriangle$impl(paramTriangle);
System.out.println(" isTriangle$impl_out_11");
return bool;
} public String getType(Triangle paramTriangle)
{
long l = System.currentTimeMillis();
System.out.println(" getType$impl_in_11");
String str = getType$impl(paramTriangle);
System.out.println(" getType$impl_out_11");
return str;
} public long diffOfBorders(long paramLong1, long paramLong2)
{
long l1 = System.currentTimeMillis();
System.out.println(" diffOfBorders$impl_in_11");
long l2 = diffOfBorders$impl(paramLong1, paramLong2);
System.out.println(" diffOfBorders$impl_out_11");
return l2;
} public long[] getBorders()
{
long l = System.currentTimeMillis();
System.out.println(" getBorders$impl_in_11");
long[] arrayOfLong = getBorders$impl();
System.out.println(" getBorders$impl_out_11");
return arrayOfLong;
}
}

 总结:

 Javassist 不仅是一个处理字节码的库,而且更因为它的另一项功能使得它成为试验 classworking 的很好的起点。这一项功能就是:可以用 Javassist 改变 Java 类的字节码,而无需真正了解关于字节码或者 Java 虚拟机(Java virtual machine JVM)结构的任何内容。

用Javassist  API的方法,对插桩程序用ctclass、classpool、CtMethod等方法进行方法插桩,然后生成第二个插桩程序的class文件,此时run这个class文件就会显示插入的语句要输出的内容。

 

Javassist进行方法插桩的更多相关文章

  1. Javsssist用InsertAt()方法对语句插桩

    基于上一篇的方法插桩,这一篇则是进一步的对每行的语句进行插桩. 对于存在分支的方法(例如if(){}else{}),对方法插桩的方法是不能够全部涉及到的.所以要对程序的每条语句进行插桩. 插入什么语句 ...

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

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

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

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

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

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

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

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

  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. APK修改神器:插桩工具 DexInjector

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

  9. Flymeos插桩适配教程

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

随机推荐

  1. 《HelloGitHub》第 14 期

    公告 欢迎通过在 GitHub 上新建 issues 方式推荐项目,我真心希望读者可以在 HelloGItHub,找到真正的编程乐趣! <HelloGitHub>第 14 期 兴趣是最好的 ...

  2. opcache开启前后性能对比

    opcache PHP新的字节码缓存扩展 字节码缓存组件 Zend Optimizer+ 现在更改名字为 Zend opcache了.且在php 5.5版本后,会集成到php的官方组件中,也就没有必要 ...

  3. arcgis sde 导出栅格文件失败,提示“Database user name and current user schema do not match ”.

    具体错误/警告如下: 翻译一下:数据库用户名和当前用户数据库对象的集合不匹配 没有空间参考存在 数据库表没找到 主要还是第一句的问题. 解决方法:切换当前sde账户为能够写入sde的账户,这块不是很了 ...

  4. Autotest添加测试用例小结

    Autotest本身是一个自动化测试框架,可以在上面添加各种测试工具用于系统测试.前几天我在上面添加了几个基于龙芯桌面5.0系统的性能测试工具.现在做以下总结,大体写以下添加的过程. 以unixben ...

  5. CoreML试水--图片识别

    今年的WWDC上,关于人工智能方面Apple开放了CoreML工具包. 今天就趁着时间还早果断的尝试了一下到底有多容易. import UIKit import CoreML import Visio ...

  6. Spring学习(11)---JSR-250标准注解之 @Resource、@PostConstruct、@PreDestroy

    1)@Resource(JSR-250标准注解,推荐使用它来代替Spring专有的@Autowired注解) Spring 不但支持自己定义的@Autowired注解,还支持几个由JSR-250规范定 ...

  7. JavaScript之语句,循环

    JavaScript中语句主要分为三类:顺序,分支,循环. 1.顺序语句: 按照循序依次执行,最普通常见的语句,这里不多赘述. 其结构如下 2.分支语句: 根据条件判断,不同的结果执行不同的语句. 其 ...

  8. JAVAEE学习——struts2_03:OGNL表达式、OGNL与Struts2的结合和练习:客户列表

    一.OGNL表达式 1.简介 OGNL:对象视图导航语言.  ${user.addr.name} 这种写法就叫对象视图导航. OGNL不仅仅可以视图导航.支持比EL表达式更加丰富的功能. 2.使用OG ...

  9. 高性能队列Disruptor系列2--浅析Disruptor

    1. Disruptor简单介绍 Disruptor是一个由LMAX开源的Java并发框架.LMAX是一种新型零售金融交易平台,这个系统是建立在 JVM 平台上,核心是一个业务逻辑处理器,它能够在一个 ...

  10. View Components as Tag Helpers,离在线模板编辑又进一步

    在asp.net core mvc中增加了ViewComponent(视图组件)的概念,视图组件有点类似部分视图,但是比部分视图功能更加强大,它更有点像一个控制器. 使用方法 1,定义类派生自View ...