掌握聚合最新动态了解行业最新趋势
API接口,开发服务,免费咨询服务

想写框架的看过来

天才程序员去创造工具,大众程序员去使用工具, 我想大部分人都处于“使用” 这个层次,  我就是其中之一。

那些创造了语言、系统、工具、框架的程序员是我们崇拜的对象,也是我们努力奋斗的目标, 希望有一天我们也能像他们那样, 攀上代码的高峰, 会当凌绝顶, 一览众山小。。。。。。

创造是一件很难的事情,但是模仿却简单的多, 今天聊一点框架相关的知识,那些想自己写框架的,想重复造轮子的同学可以看看。 

别误会,我说的是重复造轮子,不是重复发明轮子。 重复造轮子能让你体会一下创造者的艰辛和快乐,提升一下自己的内功和境界。

所谓的框架,和工具不同, 工具拿过来直接可用, 而框架只是个半成品。 换句话说就是大牛写的框架代码,加上我们写的代码,才能形成一个系统,才能运行起来。

你想一想,单独的Struts, Spring ,Hibernate能运行吗? 肯定不行吧。

框架是个半成品(但不是残次品),  我们要想用起来,非得自己填上代码不可。 

这就有点难搞了,那些大牛们写框架的时候得考虑用户怎么配合才行, 所以得定义好边界, 哪些是框架的任务,哪些是程序员应该做的扩展。 

我们想修改框架的东西,门儿都没有,对修改完全封闭。 但是框架又提供了一些地方让程序员扩展, 这就是著名的面向对象设计的原则之一: Open-Close原则

1.模板方法


想达成这个目标,最容易的方式就是“继承”, 在Java世界常用的就是抽象类

抽象类有这样的好处:

  • 1. 没法实例化去执行(符合框架半成品的特点) 
  • 2. 可以被继承, 抽象方法需要使用者去实现。
  • 3. 可以在抽象类中写代码,写框架相关的代码, 还可以调用抽象方法。

完美达到了Open-Close原则, 举个例子会看得更加清楚一些。

假设你要写一个MVC的框架, 你希望定义一个抽象类AbstractAction,把一些框架应该做的事封装起来,比如抽取HttpRequest的参数,形成一个Map 以便传递给具体的Action使用,这些操作和步骤是子类不能修改的-- Close


虽然是封闭的, 但是框架的使用者却可以去继承这类,实现那个抽象的execute 方法, 放置自己的业务代码。  ---Open

当然它除了实现自己的代码之外,没法修改AbstractAction中的任何东西, 或者说,对超类一无所知。 -- 又是Close


这样一来,你的框架就得到了这个效果: Don't call me , I'll call you  (别打电话找我,我会去找你的)(码农翻身注: 这就是Struts1 背后的思想)

2.反射+声明+约定


上面的方法看起来很美, 但是有个巨大的问题,就是对客户侵入性太强。 表现在两个方面:

  • (1) 子类(业务类)很不自由, 被迫继承一个它不想继承的类
  • (2)  难于做单元测试,你也许注意到了, AbstractAction 其实是依赖HttpRequest, 和HttpResponse的,这两个对象一般需要web 容器跑起来才能得到。

很容易想到, 如果业务类是一个普通的POJO多好:


如果我不说,估计你会以为这是一个普通的java class ,  实际上,它就是一个普通java class。

它虽然叫做LoginAction , 但是和框架一毛钱关系也没有, 毕竟它啥也没有继承, 你的框架怎么知道它就是一个潜在的Action 类?

这时候需要一点点声明性的工作, 告诉框架,这是一个Action 类, 声明可以用xml ,也可以用注解,例如用xml声明的话,使用框架的程序员需要这么做


这时候框架通过解析XML就会知道: 奥,这是一个Action类,让我把它创建起来吧!

怎么创建?  没看到配置文件中的类名吗, java 反射出马,轻松搞定。 

接下来的问题是:怎么把Http Reqeust 中的参数怎么传递给这个类?  

