`

ClassWorking技术

阅读更多
ClassWorking技术

IBM所提出的,动态地监测、修改运行时JVM中的Java字节码文件,从而在充分挖掘应用程序的动态性时,又不会像使用反射那样大大降低系统的性能,Class Working使得静态编码的代码性能与反射的灵活性得以结合。

在ClassWorking中,Java Class文件只不过是一种数据结构而已,通过编写程序或者使用相关的开源项目来对Class文件修改。

ClassWorking,虽然IBM给出的定义中看,更加偏向于对Java类字节码进行修改这个方面,但是由于修改字节码文件一般都是进行运行时的修改,(如果是静态修改的话,那我就直接修改源码然后编译运行就好了)修改往往涉及着Java Instrumentation的相关原理,因此我将Java Instrumentation也纳入ClassWorking的范畴之内。在本节中,给出ClassWorking的大致介绍,由于主要的精力在Starfish和Nutch中,因此也仅仅是一个大致的介绍。

1.1. Java Instrumentation

Java Instrumentation是JDK5.0以来诞生的新技术,JDK5.0中,Java Instrumentation更倾向于作为一种新技术而进行出现,而在JDK6.0中,Java Instrumentation才真正的成熟和实用起来。

java Instrumentation是指可以用独立于应用程序之外的代理(agent)程序来监测和协助运行在JVM上的应用程序。这种监测和协助包括但不限于获取JVM运行时状态,替换和修改类定义等。

使用 Instrumentation,开发者可以构建一个独立于应用程序的代理程序(Agent),用来监测和协助运行在 JVM 上的程序,甚至能够替换和修改某些类的定义。有了这样的功能,开发者就可以实现更为灵活的运行时虚拟机监控和 Java 类操作了,这样的特性实际上提供了一种虚拟机级别支持的 AOP 实现方式,使得开发者无需对 JDK 做任何升级和改动,就可以实现某些 AOP 的功能了。

在 Java SE 6 里面,instrumentation 包被赋予了更强大的功能:启动后的 instrument、本地代码(native code)instrument,以及动态改变 classpath 等等。

1.1.1. 接口和类的介绍

Java Instrumentation的主要内容都包含在包java.lang.instrument之中。总共只有两个接口和一个类

接口ClassFileTransfomer,主要用于类的转换,规定了用户应该是实现的转换函数byte[] transform()。转换完的类是二进制数据,从byte[]数组看出。

1.1.2. Java Instrumentation实例

给出一个使用到Java Instrumentation的例子,来更加真切地体会一下Java的动态性,在这个例子中,我们将在运行时修改类TransClass的字节码,修改的方法是将它替换成另外一个类的字节码,从而动态改变JVM中已经加载好的类:

l 准备工作

首先编写一个要被Instrumentation的类,这个类非常的简单:

public class TransClass {

public int getNumber() {

return 1;

    }

}

这个类拥有一个getNumber()函数,然后调用返回一个固定值1。接下来写一个main函数来进行测试:

public class TestMainInJar {

    public static void main(String[] args) {

        System.out.println(new TransClass().getNumber());

    }

}

Main函数将TransClass类的信息打印了出来,函数的运行结果肯定显示的是1。然后再编写一个类,这个类和TransClass基本相同,唯一不同的地方就是函数的返回值:

public class TransClass2 {

public int getNumber() {

return 2;

    }



返回值变成了2,因此把类名也修改成了TransClass2。

l 代码编写,实现Instrument包中的相应接口

为了能够进行动态替换,需要按照Instrumentation中的API进行代码的编写工作。要实现接口ClassFileTransformer,以及其中的函数byte[] transform()函数:

class Transformer implements ClassFileTransformer {

    public static final String classNumberReturns2 = "TransClass2.class";

    public static byte[] getBytesFromFile(String fileName) {

        try {

            // precondition

            File file = new File(fileName);

            InputStream is = new FileInputStream(file);

            long length = file.length();

            byte[] bytes = new byte[(int) length];


            // Read in the bytes

            int offset = 0;

            int numRead = 0;

            while (offset <bytes.length

                    && (numRead = is.read(bytes, offset, bytes.length - offset)) >= 0) {

                offset += numRead;

            }


            if (offset < bytes.length) {

                throw new IOException("Could not completely read file "

                        + file.getName());

            }

            is.close();

            return bytes;

        } catch (Exception e) {

            System.out.println("error occurs in _ClassTransformer!"

                    + e.getClass().getName());

            return null;

        }

    }


    public byte[] transform(ClassLoader l, String className, Class<?> c,

            ProtectionDomain pd, byte[] b) throws IllegalClassFormatException {

        if (!className.equals("TransClass")) {

            return null;

        }

        return getBytesFromFile(classNumberReturns2);

    }

}

现在对我们示例代码中的关键部分进行解析:Transformer实现了接口ClassFileTransformer及接口函数byte[] transform()。函数transform()传入参数包括该类的类加载器,类名,原字节码字节流等,返回被转换后的字节码字节流。也就是这个函数完成了类的转换。在此类中,我们的transform()判断被传入的类名,如果类名为“TransClass”则执行相应类的修改操作。

而函数getBytesFromFile()完成了实际的类的修改工作:读入TransClass2类的class文件的字节流,然后再替换掉TransClass类的class文件。从而体现了Java的动态性。

最后编写一个premain函数(函数名是固定的),完成Instrumentation的收尾工作。

public class Premain {

    public static void premain(String agentArgs, Instrumentation inst)

            throws ClassNotFoundException, UnmodifiableClas***ception {

        inst.addTransformer(new Transformer());

    }

}

l 打包与运行

将所有的东西打到一个Jar包里面,并修改MANIFEST.MF文件:

Manifest-Version: 1.0

Premain-Class: Premain

最后运行java命令,设置agent代理即可:

java –javaagent:TestInstrument1.jar  TestMainInJar

运行代码之后,控制台将输出2。

1.1.3. Java Instrumentation的其他方式

刚才给出的示例是在Java 5 中Instrumentation的方式,在Java 6中,这种特性被大大的加强了。用户进行Instrumentation的时候可以不用在程序运行的开始就指定agent的jar包,而是在程序运行的时候动态指定:

java attach.Test 33902

attach.Test即我们的Instrument类,用于动态监测和修改其他的类,而33902是我们Instrumentation的目标类的运行时的PID,这里就不再介绍细节了。

1.1.4. Java Instrumentation的缺陷

从上面的介绍和实例可以看出,Java Instrumentation在动态性来说实在是非常的强大,但是有一个比较大的缺陷就是对于修改Java字节码文件方面的弱点:由于我们对底层的Class文件不了解,因此修改起来就十分的困难,在刚才的实例中,我们对于TransClass的修改仅仅是通过替换方式。

1.2. 字节码修改

虽说单纯的JDK API中没有很好的字节码修改的接口函数等功能的提供,但是目前来说,已经存在很多的开源的字节码修改工具和项目了。包括BCEL、ASM、Javassist、CGLIB等开源工具。这些工具的具体的使用方法和细节都没有去了解,因为太过于复杂而且和项目的关系不是很大,因此这里给出一个简单的列表,大致展示每种开源的字节码框架的特点:

表2.1几种流行的字节码修改框架

框架名称
Class修改视角
性能

BCEL
字节码
很差

ASM
访问模式+字节码
最好

Javassist
源代码
稍差

CGLIB
封装了ASM
未知


上表可以看出,给出了几种不同的修改Class视角的方式,源代码层级的修改更加容易,用户就像修改源代码一样进行相应的修改即可。而字节码层面的修改方式则比较的晦涩难懂,用户需要对JVM的底层有一些了解才行。(例如apache的BCEL,在介绍使用BCEL之前,花费了大量的篇章讲述了JVM的底层的简化知识,就是为了使得用户便于使用)而其余的一些开源项目则是在字节码的层面上进行了相应的封装,就是为了便于使用。

1.3. BTrace

BTrace可以说是ClassWorking的一个非常好的开源的软件。它并不像其他字节码修改工具那样,只是单纯进行Class文件的修改,而是结合了Java Instrumentation,使得开发人员可以使用BTrace作为一个工具对代码进行相应的调试。

1.3.1. BTrace的功能结构

有一个BTrace的比较好的公式:

BTrace脚本解析引擎 + Objectweb ASM + JDK6 Instumentation

可以看出BTrace主要由三大部分组成。三大部分各司其职,从用户编程接口到字节码修改的修改再到Java Instrumentation动态监测和修改程序,形成一个完整的机体。

l BTrace脚本解析引擎

这一部分主要面向用户进行编程使用,将用户编写的BTrace进行解析,变成ASM使用的代码,有点像高级语言编译的意味。用户使用BTrace脚本进行编程就变得非常容易了,远远不像BCEL、ASM那样来得麻烦。

用户使用BTrace脚本利用到了Java注解编程的相关技术例如:

@BTrace

public class HelloWorld {

    @OnMethod(

        clazz="java.lang.Thread",

        method="start"

    )



    public static void func() {

        println("about to start a thread!");

    }

}

@OnMethod告诉Btrace解析引擎需要代理的类和方法。这个例子的作用是当java.lang.Thread类的任意一个对象调用 start 方法后,会调用 func 方法。

l ASM修改字节码文件

解析完脚本后,Btrace会使用ASM将脚本里标注的类java.lang.Thread的字节码重写,植入跟踪代码或新的逻辑。在上面那个例子中,Java.lang.Thread这个类的字节码被重写了。并在start方法体尾部植入了 func 方法的调用。ASM的使用,由于比较困难,就不再进行介绍了。

l Java Instrumentation动态性

ASM修改字节码的代码逻辑则被放到了Java Instrumentation中函数transform()中,来完成对特定类的字节码的修改。

这样在软件具体的运行时,可以这样:

Btrace 1234 HelloWorld.java

来对pid为1234的JVM进程进行Instrumentation,体现ClassWorking的完整的精髓。

1.3.2. BTrace的学习难度

可以看出BTrace确实是一个比较强大的工具,但是当前BTrace还是有一些问题的,主要的问题就是BTrace的相关资料实在是太少了。官方给出的资料也只是一些BTrace的示例程序,官网给出的BTrace的源码地址也下载不下来。互联网上搜索BTrace的教程之类的技术帖也讲述的比较浅显,因此目前对于BTrace的了解也非常浅显,希望以后可以努力加强这方面的工作。
分享到:
评论

相关推荐

    Jwebap使用手册(强烈推荐)

    Jwebap的部署应当非常简单,利用动态类加载和字节码注入技术,只需少量配置即可完成部署。这种便捷性使得用户能够在不影响生产系统的情况下轻松启用监控。 ### 5. 可扩展性 Jwebap的核心在于其插件式架构,允许不断...

    (179979052)基于MATLAB车牌识别系统【带界面GUI】.zip

    基于MATLAB车牌识别系统【带界面GUI】.zip。内容来源于网络分享,如有侵权请联系我删除。另外如果没有积分的同学需要下载,请私信我。

    DG储能选址定容模型matlab 程序采用改进粒子群算法,考虑时序性得到分布式和储能的选址定容模型,程序运行可靠 这段程序是一个改进的粒子群算法,主要用于解决电力系统中的优化问题 下面我将对程序进行详

    DG储能选址定容模型matlab 程序采用改进粒子群算法,考虑时序性得到分布式和储能的选址定容模型,程序运行可靠 这段程序是一个改进的粒子群算法,主要用于解决电力系统中的优化问题。下面我将对程序进行详细分析。 首先,程序开始时加载了一些数据文件,包括gfjl、fljl、fhjl1、cjgs和fhbl。这些文件可能包含了电力系统的各种参数和数据。 接下来是一些参数的设置,包括三种蓄电池的参数矩阵、迭代次数、种群大小、速度更新参数、惯性权重、储能动作策略和限制条件等。 然后,程序进行了一些初始化操作,包括初始化种群、速度和适应度等。 接下来是主要的迭代过程。程序使用粒子群算法的思想,通过更新粒子的位置和速度来寻找最优解。在每次迭代中,程序计算了每个粒子的适应度,并更新个体最佳位置和全局最佳位置。 在每次迭代中,程序还进行了一些额外的计算,如潮流计算、储能约束等。这些计算可能涉及到电力系统的潮流计算、功率平衡等知识点。 最后,程序输出了一些结果,包括最佳位置和适应度等。同时,程序还绘制了一些图形,如电压和损耗的变化等。 综上所述,这段程序主要是一个改进的粒子群算法,用于解决电力

    三保一评关系与区别分析

    三保一评关系与区别分析

    Day-05 Vue22222222222

    Day-05 Vue22222222222

    多功能知识付费源码下载实现流量互导多渠道变现+搭建教程

    多功能知识付费源码下载实现流量互导多渠道变现+搭建教程。资源变现类产品的许多优势,并剔除了那些无关紧要的元素,使得本产品在运营和变现能力 方面实现了质的飞跃。多领域素材资源知识变现营销裂变独立版本。 支持:视频、音频、图文、文档、会员、社群、用户发布、创作分成、任务裂变、流量主、在线下载等多种功能,更多功能 正在不断更新中... 支持流量主变现模式,付费下载付费古观看等变现模式。 实现流量互导,多渠道变现。可以独立部署,并绑定自有独立域名,没有域名限制。

    住家保姆的工作职责、照顾老人住家保姆服务内容.docx

    住家保姆的工作职责、照顾老人住家保姆服务内容.docx

    《高温中暑事件卫生》一级(红色),二级(橙色),三级(黄色),四级(蓝色).docx

    《高温中暑事件卫生》一级(红色),二级(橙色),三级(黄色),四级(蓝色).docx

    职业中专技工学校专业评估表.docx

    职业中专技工学校专业评估表.docx

    统计计算使用R一书的源代码Rcode.zip

    统计计算使用R一书的源代码Rcode.zip

    YOLO算法-火灾和人员探测数据集-850张图像带标签-人-烟-火.zip

    YOLO系列算法目标检测数据集,包含标签,可以直接训练模型和验证测试,数据集已经划分好,包含数据集配置文件data.yaml,适用yolov5,yolov8,yolov9,yolov7,yolov10,yolo11算法; 包含两种标签格:yolo格式(txt文件)和voc格式(xml文件),分别保存在两个文件夹中,文件名末尾是部分类别名称; yolo格式:<class> <x_center> <y_center> <width> <height>, 其中: <class> 是目标的类别索引(从0开始)。 <x_center> 和 <y_center> 是目标框中心点的x和y坐标,这些坐标是相对于图像宽度和高度的比例值,范围在0到1之间。 <width> 和 <height> 是目标框的宽度和高度,也是相对于图像宽度和高度的比例值; 【注】可以下拉页面,在资源详情处查看标签具体内容;

    社区居民诊疗健康-JAVA-基于SpringBoot的社区居民诊疗健康管理系统设计与实现(毕业论文)

    社区居民诊疗健康功能描述 社区居民诊疗健康系统是一个为社区居民提供健康管理、疾病预防、诊疗服务和健康教育的综合平台。该平台致力于提升居民的健康水平,通过智能化、便捷化的服务为居民提供高效的健康保障。以下是该系统的主要功能描述: 1. 用户注册与登录 居民注册:居民可以通过身份证、手机号或社交媒体账号进行注册,填写个人基本信息(如姓名、性别、年龄、联系方式等)并创建账户。 健康档案管理:每个居民注册后,系统会自动生成个性化健康档案,记录个人的健康历史、疾病记录、体检报告等。 2. 健康档案与记录管理 个人健康档案:包括居民的基础健康信息、既往病史、用药记录、免疫接种记录、体检报告等。 诊疗记录管理:记录每次诊疗信息,如诊断、治疗方案、用药情况及随访记录。 健康指标监测:定期记录和更新如血压、血糖、体重、体脂等常见健康指标,便于长期追踪和分析。 3. 在线问诊与诊疗服务 在线咨询:居民可以通过平台预约或直接向社区医生发起在线问诊,获取健康咨询、疾病预防建议、用药指导等服务。 远程诊疗:提供视频问诊功能,方便居民与医生进行实时面对面的远程交流,获得更加详细的诊疗建议。 预约就诊:居民可以

    面部、耳廓损伤损伤程度分级表.docx

    面部、耳廓损伤损伤程度分级表.docx

    java毕设项目之ssm校园美食交流系统+vue(完整前后端+说明文档+mysql+lw).zip

    项目包含完整前后端源码和数据库文件 环境说明: 开发语言:Java 框架:ssm,mybatis JDK版本:JDK1.8 数据库:mysql 5.7 数据库工具:Navicat11 开发软件:eclipse/idea Maven包:Maven3.3 服务器:tomcat7

    功能完善的小说CMS系统项目全套技术资料.zip

    功能完善的小说CMS系统项目全套技术资料.zip

    YOLO算法-回收站数据集-501张图像带标签-黑色垃圾箱-绿色垃圾桶-箱子-杯子-老鼠-蓝色垃圾桶.zip

    YOLO系列算法目标检测数据集,包含标签,可以直接训练模型和验证测试,数据集已经划分好,包含数据集配置文件data.yaml,适用yolov5,yolov8,yolov9,yolov7,yolov10,yolo11算法; 包含两种标签格:yolo格式(txt文件)和voc格式(xml文件),分别保存在两个文件夹中,文件名末尾是部分类别名称; yolo格式:<class> <x_center> <y_center> <width> <height>, 其中: <class> 是目标的类别索引(从0开始)。 <x_center> 和 <y_center> 是目标框中心点的x和y坐标,这些坐标是相对于图像宽度和高度的比例值,范围在0到1之间。 <width> 和 <height> 是目标框的宽度和高度,也是相对于图像宽度和高度的比例值; 【注】可以下拉页面,在资源详情处查看标签具体内容;

    java毕设项目之ssm助学贷款+jsp(完整前后端+说明文档+mysql+lw).zip

    项目包含完整前后端源码和数据库文件 环境说明: 开发语言:Java 框架:ssm,mybatis JDK版本:JDK1.8 数据库:mysql 5.7 数据库工具:Navicat11 开发软件:eclipse/idea Maven包:Maven3.3 服务器:tomcat7

    (3127654)超级玛丽游戏源码下载

    内容来源于网络分享,如有侵权请联系我删除。另外如果没有积分的同学需要下载,请私信我。

    hw06.zip

    hw06

Global site tag (gtag.js) - Google Analytics