今天在学习java的sychronized偏向锁的时候,看了一篇视频教程
https://www.bilibili.com/video/BV16J411h7Rd?p=83&spm_id_from=pageDriver
教程中老师引用了一个第三方的jar包(jol-core),maven坐标如下,来打印锁对象的Mark Word字节码,从而更直观察地多线程下加偏向锁的情况。
<dependency>
<groupId>org.openjdk.jolgroupId>
<artifactId>jol-coreartifactId>
<version>0.16version>
dependency>
这些都是挺常规的操作,接着老师的骚操作来了,扩展了jar包里面的方法。弹幕里面都是惊呼声,老师改了jar包,我们学习者没法复现了呀。
我寻思是不是用了动态代理啥的做的,可是一看他调用的方法是ClassLayout.parseInstance(dog).toPrintableSimple()
,这个类的构造方法是私有的,另外内部都是静态方法。作为一个工具类扩展 出来的,那动态代理就行不通了。toPrintableSimple()
这个方法是jar包中不存在的,也就是说这个方法是老师自己写的。
抱着刨根问底的心态,我尝试去下载源码查看ClassLayout.toPrintable()这个方法,代码大概100多行的样子,就是基于这个方法扩展一个toprintableSimple()方法,然后重新编译项目,找到ClassLayout.class字节码,然后把这个文件去替换官方jar包中的这个类,再去项目里面一刷新,最后一运行,nice搞定,跟老师的一样炫酷。
下面我把这个改官方jar包代码的骚操作告诉你,分为以下几个步骤:
在idea右侧maven导航栏里面找到对应的依赖项,然后右键下载源代码
在左侧项目的外部库下面展开对应jar包,然后右键选打开于Explorer
在系统文件浏览器里面再用压缩工具解压源代码
打开解压后的目录跟我们平时看到的maven项目不太一样,这是因为打成jar包以后,文件目录会变
去idea新建一个项目,就跟这个文件夹同名好了jol-core-0.16-sources
,将刚才解压的目录下面的代码放到maven指定的位置
org
放到项目的main/java
META-INF
放到项目的resources
目录下META-INF\maven\org.openjdk.jol\jol-core
下的pom.xml
放到项目根目录下刷新maven项目
这里如果编译代码的话会报一个错误,提示sun.misc不存在,经过搜索资料得知是jdk版本过高导致的,在项目结构里面改成jdk 8即可
编辑org.openjdk.jol.info.ClassLayout
类,
先按Ctrl + o
看一下ClassLayout
中的方法,只有toPrintable()
和toPrintable(Object)方法,
我们参考着写toPrintableSimple()
和toPrintableSimple()
方法
public String toPrintableSimple() {
return toPrintableSimple(classData.instance());
}
public String toPrintableSimple(Object instance) {
StringBuilder sb = new StringBuilder();
String markStr = "";
String remind = "";
int markSize = model.markHeaderSize();
int markOffset = 0;
if (instance != null) {
VirtualMachine vm = VM.current();
if (markSize == 8) {
long mark = vm.getLong(instance, markOffset);
markStr = Long.toBinaryString(mark);
remind = parseMarkWord(mark);
} else if (markSize == 4) {
int mark = vm.getInt(instance, markOffset);
markStr = Integer.toBinaryString(mark);
remind = parseMarkWord(mark);
}
}
// 高位补0
int i = 1;
for (; i <= 8 * markSize - markStr.length(); i++) {
sb.append('0');
if (i % 8 == 0) {
sb.append(" ");
}
}
for (; i <= 8 * markSize; i++) {
sb.append(markStr.charAt(i - (8 * markSize - markStr.length()) - 1));
if (i % 8 == 0) {
sb.append(" ");
}
}
sb.append(remind);
return sb.toString();
}
按Ctrl + F9
重新编译代码
项目重新编译后,去target目录下,找到org.openjdk.jol.info.ClassLayout
对应的class文件,复制,用压缩软件,我用的WinRAR,打开maven仓库中对应的jar包,进到org\openjdk\jol\info
粘贴,这就完成了替换。
然后去之前的项目中试着调用toPrintableSimple()
看看吧,调用成功!
如果不想污染官方的jar包,也可以自己导出jar包,具体方法很简单,去idea右侧maven导航栏的项目的生命周期里面install
即可,jar包会自动导出到本地的maven仓库中。
注:1. 需要把pom.xml中的java-doc和license的plugin注释掉,否则会报错; 2.需要把在pom.xml中指定version,否则会覆盖官方的jar包
打包后,去maven仓库查看
尝试在项目中引入,不报错且能正常运行