转载

爱情 36 技之暗送秋波的技术

【这是一猿小讲的第  26  篇原创分享】

1. 

这篇文章想分享已经很久了,苦于皱巴巴的技术比较生涩难懂,迟迟没有找到好的分享方式,今天结合爱情中暗送秋波的故事的形式,尝试分享一下。

以后如果再有人问你们:能否在加载类的时候,对字节码进行修改?如何实现零侵入、松耦合的方式,采集 Java 程序的各种运行信息?等等一系列的问题时,请不要再懵圈 摇头 ,尝试结合下面分享的爱情故事,回答一下也未尝不可。

准备好小板凳,爱情故事要开始啦。

2. 

自从 Java 那小子靠《爱情36技之追美妹的技术》喜获 Python 菇凉的芳心后,就开始与 Python 菇凉一起进行着各种浪漫的旅行。

Java 先带着心爱的 Python 菇凉去了一趟浪漫的土耳其,然后一起又去了东京和巴黎,其实 Java 特别喜欢迈阿密,和有黑人的洛杉矶,但是 Python 菇凉还是比较想去云南的大理,在云南大理的天涯海角来个海枯石烂、海誓山盟,这样保留着回忆才更有意义。 于是让 Java 那小子订了两张 9012 年 7 月 4 日的飞往云南大理的机票。

往往计划赶不上变化,Java 那小子所在的公司要组织晋升培训,为了能让 Python 菇凉以后的生活更惬意,Java 那小子决定不去云南大理了,但是还想让 Python 菇凉独自去欣赏一下大理的美景。

Java 那小子还实在放心不下 Python 菇凉,担心她坐飞机是否安全? 担心她独自一人是否会难过?

有没有一种方式能够实时关注到她的动态呢? 只有知道 Python 菇凉的动态,Java 那小子才能安心工作,才能时不时的送去爱意的秋波。

好奇、郁闷、纠结的 Java,遂拿起压箱底的《爱情36技》捣腾了好一会儿,忽然捣腾第 34 技—— 暗送秋波的技术 ,Java 眼前一亮,心中乐开了花。

Java 那小子结合自身需求,认真展开了对《爱情 36 技》中的暗送秋波攻略使用解读。

暗送秋波的技术攻略 ,谨献给那些: 想实时关心对方的动态,时不时还想再来个嘘寒问暖的小哥哥小姐姐们。 不过此良策说的好听一点是关心对方,说的不好听一点那就是监控对方(捂嘴笑)。

首先声明此良策,需要为对方亲手打造一款 随身携带的设备 ,需要耗费人力财力... ...

认真读完攻略,Java 下意识的认为,需要先给将要打造的随身携带的设备起个名字,以彰显自己的才能,并能让 Python 菇凉欣然接受这个设备。

Java 灵光乍现,由于设备出自我之手,肯定名字也要随着我的姓,不妨就取个卡哇伊的中文名字为: Java 的随身听,英文名字: JavaWalkmanAgent,学名为: javaagent

考虑到 Python 菇凉远行,不能携带太重的行李。 于是 Java 按照书中的步骤,绞尽脑汁, 一步一步 开始为 Python 菇凉打造这款便捷小巧的随身听设备。

第一步: 确定要把秋天的菠菜送给谁

Java 准备把亲身打造的随身听设备,送给 Python 菇凉。 其中 Python 菇凉去大理会经历两件事,一件事是坐飞机去大理,另一件是去天涯海角来一场浪漫的海誓山盟。


 

package com.love36.ogle;


public class PythonGirl {


public static void goByPlane() {

System.out.println("Python 菇凉说==> 我心爱的Java没有一起坐飞机,心中还是有点失落");

}


public static void sayLoveOath() {

System.out.println("Python 菇凉说==> Java没来大理有遗憾,但是海可枯石可烂,我对他的爱情永不变");

}


public static void main(String[] args) {

System.out.println("随身听 设备说==> 我是 Java 制作的随身听设备,专门替 Java 照顾 Python 菇凉,并默默暗送秋波");

// 1. 坐飞机去大理浪漫

goByPlane();

// 2. 一个人的山盟海誓

sayLoveOath();

}

}

第二步: 用心打造暗送秋波设备的骨架

在 Python 菇凉做每件事之前,JavaWalkmanAgent 都会第一时间拦截到。

爱情 36 技之暗送秋波的技术

第三步: 伺机而动。

Java 提前种植好情意绵绵、嘘寒问暖的菠菜,然后在 Python 菇凉做每件事之前或者之后默默送过去。


 

package com.love36.ogle;


import javassist.ClassPool;

import javassist.CtClass;

