mini-spring 第五期:bean 管理

本贴最后更新于 2075 天前,其中的信息可能已经时移世改

1.首先解决上一节课的疑问:没有找到参数传递的地方,经过我的 debug,发现参数传递确实是无效的,
image.png
问题在这里,这个地方是创建 mappinghandler 的地方,这里的 paramNameList 只是把带有 requestParam 注解的参数的 vaule 值,也就是这个参数的名字,换句话说是从前端传过来的参数的名称,但并不是真正的参数的值,然后把这个 list 转化为 args 数组,建立一个 mappinghandler。

2.接着在 handler 执行 handle 方法的时候
image.png
向 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);
  • Spring

    Spring 是一个开源框架,是于 2003 年兴起的一个轻量级的 Java 开发框架,由 Rod Johnson 在其著作《Expert One-On-One J2EE Development and Design》中阐述的部分理念和原型衍生而来。它是为了解决企业应用开发的复杂性而创建的。框架的主要优势之一就是其分层架构,分层架构允许使用者选择使用哪一个组件,同时为 JavaEE 应用程序开发提供集成的框架。

    947 引用 • 1460 回帖
  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3201 引用 • 8217 回帖
  • 框架
    47 引用 • 346 回帖 • 1 关注

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • Node.js

    Node.js 是一个基于 Chrome JavaScript 运行时建立的平台, 用于方便地搭建响应速度快、易于扩展的网络应用。Node.js 使用事件驱动, 非阻塞 I/O 模型而得以轻量和高效。

    139 引用 • 269 回帖 • 3 关注
  • 支付宝

    支付宝是全球领先的独立第三方支付平台,致力于为广大用户提供安全快速的电子支付/网上支付/安全支付/手机支付体验,及转账收款/水电煤缴费/信用卡还款/AA 收款等生活服务应用。

    29 引用 • 347 回帖 • 2 关注
  • 锤子科技

    锤子科技(Smartisan)成立于 2012 年 5 月,是一家制造移动互联网终端设备的公司,公司的使命是用完美主义的工匠精神,打造用户体验一流的数码消费类产品(智能手机为主),改善人们的生活质量。

    4 引用 • 31 回帖 • 1 关注
  • InfluxDB

    InfluxDB 是一个开源的没有外部依赖的时间序列数据库。适用于记录度量,事件及实时分析。

    2 引用 • 104 关注
  • 周末

    星期六到星期天晚,实行五天工作制后,指每周的最后两天。再过几年可能就是三天了。

    14 引用 • 297 回帖
  • SendCloud

    SendCloud 由搜狐武汉研发中心孵化的项目,是致力于为开发者提供高质量的触发邮件服务的云端邮件发送平台,为开发者提供便利的 API 接口来调用服务,让邮件准确迅速到达用户收件箱并获得强大的追踪数据。

    2 引用 • 8 回帖 • 506 关注
  • RIP

    愿逝者安息!

    8 引用 • 92 回帖 • 409 关注
  • golang

    Go 语言是 Google 推出的一种全新的编程语言,可以在不损失应用程序性能的情况下降低代码的复杂性。谷歌首席软件工程师罗布派克(Rob Pike)说:我们之所以开发 Go,是因为过去 10 多年间软件开发的难度令人沮丧。Go 是谷歌 2009 发布的第二款编程语言。

    500 引用 • 1396 回帖 • 253 关注
  • 七牛云

    七牛云是国内领先的企业级公有云服务商,致力于打造以数据为核心的场景化 PaaS 服务。围绕富媒体场景,七牛先后推出了对象存储,融合 CDN 加速,数据通用处理,内容反垃圾服务,以及直播云服务等。

    29 引用 • 230 回帖 • 122 关注
  • 数据库

    据说 99% 的性能瓶颈都在数据库。

    346 引用 • 757 回帖
  • wolai

    我来 wolai:不仅仅是未来的云端笔记!

    2 引用 • 14 回帖 • 2 关注
  • Git

    Git 是 Linux Torvalds 为了帮助管理 Linux 内核开发而开发的一个开放源码的版本控制软件。

    211 引用 • 358 回帖 • 2 关注
  • 开源

    Open Source, Open Mind, Open Sight, Open Future!

    415 引用 • 3596 回帖
  • TensorFlow

    TensorFlow 是一个采用数据流图(data flow graphs),用于数值计算的开源软件库。节点(Nodes)在图中表示数学操作,图中的线(edges)则表示在节点间相互联系的多维数据数组,即张量(tensor)。

    20 引用 • 19 回帖 • 1 关注
  • Outlook
    1 引用 • 5 回帖 • 3 关注
  • Latke

    Latke 是一款以 JSON 为主的 Java Web 框架。

    71 引用 • 535 回帖 • 830 关注
  • Hibernate

    Hibernate 是一个开放源代码的对象关系映射框架,它对 JDBC 进行了非常轻量级的对象封装,使得 Java 程序员可以随心所欲的使用对象编程思维来操纵数据库。

    39 引用 • 103 回帖 • 726 关注
  • Pipe

    Pipe 是一款小而美的开源博客平台。Pipe 有着非常活跃的社区,可将文章作为帖子推送到社区,来自社区的回帖将作为博客评论进行联动(具体细节请浏览 B3log 构思 - 分布式社区网络)。

    这是一种全新的网络社区体验,让热爱记录和分享的你不再感到孤单!

    134 引用 • 1127 回帖 • 109 关注
  • 小说

    小说是以刻画人物形象为中心,通过完整的故事情节和环境描写来反映社会生活的文学体裁。

    32 引用 • 108 回帖 • 1 关注
  • danl
    179 关注
  • 以太坊

    以太坊(Ethereum)并不是一个机构,而是一款能够在区块链上实现智能合约、开源的底层系统。以太坊是一个平台和一种编程语言 Solidity,使开发人员能够建立和发布下一代去中心化应用。 以太坊可以用来编程、分散、担保和交易任何事物:投票、域名、金融交易所、众筹、公司管理、合同和知识产权等等。

    34 引用 • 367 回帖 • 1 关注
  • 友情链接

    确认过眼神后的灵魂连接,站在链在!

    24 引用 • 373 回帖 • 1 关注
  • App

    App(应用程序,Application 的缩写)一般指手机软件。

    91 引用 • 384 回帖
  • 安装

    你若安好,便是晴天。

    132 引用 • 1184 回帖 • 2 关注
  • Postman

    Postman 是一款简单好用的 HTTP API 调试工具。

    4 引用 • 3 回帖
  • 学习

    “梦想从学习开始,事业从实践起步” —— 习近平

    172 引用 • 540 回帖
  • Ant-Design

    Ant Design 是服务于企业级产品的设计体系,基于确定和自然的设计价值观上的模块化解决方案,让设计者和开发者专注于更好的用户体验。

    17 引用 • 23 回帖 • 3 关注