1.首先解决上一节课的疑问:没有找到参数传递的地方,经过我的 debug,发现参数传递确实是无效的,
问题在这里,这个地方是创建 mappinghandler 的地方,这里的 paramNameList 只是把带有 requestParam 注解的参数的 vaule 值,也就是这个参数的名字,换句话说是从前端传过来的参数的名称,但并不是真正的参数的值,然后把这个 list 转化为 args 数组,建立一个 mappinghandler。
2.接着在 handler 执行 handle 方法的时候
向 invoke 传递的参数,仍然是那个 args 数组。所以也就不存在传值的了。
3.至此,把 springmvc 的大致过程捋一捋
@1.项目启动的时候首先是创建一个 tomcat 服务器,服务器中绑定了一个 dispatcherServlet,这个 servlet 的映射为“/”根路径,意思是所有的请求都会经过这个请求。
@2.启动 server 之后,会进行类扫描,把项目包下的所有类都扫描到一个集合中。
@3.进行 mappinghandler 的创建,这个创建过程就是对上一步的类进行解析,对带有 controller 的进行解析,然后对类中带有 requestMapping 和 requestParam 注解的方法进行解析。这里是每一个方法对应一个 handler
@4.请求来临,被 dispatcherServlet 拦截,用循环根据请求的 url 判断由哪个 mappingHandler 所处理
@5.找到了对应的 handler,这个时候由于 handler 已经具有了如下属性:如自己是属于哪个 controller,参数是什么,然后利用反射初始化 controller 实例,接着用 invoke 方法调用处理,并返回处理结果。
4.接着完成框架,bean 的管理
创建的流程:
@1 循环遍历所需要创建的 bean
@2 判断这个 bean 是否需要先创建依赖,如果不需要,则直接创建 bean 并放入 beanFactory,如果需要创建依赖,则先判断所依赖的 baen 是否在 beanFactory 中,如果存在的拿到该依赖并 set,然后成功创建该 bean,如果不再 beanFactory 中,则先放弃创建这个 bean,先创建后边的。
@3 以此循环,知道所有的 bean 创建结束
@4 这样有一个问题,就是无法处理相互依赖的情况,类似于死锁,这个时候就需要判断每次循环之后所需要创建 bean 的数量是不是有变化,如果没有变化,说明陷入了训话,需要抛出异常,退出循环。
5.首先创建与 bean 管理相关的注解,@Bean 和 @AutoWired
package cn.chenforcode.beans;
import java.lang.annotation.*;
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface Bean {
}
package cn.chenforcode.beans;
import java.lang.annotation.*;
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface AutoWired {
}
6.创建 beanFactory
private static Map<Class<?>, Object> classToBean = new ConcurrentHashMap<>(); public static Object getBean(Class<?> cls) { return classToBean.get(cls); }
保存一个 map,存储类和对象,能够根据类类型得到类对象
然后写出一个方法
7.实现 initBean 方法,实现 bean 的初始化
/** * @Author <a href="http://www.chenforcode.cn">PKUCoder</a> * @Date 2019/11/7 11:44 下午 * @Param [classList] * @Return void * @Description 根据扫描到的类定义,进行bean的创建 **/ public static void initBean(List<Class<?>> classList) throws Exception { List<Class<?>> toCreate = new ArrayList<>(classList); //如果容器中还有,就一直循环 while (toCreate.size() != 0) { //保存每次遍历之前的容器大小 int remainSize = toCreate.size(); //进行遍历创建 for (int i = 0; i < toCreate.size(); i++) { //如果创建成功就把它给移除 if (finishCreate(toCreate.get(i))) { toCreate.remove(i); } } //如果一次遍历之后,大小没有变化,说明陷入了循环依赖 if (toCreate.size() == remainSize) { throw new Exception("cycle dependency"); } } }
8.实现 finishBean 方法,完成 bean 创建的具体流程
/** * @Author <a href="http://www.chenforcode.cn">PKUCoder</a> * @Date 2019/11/8 12:06 上午 * @Param [cls] * @Return boolean * @Description 实现bean创建的具体流程 **/ private static boolean finishCreate(Class<?> cls) throws IllegalAccessException, InstantiationException { //如果没有带这个两个注解,说明不需要创建,则直接返回true if (!cls.isAnnotationPresent(Controller.class) && !cls.isAnnotationPresent(Bean.class)) { return true; } //创建这个类的实例 Object bean = cls.newInstance(); //然后判断这个类是否需要其他的依赖 for (Field field: cls.getDeclaredFields()) { //如果带有@AutoWired注解,则说明需要先初始化该属性 if (field.isAnnotationPresent(AutoWired.class)) { //获取到这个属性的type Class<?> fieldType = field.getType(); //从工厂中拿到这个bean Object reliantBean = BeanFactory.getBean(fieldType); if (reliantBean == null) { //如果工厂中还没有这个bean,那么这次创建失败 return false; } //把这个字段的属性设置为可接入,就相当于把private变成public field.setAccessible(true); //将这个属性设置进去 field.set(bean, reliantBean); //加入beanFactory BeanFactory.classToBean.put(cls, bean); } } return true; }
9.最后,在 handlerMapping 修改 controller 的创建方式
把直接创建实例变成从工厂中获取
Object ctl = BeanFactory.getBean(controller);
欢迎来到这里!
我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。
注册 关于