你好呀,我是歪歪。 上周发布了《我试图通过这篇文章告诉你,这行源码有多牛逼。》这篇文章。 文章中有这样的一段描述: 然后有个读者来问我:
这个问题确实不错,属于一个偶尔用一下能起到奇效的源码调试技巧。所以我决定写个文章来说明一下这个问题。 但是这个技巧确实非常的简单,简单到一句话就能说明白,所以正如标题说到的“短小精悍,简单粗暴,但足够好用”,这篇文章也会非常的短。 首先,把问题换个问法,既然我能把源码注释了,那说明我能修改源码。所以,问题就变成了:我怎么去修改 JDK 的源码呢? 这个问题有很多个回答,但是我这里的回答很简单。把源码拷贝一份出来,原模原样的放一份到自己的项目中即可。 就像是这样: 然后你在使用的时候,直接用你 CV 过来的源码,就行了: 但是我一般使用这个方法的时候,CV 过来时,会把类名称重命名一下,以示区分,其他的啥都不改。 反正不管怎么样吧,这样在你的项目里面有一份“源码”了,这个“源码”和 JDK 里面的源码一模一样,这样你就能随便进行修改了。 比如,我在调用 put 方法的时候,加一点日志输出: 这样测试用例跑起来的时候,就能直接输出你添加的内容: 你都能添加代码了,注释代码,甚至是修改代码逻辑,那还不是手到擒来的事情吗? 对于一些比较复杂的场景,比如异步或者循环等等场景,当你想要在源码中加入输出语句方便进行学习和调试的时候,你就可以用到这招。 这就是我这篇文章要教你的一个关于 JDK 源码的调试技巧。 整体用处不大,但是当你能想到用它的时候,就是发挥奇效的时候。 既然话题都到这里了,那么我再给你补充一个关于第三方框架的类似的调试技巧。 还是先举个例子。 比如我在项目中使用到了 @Async 注解,然后有一个自定义线程池,发起一个请求之后可以看到确实是使用了我的自定义线程池: 然后,问题就来了。 假设,我想让 @Async 注解支持 EL 表达式,也就是这样的写法: 目前,Spring 是不支持这样的配置的,当你这样配置并发起调用,会抛出这样的一个异常: 它会把 ${thread-pool.name} 认为是一个 Bean,然后 Spring 里面并没有这样的一个 Bean,所以抛出找不到 Bean 的异常。 那么怎么才能让 @Async 注解支持 EL 表达式呢? 我之前写过《舒服,给Spring贡献一波源码。》这篇文章,里面用的就是这个案例,有兴趣的话可以去看看,我就不展开说了。 在文章里面,经过分析,我们知道只需要在 org.springframework.aop.interceptor.AsyncExecutionAspectSupport.findQualifiedExecutor(BeanFactory,String) 这个方法中,加入这几行代码就行了:
但是我当时采取的方案是通过 idea 的 Evaluate Expression 功能: 经过评论区提醒,其实用 CV 大法,更加直接、方便。 同样的道理,直接把 AsyncExecutionAspectSupport 这个类粘到我们自己的项目中去: 这里需要注意的是,要保证包名称也一模一样,因为这个方法的底层逻辑是基于类加载机制实现的。 这样,我们就能针对我们自己项目中的 AsyncExecutionAspectSupport 类进行修改: 再次发起调用,这事儿就算成了: 这个方法,适用于任何你能拿到源码的任何第三方框架。 虽然,很多第三方框架里面都会主动留下足够多的扩展点,以便使用者进行定制化开发。 所以我提供的这个方法好像用处并不是很大,但是我当年看 Dubbo 源码的时候,就是这样的看的。 就像是这样,在源码里面加入了大量的输出语句,然后基于输出语句去做分析: 虽然现在想起来,更加正确的操作应该是基于它的 SPI 机制去做。 但是,管它呢,反正当时我就是靠这种歪门邪道,也看的明明白白的。 好了,以上就本文的全部内容。 突出的就是一个短小精悍,简单粗暴,又足够好用。 玩去吧。 |
点击查看更多