利用JSP实现动态页面,常常会写一个自定义标签来实现页面的组件化设计。一般是写一个类让它继承TagSupport类,然后将页面的逻辑封装在TagSupport中,然后在tld文件中添加自定义标签的属性及标签名称声明。 之后就可以在jsp页面使用自定义标签了,以上提到的自定义标签只是一个大概,在此不赘述了。
另外实现页面组件化设计还有另外一个途径,那就是通过tagplugin来实现。
package org.apache.jasper.compiler.tagplugin;
public interface TagPlugin {
void doTag(TagPluginContext ctxt);
}
package org.apache.jasper.compiler.tagplugin;
public interface TagPluginContext {
boolean isScriptless();
// 是否设置了自定义标签上的值
boolean isAttributeSpecified(String attribute);
// 以该自定义标签为上线文,取得该自定义标签所对应的临时变量名
String getTemporaryVariableName();
// 在JSP生成java文件的过程中会在import区块中导入import的内容
void generateImport(String s);
//
void generateDeclaration(String id, String text);
/**
* Generate Java source codes
*/
void generateJavaSource(String s);
//
boolean isConstantAttribute(String attribute);
// 取得自定义标签定义的属性值,isAttributeSpecified()方法为查询该属性值是否有值
String getConstantAttribute(String attribute);
// 由于属性值是可以写el的,所以在标签生成结果时候,会在生成java文件的过程中自动调用解析el语法的功能模块
void generateAttribute(String attribute);
// 生成自定义标签的body内容,注意:如果在doTag函数中中没有调用该方法的话,tagmanager会在标签解析呃呃最后自动调用标签的内容块
void generateBody();
void dontUseTagPlugin();
TagPluginContext getParentContext();
void setPluginAttribute(String attr, Object value);
Object getPluginAttribute(String attr);
}
以上这两个接口是包装在tomcat的jasper.jar这个jar包中的。
下面介绍如何在web应用中实现一个简单的tagplugin
首先,写一个类实现TagPlugin 接口:
public class NameTagPlugin implements TagPlugin {
private static int count = 1;
public void doTag(TagPluginContext ctx) {
System.out.println(ctx.getTemporaryVariableName());
ctx.generateDeclaration("cssDependants","private static final java.util.List css_dependants = new java.util.ArrayList(1); ");
if (ctx.isAttributeSpecified("name")) {
ctx.generateDeclaration("cssDependantsPath" + (count++),
"static{css_dependants.add(\""
+ ctx.getConstantAttribute("name") + "\");}");
}
System.out.println(ctx.getConstantAttribute("name"));
}
}
注意:以上这个类需要放在 $catalina_home$/common/classes这个目录下,因为tagplugin这个类是在jsp编译成java文件的时候调用的,当java文件被编译成class字节码之后就不会调用tagplugin了。
TagPluginContext类的generateDeclaration有两个参数,id和 content,当在一个pagecontext中重复调用generateDeclaration()这个方法如果前后两次调用id参数是一样的,那么在java文件中生成的declaration块就会显示第一次调用所输入content内容。
以上这段代码会使自定义标签编译成java文件的时候生成,如下内容(代码片段):
public final class nametagtest_jsp extends org.apache.jasper.runtime.HttpJspBase
implements org.apache.jasper.runtime.JspSourceDependent {
private static final java.util.List css_dependants = new java.util.ArrayList(1);
static{css_dependants.add("hello");}
static{css_dependants.add("code");}
private static java.util.List _jspx_dependants;
static {
_jspx_dependants = new java.util.ArrayList(1);
_jspx_dependants.add("/WEB-INF/koubeitagtest.tld");
}
再写一个自定义标签(这个尽量简单一些,写一个类似helloworld的就行了)
public class TestTag extends TagSupport {
private String name;
public int doStartTag() throws JspException {
try {
this.pageContext.getOut().write(name);
} catch (IOException e) {
throw new JspException(e);
}
return EVAL_BODY_INCLUDE;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
将自定义标签声明在/WEB-INF/koubeitagtest.tld,这个文件中。
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<taglib xmlns="http://java.sun.com/xml/ns/j2ee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" version="2.0" xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd">
<display-name>koubeitest</display-name>
<tlib-version>2.2.4</tlib-version>
<short-name>tt</short-name>
<uri>koubeitest-tags</uri>
<tag>
<name>name</name>
<tag-class>com.koubei.example.tag.TestTag</tag-class>
<body-content>JSP</body-content>
<attribute>
<name>name</name>
<required>false</required>
<rtexprvalue>false</rtexprvalue>
</attribute>
</tag>
</taglib>
创建tagPlugins.xml文件,注意文件名不能有错误,大小写不能错。存放在WEB-INF文件夹下,当时我因为文件名写成了tagplugin.xml造成plugin没有起作用。
<?xml version="1.0"?>
<tag-plugins>
<tag-plugin>
<tag-class>com.koubei.example.tag.TestTag</tag-class>
<plugin-class>com.koubei.example.tag.NameTagPlugin</plugin-class>
</tag-plugin>
</tag-plugins>
总结:
- 使用tagplugin和customtag都是为了在jsp页面中实现页面组件化设计的,但是他们两个各有侧重,tagplugin是在jsp编译成java文件的编译期使用的(tomcat中是使用JDTCompiler来编译的),而customTag是在jsp页面运行期执行的,tagplugin只会执行一次,而customtag会jsp页面运行的时候每次都执行的。
- 在试验过程中发现,tagplugin和customtag这两种技术的使用是互斥的,如果在tagPlugins.xml中为某个自定义标签配置了插件的话那么,改custometag的dostart和doend方法就都不会执行了,而且全要依赖在tagplugin的dotag方法中在jsp编译的时候预先静态写入到java文件中的内容来执行了(这点稍微有点不爽)
- 何时应该使用tagplugin这个技术,原先有的内容生成如果是在自定义标签输出的,但是这个内容又不是以来每个访问页面的用户身份而取得的就能用tagplugin技术,这样会大幅提升页面运行的效率。 唯一不足的是,使用tagplugin还是要写一个plugin所对因的customtag,说白了这个customtag就是一个傀儡。
分享到:
相关推荐
曹旭东所译的深入解析tomcat的完整版,各章节的内容在此不详述,但是强调一点,绝对是完整版本。方便开发人员详细了解tomcat的构成、原理以及更好地帮助于开发和项目部署的技巧。
Tomcat架构解析文档.zip
tomcat 架构解析和优化。pdf tomcat 架构解析和优化。pdftomcat 架构解析和优化。pdf
apache+tomcat域名泛解析
网上JMS的资料很少,自己好不容易收集的,一起分享,没分了不能免费了
Tomcat架构解析 基于Tomcat 8.5.x全面介绍了Tomcat的架构、各组件的实现方案以及使用方式,主要包括Tomcat的基础组件架构以及工作原理,Tomcat各组件的实现方案、使用方式以及详细配置说明,Tomcat与Web服务器集成...
本书只是从架构设计上,对Tomcat的各组件进行了概念性讲解,如果你想阅读Tomcat的源码,本书的内容会让你更容易了解Tomcat的组件结构、设计方案,更容易去由概要到具体的熟悉Tomcat各组件的实现。 其次,本书不局限...
Tomcat 的三个最重要的启动脚本: startup.bat catalina.bat setclasspath.bat 上一篇咱们分析了 startup.bat 脚本 这一篇咱们来分析 catalina.bat 脚本. 至于 setclasspath.bat 这个脚本, 相信看完这一篇, 就可以...
tomcat的源码解析资料,加上tomcat的pdf解析中文版的,tomcat的源码。 挺不错的资源,大家可以学习下。
以类图,剪短的文字讲解了tomcat的架构。真正做到会用tomcat和懂tomcat内部运行机制。
基于Tomcat的全面解析应用服务器架构 涵盖Tomcat所有组件的详细配置
Tomcat解析与性能优化,挺好的,分析Tomcat的文档
tomcat解析多个项目配置,并设置默认访问项目。
全书380页,压缩包里的是PDF格式文档, 带索引书签目录,刘光瑞(著) ,Tomcat架构解析。最近自己在看的书。
tomcat源码解析
Tomcat源码解析.pdf
JSP中常见的Tomcat报错错误解析JSP中常见的Tomcat报错错误解析JSP中常见的Tomcat报错错误解析JSP中常见的Tomcat报错错误解析
tomcat架构解析_PDF电子书下载 高清 带索引书签目录_刘光瑞(著) 人民邮电出版社
NULL 博文链接:https://liudeh-009.iteye.com/blog/1563948
对于学习了一段时间tomcat的同学很有帮助,帮助解析tomcat的架构。方便对tomcat的理解,以及更好的tomcat调优。