import javassist.CtMethod;


import java.lang.instrument.ClassFileTransformer;

import java.lang.instrument.IllegalClassFormatException;

import java.lang.instrument.Instrumentation;

import java.security.ProtectionDomain;


public class JavaWalkmanAgent {


public static void premain(String agentArgs, Instrumentation inst) {

System.out.println("随身听 设备说==> 我是 Java 制作的随身听设备,专门替 Java 照顾 Python 菇凉,并默默暗送秋波");

// 随身听开始发挥自己的作用吧

inst.addTransformer(new ClassFileTransformer() {

@Override

public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined,

ProtectionDomain protectionDomain, byte[] classfileBuffer) throws IllegalClassFormatException {

byte[] transformed = null;

CtClass cl = null;

try {

ClassPool pool = ClassPool.getDefault();

cl = pool.makeClass(new java.io.ByteArrayInputStream(classfileBuffer));

if (!cl.isInterface()) {

CtMethod[] methods = cl.getDeclaredMethods();

for (int i = 0; i < methods.length; i++) {

CtMethod method = methods[i];

//随身听发现 Python 菇凉正在坐飞机,提醒系好安全带

if ("com.love36.ogle.PythonGirl.goByPlane()".equals(method.getLongName())) {

//在 Python 菇凉坐飞机之前问候一下

method.insertBefore("System.out.println(/"Java 那小子说==> 心爱的Python,飞机要起飞了,请系好安全带,一路平安。/");");

//在 Python 菇凉坐飞机之后慰问一下

method.insertAfter("System.out.println(/"Java 那小子说==> 心爱的Python,安全抵达大理,愿玩的开心。/");");

}


//随身听发现 Python 菇凉正在海誓山盟,Java远程寄相思

if ("com.love36.ogle.PythonGirl.sayLoveOath()".equals(method.getLongName())) {

//在 Python 菇凉海誓山盟之后再补充一下

method.insertAfter("System.out.println(/"Java 那小子说==> 我心爱的Python," +

"虽然未能一起去大理的天涯海角浪漫,但是我的心却始终在你身边,注意回来的时候买点土特产(捂嘴笑)。/");");

}

}

transformed = cl.toBytecode();

}

} catch (Exception e) {

System.err.println("随身听 设备说==> 我只照顾像Python这样的姑娘,老爷们就不要硬往上凑了。");

} finally {

if (cl != null) {

cl.detach();

}

}

return transformed;

}

});

}

}

第四步: 为暗送秋波的随身听设备提供配置手册。


 

Manifest-Version: 1.0

Premain-Class: com.love36.ogle.JavaWalkmanAgent

Boot-Class-Path: /app/love36/lib/javassist-3.16.1-GA.jar

至此,暗送秋波的随身听设备在 Java 那小子用心打造下完成了,只见 Java 那小子轻轻松松的在开发工具上把上面做的一切合成到一起,导出了 walkman.jar。

爱情 36 技之暗送秋波的技术

Java 已经忍不住想一键启动,暗送秋波的随身听设备,看看有没有达到预期的效果。于是 Java 在心中默默输入如下命令。

java -javaagent:walkman.jar com.love36.ogle.PythonGirl

结果确实让 Java 那小子 惊喜,因为在没有打扰 Python 菇凉的行程下,Java 那小子也完成了对 Python 菇凉的暗送秋波,嘘寒问暖。

锻炼视力的时候到了,睁大眼睛看效果。

爱情 36 技之暗送秋波的技术

视力不好,那我们再放大一点看,居然 Java 那小子无时无刻不在关心着 Python 姑娘,确实挺神奇。

爱情 36 技之暗送秋波的技术

3. 

故事讲完了,暗送秋波的攻略你 get 到了没?容我们用技术的行话再来总结一下。

暗送秋波是指: 通过 javaagent 植入代码的方式,松耦合的完成目标对象的字节码修改。

经常使用场景: 在方法体前后加入统计方法耗时的代码段;统计 SQL 耗时;最重要的是实现调用链路跟踪;许多 APM 开源轮子,比如 Pinpoint、SkyWalking 等,就是使用这项技术对代码进行的增强。

题外话: 其实我在面试别人的时候,偶尔也会谈到这个知识点,因为可以考察一下求职者有没有相关解决思路以及技术的深度。

好了,今天的故事就讲完了,希望对你们有帮助。

爱情 36 技之暗送秋波的技术

推荐阅读:

爱情36技之追美妹的技术

一篇文章讲透线上应用监控

原文  https://mp.weixin.qq.com/s/fVS0ImL7xZlcddPX9zbeXA
正文到此结束
Loading...