还是用反射! 看到LoginAction 的setUsername 和 setPassword没有, 框架和使用者约定好这么办:如果从浏览器传递到服务器的的参数是username=liuxin&password=123456, 那你框架应该做的事情就是调用setUsername("liuxin") , setPassword("123456") 

框架和使用者还可以再增加一个约定: 调用了setXXX方法以后,就去调用exectue 方法,这是个真正的业务方法。

如果觉得这样的约定太强, 还可以采用声明的方式:


(码农翻身注:这其实就是strut2背后的思想)

如果你讨厌繁琐的xml,   现在Java提供了另外一种强大的武器: Java 注解

Java注解能够让你在源代码中添加一些“元数据”, 为一个类或者方法的增加一些“额外”的信息。


作为框架开发者, 你需要给使用者提供上图中用到的几个注解

  • @Action 用来表明这是一个Action 类
  • @RequestMapping 表明如果用户通过浏览器访问/login , 那这个excute 方法就会被执行。

还要加上这么一个约定 :  如果用户访问的是/login?username=liuxin&password=123456的话那么execute 方法的 username和password 两个参数将会绑定到值“liuxin" 和 "123456"

有了这些注解和约定,你的框架就能找到程序员写的Action 类,并且直到如何调用方法, 如何传递参数了。

(码农翻身注:这其实就是spring mvc背后的思想)

本文已获得作者授权,如需转载请与作者联系

你看到的只是冰山一角, 更多精彩文章,尽在“码农翻身” 微信公众号, 回复消息"m"或"目录" 查看更多文章

有心得想和大家分享? 欢迎投稿 ! 我的联系方式:微信:liuxinlehan  QQ: 3340792577

我是一个线程

我是一个Java class

Javascript: 一个屌丝的逆袭

Java : 一个帝国的诞生

我是一个网卡

我是一个路由器

我是一个进程

我是一块硬盘

TCP/IP 之 大明王朝的邮差

CPU 阿甘

CPU 阿甘之烦恼

Basic : 一个老兵的自述

小王的架构师之路

编程的自学方法

程序员在工作中必备的能力

码农需要知道的潜规则

IE为什么把Chrome和火狐打伤了

Node.js :我只需要一个店小二

假如我是计算机系老师

假如时光倒流,我会这么学Java

学会编程,而不是学会Java

15年编程生涯,资深架构师总结的7条经验


公众号:码农翻身“码农翻身”公众号由工作15年的前IBM架构师创建,分享编程和职场的经验教训。

声明:所有来源为“聚合数据”的内容信息,未经本网许可,不得转载!如对内容有异议或投诉,请与我们联系。邮箱:marketing@think-land.com

  • 全球天气预报

    支持全球约2.4万个城市地区天气查询,如:天气实况、逐日天气预报、24小时历史天气等

    支持全球约2.4万个城市地区天气查询,如:天气实况、逐日天气预报、24小时历史天气等

  • 购物小票识别

    支持识别各类商场、超市及药店的购物小票,包括店名、单号、总金额、消费时间、明细商品名称、单价、数量、金额等信息,可用于商品售卖信息统计、购物中心用户积分兑换及企业内部报销等场景

    支持识别各类商场、超市及药店的购物小票,包括店名、单号、总金额、消费时间、明细商品名称、单价、数量、金额等信息,可用于商品售卖信息统计、购物中心用户积分兑换及企业内部报销等场景

  • 涉农贷款地址识别

    涉农贷款地址识别,支持对私和对公两种方式。输入地址的行政区划越完整,识别准确度越高。

    涉农贷款地址识别,支持对私和对公两种方式。输入地址的行政区划越完整,识别准确度越高。

  • 人脸四要素

    根据给定的手机号、姓名、身份证、人像图片核验是否一致

    根据给定的手机号、姓名、身份证、人像图片核验是否一致

  • 个人/企业涉诉查询

    通过企业关键词查询企业涉讼详情,如裁判文书、开庭公告、执行公告、失信公告、案件流程等等。

    通过企业关键词查询企业涉讼详情,如裁判文书、开庭公告、执行公告、失信公告、案件流程等等。

0512-88869195
数 据 驱 动 未 来
Data Drives The Future