mini-spring 第五期:bean 管理

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

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 回帖 • 1 关注
  • Java

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

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

相关帖子

欢迎来到这里!

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

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

推荐标签 标签

  • GitLab

    GitLab 是利用 Ruby 一个开源的版本管理系统,实现一个自托管的 Git 项目仓库,可通过 Web 界面操作公开或私有项目。

    46 引用 • 72 回帖 • 2 关注
  • Git

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

    211 引用 • 358 回帖
  • Latke

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

    71 引用 • 535 回帖 • 829 关注
  • Dubbo

    Dubbo 是一个分布式服务框架,致力于提供高性能和透明化的 RPC 远程服务调用方案,是 [阿里巴巴] SOA 服务化治理方案的核心框架,每天为 2,000+ 个服务提供 3,000,000,000+ 次访问量支持,并被广泛应用于阿里巴巴集团的各成员站点。

    60 引用 • 82 回帖 • 615 关注
  • Markdown

    Markdown 是一种轻量级标记语言,用户可使用纯文本编辑器来排版文档,最终通过 Markdown 引擎将文档转换为所需格式(比如 HTML、PDF 等)。

    171 引用 • 1537 回帖
  • 周末

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

    14 引用 • 297 回帖
  • Windows

    Microsoft Windows 是美国微软公司研发的一套操作系统,它问世于 1985 年,起初仅仅是 Microsoft-DOS 模拟环境,后续的系统版本由于微软不断的更新升级,不但易用,也慢慢的成为家家户户人们最喜爱的操作系统。

    228 引用 • 476 回帖
  • 大数据

    大数据(big data)是指无法在一定时间范围内用常规软件工具进行捕捉、管理和处理的数据集合,是需要新处理模式才能具有更强的决策力、洞察发现力和流程优化能力的海量、高增长率和多样化的信息资产。

    89 引用 • 113 回帖 • 1 关注
  • Caddy

    Caddy 是一款默认自动启用 HTTPS 的 HTTP/2 Web 服务器。

    10 引用 • 54 回帖 • 181 关注
  • V2Ray
    1 引用 • 15 回帖 • 4 关注
  • BND

    BND(Baidu Netdisk Downloader)是一款图形界面的百度网盘不限速下载器,支持 Windows、Linux 和 Mac,详细介绍请看这里

    107 引用 • 1281 回帖 • 36 关注
  • Ngui

    Ngui 是一个 GUI 的排版显示引擎和跨平台的 GUI 应用程序开发框架,基于
    Node.js / OpenGL。目标是在此基础上开发 GUI 应用程序可拥有开发 WEB 应用般简单与速度同时兼顾 Native 应用程序的性能与体验。

    7 引用 • 9 回帖 • 404 关注
  • Gitea

    Gitea 是一个开源社区驱动的轻量级代码托管解决方案,后端采用 Go 编写,采用 MIT 许可证。

    5 引用 • 16 回帖 • 1 关注
  • Word
    13 引用 • 41 回帖
  • Scala

    Scala 是一门多范式的编程语言,集成面向对象编程和函数式编程的各种特性。

    13 引用 • 11 回帖 • 156 关注
  • SEO

    发布对别人有帮助的原创内容是最好的 SEO 方式。

    36 引用 • 200 回帖 • 33 关注
  • OkHttp

    OkHttp 是一款 HTTP & HTTP/2 客户端库,专为 Android 和 Java 应用打造。

    16 引用 • 6 回帖 • 93 关注
  • 深度学习

    深度学习(Deep Learning)是机器学习的分支,是一种试图使用包含复杂结构或由多重非线性变换构成的多个处理层对数据进行高层抽象的算法。

    43 引用 • 44 回帖
  • 服务器

    服务器,也称伺服器,是提供计算服务的设备。由于服务器需要响应服务请求,并进行处理,因此一般来说服务器应具备承担服务并且保障服务的能力。

    125 引用 • 585 回帖
  • 单点登录

    单点登录(Single Sign On)是目前比较流行的企业业务整合的解决方案之一。SSO 的定义是在多个应用系统中,用户只需要登录一次就可以访问所有相互信任的应用系统。

    9 引用 • 25 回帖 • 2 关注
  • 浅吟主题

    Jeffrey Chen 制作的思源笔记主题,项目仓库:https://github.com/TCOTC/Whisper

    1 引用 • 28 回帖
  • PWL

    组织简介

    用爱发电 (Programming With Love) 是一个以开源精神为核心的民间开源爱好者技术组织,“用爱发电”象征开源与贡献精神,加入组织,代表你将遵守组织的“个人开源爱好者”的各项条款。申请加入:用爱发电组织邀请帖
    用爱发电组织官网:https://programmingwithlove.stackoverflow.wiki/

    用爱发电组织的核心驱动力:

    • 遵守开源守则,体现开源&贡献精神:以分享为目的,拒绝非法牟利。
    • 自我保护:使用适当的 License 保护自己的原创作品。
    • 尊重他人:不以各种理由、各种漏洞进行未经允许的抄袭、散播、洩露;以礼相待,尊重所有对社区做出贡献的开发者;通过他人的分享习得知识,要留下足迹,表示感谢。
    • 热爱编程、热爱学习:加入组织,热爱编程是首当其要的。我们欢迎热爱讨论、分享、提问的朋友,也同样欢迎默默成就的朋友。
    • 倾听:正确并恳切对待、处理问题与建议,及时修复开源项目的 Bug ,及时与反馈者沟通。不抬杠、不无视、不辱骂。
    • 平视:不诋毁、轻视、嘲讽其他开发者,主动提出建议、施以帮助,以和谐为本。只要他人肯努力,你也可能会被昔日小看的人所超越,所以请保持谦虚。
    • 乐观且活跃:你的努力决定了你的高度。不要放弃,多年后回头俯瞰,才会发现自己已经成就往日所仰望的水平。积极地将项目开源,帮助他人学习、改进,自己也会获得相应的提升、成就与成就感。
    1 引用 • 487 回帖 • 2 关注
  • 996
    13 引用 • 200 回帖 • 5 关注
  • 以太坊

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

    34 引用 • 367 回帖 • 3 关注
  • AngularJS

    AngularJS 诞生于 2009 年,由 Misko Hevery 等人创建,后为 Google 所收购。是一款优秀的前端 JS 框架,已经被用于 Google 的多款产品当中。AngularJS 有着诸多特性,最为核心的是:MVC、模块化、自动化双向数据绑定、语义化标签、依赖注入等。2.0 版本后已经改名为 Angular。

    12 引用 • 50 回帖 • 517 关注
  • WordPress

    WordPress 是一个使用 PHP 语言开发的博客平台,用户可以在支持 PHP 和 MySQL 数据库的服务器上架设自己的博客。也可以把 WordPress 当作一个内容管理系统(CMS)来使用。WordPress 是一个免费的开源项目,在 GNU 通用公共许可证(GPLv2)下授权发布。

    46 引用 • 114 回帖 • 167 关注
  • 国际化

    i18n(其来源是英文单词 internationalization 的首末字符 i 和 n,18 为中间的字符数)是“国际化”的简称。对程序来说,国际化是指在不修改代码的情况下,能根据不同语言及地区显示相应的界面。

    8 引用 • 26 回帖 • 1 关注