一. 学习源码的目的
1. 为了扩展和调优:掌握框架的工作流程和原理
2. 为了提升自己的编程技能:学习他人的设计思想、编程技巧
二. 学习源码的方法
方法一:
1)掌握研究的对象和研究对象的核心概念:搞明白框架都能做什么,是怎么做的。
比如我们要研究Spring的源码,那么研究的对象就是Spring,Spring的核心概念有IOC、DI、AOP等,那么我们就需要搞明白这些核心概念能做什么,是怎么做的
2)从整体到部分
首先要弄清楚整体是由哪些部分组成起来工作的,然后再去研究各个部分是怎么做的。比如我们现在研究Spring IOC,就要先弄清楚IOC由创建bean定义、注册bean定义、bean工厂几部分组合起来工作的,然后再去研究这几个部分是怎么做的
3)找到入口,先理清楚主干流程,然后再去研究各个流程的细节
比如我们要研究Spring是怎么加载xml配置文件去创建实例的,那么我们就需要找到入口,从入口进去以后先不要急着看具体的实现细节,要先理清楚主方法里面大概做了哪些事,然后再去看各个方法是怎么做的
4)多折腾、勤折腾
研究完源码以后,知道源码能做什么、是怎么做的了,那么就可以写一些代码去测试他是不是确实能做到。或者知道源码有一些扩展点,就自己尝试去扩展一下
方法二:
1)掌握研究的对象和研究对象的核心概念:搞明白框架都能做什么,是怎么做的。
比如我们要研究Spring的源码,那么研究的对象就是Spring,Spring的核心概念有IOC、DI、AOP等,那么我们就需要搞明白这些核心概念能做什么,是怎么做的
2)自己动手尝试去实现。
3)自己实现完以后,再去对比看源码里面是怎么实现的,源码里面为什么要那样去实现,有什么好处,学习源码里面好的地方
前面的两种方法,个人比较推荐方法一,因为每个人的水平不一样,方法一相对方法二来说要简单一点
前面已经介绍了两种学习源码的方法,那么怎么深入到源码里面去看别人是怎么做、选择性的去看某一部分源码是怎么做的呢?针对不同的情况,有下面的方法:
第一种方式:
刚刚接触到一个新框架,完全不知道源码是干什么的,这种情况就得找到官方文档里面的使用示例找到入口,然后一步一步的去调试拿到整个调用栈,拿到整个调用栈以后就知道主干流程了,再去分析自己感兴趣的部分是怎么实现的就比较简单了(比较费时)
第二种方式:
经过 学习源码的方法的方法一中的第 1)步 分析,知道源码里面要做某一件事,那就找到官方文档里面的使用示例找到入口,然后找到做某一件事的地方,打个断点,直接debug到做这件事地方拿到调用栈去分析,看都经过了哪些处理,然后去分析自己感兴趣的部分是怎么实现的。
比如IOC容器加载配置文件创建bean实例时一定会有注册bean定义的步骤,那么我们就先根据类的继承体系找到注册bean定义的类和对应的方法的实现的地方,在里面打个断点,直接debug到注册的方法的断点的地方时就能获取到调用栈了,然后再根据调用栈分析Spring要完成根据xml配置创建bean实例都有哪些类参与进来。
下面就来举例说明:
首先从官方文档可以得出如下使用示例:
官方文档:
使用示例:
ApplicationContext context = new ClassPathXmlApplicationContext("application.xml");
然后进入ClassPathXmlApplicationContext里面
我们知道IOC容器的工作过程是加载xml、解析xml得到bean定义、把bean定义注册到bean工厂里面、bean工厂根据bean定义创建bean实例,那么我们就根据ClassPathXmlApplicationContext的继承体系先找到哪个类里面持有bean工厂,找到持有bean工厂的地方以后先看有没有注册bean定义相关的方法,根据继承体系寻找,最终发现在父类AbstractRefreshableApplicationContext里面持有Bean工厂DefaultListableBeanFactory:
/** Bean factory for this context. */ @Nullable private DefaultListableBeanFactory beanFactory;
通过在AbstractRefreshableApplicationContext里面查找,没有找到注册bean定义相关的方法,那么我们就看DefaultListableBeanFactory的里面有没有注册bean定义相关的方法,最终发现DefaultListableBeanFactory里面果然有注册bean定义的方法registerBeanDefinition
@Override public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition) throws BeanDefinitionStoreException {
通过查看DefaultListableBeanFactory继承体系,我们可以看到DefaultListableBeanFactory实现了BeanDefinitionRegistry这个接口来实现bean定义注册
那么我们就在registerBeanDefinition(String beanName, BeanDefinition beanDefinition)方法里面打个断点,然后debug运行前面的示例代码到这里:
到这里我们就能拿到整个调用栈了:
拿到调用栈以后,我们就可以来分析Spring是怎么加载xml、解析xml获取bean定义、注册bean定义到bean工厂的了
那么怎么来具体分析调用栈呢?
主要看调用栈(看从开始到注册bean定义这件事情)用到了哪些类的哪些方法,看传参。工作是如何分配、分工协作完成的。
看传参重点是看输入的参数(如xml配置文件的路径)在这些类中是怎么变化的(代码的本质其实就是对输入数据的各种处理得到最终想要的结果),从而知道每一个类是干什么用的。分析完整个调用栈以后,想要了解哪一部分就点击对应的调用栈去分析就行了。从调用栈可以看到要加载xml、解析xml获取bean定义、注册bean定义到bean工厂这些事需要三个类依次参与进来:
说明:
方法里面含有<init>的表示是在构造函数进行初始化,方法里面带有(AbstractApplicationContext).refresh()的表示调用的是AbstractApplicationContext父类的refresh()方法,其他的类似
1)ClassPathXmlApplicationContext:初始化、准备BeanFactory
2)XmlBeanDefinitionReader:输入的xml配置文件路径的字符串到Document的转换
3)DefaultBeanDefinitionDocumentReader 解析Document得到bean定义持有器BeanDefinitionholder
4) 后面的步骤就是把BeanDefinitionholder里面的Bean定义注册到Bean工厂里面
三、Eclipse里面查看源码的常用快捷键和方法
前提:在eclipse里面导入了源码或者用maven引入了源码的依赖
1. 查找源码的类的快捷键
Ctrl+Shift+T 然后输入要看的类
2. 查看一个类继承了哪些类的以及有哪些子类
选中要查看的类,右键-选择Open Type Hierarchy(打开类的继承体系)
public interface ApplicationContext extends EnvironmentCapable, ListableBeanFactory,HierarchicalBeanFactory,MessageSource,ApplicationEventPublisher, ResourcePatternResolver
查看继承了哪些类:
查看有哪些子类:
选中部分里面对应的成员:
3. 查看一个类都有哪些子类
选择要看的类 然后Ctrl+T