一篇文章读懂Java Web的框架标签

开发 开发工具
在Java Web应用开发时,如果不是前后端分离进行数据交互的实现,一般都会通过JSP、FreeMarker、Velocity之类的技术进行页面的渲染。而在页面的渲染过程中,很多时候会使用到标签(taglib)这个技术。

在Java Web应用开发时,如果不是前后端分离进行数据交互的实现,一般都会通过JSP、FreeMarker、Velocity之类的技术进行页面的渲染。而在页面的渲染过程中,很多时候会使用到标签(taglib)这个技术。

比如为了控制页面一些显示逻辑,实现类似代码里if/else这种效果,就会使用到core标签里的内容,类似这样:

实现类似代码里if/else这种效果,就会使用到core标签里的内容

要实现一个容器数据的遍历,可以直接使用core标签的foreach。

要实现页面上数据的格式化,则可以直接使用format标签,进行对应数据的格式化展现。

甚至在一些MVC框架中,也都包含一些便捷的用于页面数据渲染的标签,可以直接使用。例如Spring中的form标签就可以直接进行数据的绑定。

那这些标签背后是如何工作的呢?

之前的文章里有写过JSP的工作原理,概括起来,就是会在执行时将JSP生成Servlet文件,然后再执行对应的service方法,进行请求的处理。其中涉及到使用标签的部分也会生成对应的执行逻辑。

而实际上,一个tag,对应的是一个Java类,根据规范实现相应的方法。由JSP生成的Servlet在执行标签的过程中,会直接调用标签对应类的指定方法,根据返回值,来进行页面上对应内容的输出。如果是继续则输出内容,如果是跳过内容就会被忽略。基本就是这样一个思路。

例如本文前面的 if 标签,对应生成的Servlet内容是这样的:

本文前面的 if 标签,对应生成的Servlet内容

再看 foreach 这个标签,生成的内容是这样的:

 foreach 这个标签

我们看到,foreach 被直接转换成了do while 循环。

其中,最核心的两个方法是:

  • doStartTag
  • doEndTag

doStartTag

该方法会在JSP类内,被具体的标签实现类调用。用于实例的标签开始,执行时pageContext和一系列属性被认为已经设置完成。

会返回EVAL_BODY_INCLUDE或者SKIP_BODY,从这两个结果的变量名称可以看出,如果如果tag期望继续处理body,就返回前者,否则不处理就返回后者。

doEndTag用于确认该标签执行后,页面是否要继续渲染。

整个taglib使用起来都比较便捷,直接在JSP中声明 prefix 和 uri ,相当于把这部分内容依赖添加了进来,然后直接使用标签进行属性的设置,对应命名空间下标签的使用等。

而这些标签的声明,是通过类似这样的形式,被保存在Jar文件或者WEB-INF这些地方。

标签的声明

具体的标签实现类,一般继承TagSupport,然后重写我们上面提到的doStartTag 和 doEndTag 方法。

所以,对于一般在JSP页面内有大堆的 <% %>这种所谓的 scriptlet,可以直接定义一个标签,然后把逻辑移动到重写方法内即可。

这些Tag声明的tld,一般会在应用部署后启动时进行扫描,然后添加到Map里。

Tag声明的tld

页面解析执行的时候,会判断对应声明的tld是否存在,没有就会停止页面执行。

页面解析执行的时候,会判断对应声明的tld是否存在

页面解析生成Servlet类之后,执行时会调用具体标签的属性设置,doStartTag这些方法,此时如果一些属性不存在,绑定不成功等这些具体的标签逻辑会被暴露出来。

比如我们在使用 Spring 标签时,经常使用其 form 标签进行参数绑定。为了试验,我们随便写一个Spring 的标签使用

  1. <form:input path="abc" id="test"/> 

此时,JSP生成的内容是这样的:

JSP生成的内容

然后页面渲染时,会真正的进行数据绑定,判断这些属性的合法性等等,这里由于是随便写的一行代码,无法绑定所有报错了。

数据绑定

所以,对于 Spring、Struts2 等等这些框架的标签,本质上执行也还是这些逻辑,如果页面在渲染的时候出现问题,了解清楚是在哪一步的时候出了问题,知道具体这些标签的工作原理。

例如下图 Spring 的 InputTag 继承关系,也没有脱离Servlet 的Tag这个框。

总结下,Taglib的工作原理,是在应用部署的时候,解析tld的声明。页面渲染的时候,解析如果引入的tld不存在,就会报错。tld合法之后执行 tag 的具体逻辑,根据返回值判断是否继续页面的渲染。

框架的标签也是如此。 

【本文为51CTO专栏作者“侯树成”的原创稿件,转载请通过作者微信公众号『Tomcat那些事儿』获取授权】

戳这里,看该作者更多好文

责任编辑:赵宁宁 来源: 51CTO专栏
相关推荐

2020-04-22 13:27:20

数据分析模块解决

2017-09-04 13:44:00

Java

2021-05-09 09:06:24

Python批处理命令

2018-04-09 16:35:10

数据库MySQLInnoDB

2019-09-24 14:19:12

PythonC语言文章

2015-10-22 14:32:44

微服务PaaS应用开发

2020-10-09 08:15:11

JsBridge

2019-05-21 14:52:57

2014-05-14 11:15:02

历史起源iOSAndroid

2021-07-01 10:01:16

JavaLinkedList集合

2023-05-08 08:21:15

JavaNIO编程

2017-12-20 10:08:07

数据库阿里巴巴分库分表技术

2023-05-12 08:19:12

Netty程序框架

2022-02-21 09:44:45

Git开源分布式

2019-04-17 15:16:00

Sparkshuffle算法

2021-06-30 00:20:12

Hangfire.NET平台

2021-04-09 08:40:51

网络保险网络安全网络风险

2017-09-05 08:52:37

Git程序员命令

2011-07-12 13:35:04

程序员

2020-10-23 07:56:04

Java中的IO流
点赞
收藏

51CTO技术栈公众号