目录
Set标签类:这个我们就是拿到一个存储数据的地方:pageContext
Out标签类:这个我们主要就是拿到一个输出流:pageContext.getOut();
思维导图
一、标签语言的特点
形式是:
<开始标签 属性=“属性值”>标签体</结束标签>
分类有:控制标签、数据标签、UI标签
二、自定义标签的开发及其使用步骤
2.1 在了解了标签的一些基本结构以及特点后,我们进一步探索,
根据上面点击 if 标签 发现自定义标签是与tld文件有关的;
我们再次试验发现把c.tld文件中的tag元素删除后相关的标签就不能用了;
因此我们得到两条结论:
<!--
1、自定义标签是与tld文件相关的
2、标签库中的标签与tld中的tag元素有关,也就是跟tag元素对应的助手类有关。
-->
于是我们想到如果把这个tld文件复制一份,把需要改的地方改成自己的,那是不是也能使用?
如下:
我们就创一个demo来测试一下,可以看到它是存在的
我们在jsp里写一个
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ taglib uri="http://jsp.hmj.cn" prefix="z"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<!-- 1.自定义标签库是与tld文件相关的
2.标签库中的标签与tld中的tag元素有关,也就是跟tag元素有关
-->
<z:if test="true">true</z:if>
<z:if test="false">false</z:if>
<z:set value="老六" var="name"></z:set>
<z:out value="${name}"></z:out>
</body>
</html>
然后运行一下
我们看出,可正常使用
三、标签的生命周期
3.1 我们又了解了跟自定义标签相关的tld文件、tag元素,
接下来就来了解一下它的生命周期吧:
我们新建一个类,继承tag元素对应的助手类:BodyTagSupport
DemoTag1类:
package com.heminjie.tag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyTagSupport;
/**
* 助手类 必须继承bodytagSupport
* @author Administrator
*
*/
public class DemoTag1 extends BodyTagSupport{
@Override
public int doStartTag() throws JspException {
System.out.println("=======doStartTag==========");
return EVAL_BODY_INCLUDE;
}
@Override
public int doAfterBody() throws JspException {
System.out.println("=======doAfterBody==========");
return super.doAfterBody();
// return EVAL_BODY_AGAIN;
}
@Override
public int doEndTag() throws JspException {
System.out.println("==========doEndTag==========");
return super.doEndTag();
// return SKIP_PAGE;
}
}
经过多次运行试验我们得到一些结论:
1.有标签体的情况下,默认会调用助手类的doStartTag、doAfterBody、doEndTag方法
2.如果将doStartTag的返回值为skip——body,那么doafterbody就不会调用执行(路线1)
3.如果将dostartag的返回值改为EVAL_BODY_INCLUDE,那么doafterbody就会执行(路线2)
4.如果将doAfterBody的返回值改为EVAL_BODY_AGAIN,那么会一直调用doafterbody,进入循环(路线3)
四、实践自定义标签:if、set、out
了解了那么多理论知识,大多都是试验测试,不怎么好看运行结果,而接下来我们就来利用我们了解的理论知识来实践一下:
4.1自定义几个属于自己的标签吧~
4.2 if标签的的定义
我们需要先建一个if标签类,继承助手类:定义一个if属性test:boolean类型
package com.heminjie.tag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyTagSupport;
/**
* if标签;
* 分析
* 如果满足条件,就打印标签体-》dostarttag的返回值EVAL_BODY_INCLUDE
* 如果不满足条件,就不输出标签体->dostarttag的返回值skip_body
* 需要获取到是否满足条件的结果值,那么该标签就有一个属性,属性值是boolean
* @author Administrator
*
*/
public class IfTag extends BodyTagSupport{
private boolean test;
public boolean isTest() {
return test;
}
public void setTest(boolean test) {
this.test = test;
}
@Override
public int doStartTag() throws JspException {
// 如果满足条件,就打印标签体-》dostarttag的返回值EVAL_BODY_INCLUDE
// 如果不满足条件,就不输出标签体->dostarttag的返回值skip_body
return test ? EVAL_BODY_INCLUDE : SKIP_BODY;
}
}
我们写好这个标签类后就要去heminjie.tld文件里写好相关的配置,我们把其他的配置全部删除,
留下一个范本:
4.3 配置tld文件:<c:if test=""></c:if>
<?xml version="1.0" encoding="UTF-8" ?>
<taglib xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd"
version="2.0">
<description>JSTL 1.1 core library</description>
<display-name>JSTL core</display-name>
<tlib-version>1.1</tlib-version>
<short-name>z</short-name>
<uri>http://jsp.hmj.cn</uri>
<validator>
<description>
Provides core validation features for JSTL tags.
</description>
<validator-class>
org.apache.taglibs.standard.tlv.JstlCoreTLV
</validator-class>
</validator>
<tag>
<!-- 代表标签库的名字 -->
<name>demo1</name>
<!-- 该标签对应的助手类 -->
<tag-class>com.heminjie.tag.DemoTag1</tag-class>
<!-- 代表是一个JSP标签 -->
<body-content>JSP</body-content>
<!-- <attribute> -->
<!-- 该自定义jsp标签的属性名称 -->
<!-- <name>var</name> -->
<!-- 该属性是否必填 -->
<!-- <required>false</required> -->
<!-- 该属性值是否支持表达式 -->
<!-- <rtexprvalue>false</rtexprvalue> -->
<!-- </attribute> -->
</tag>
<tag>
<!-- 代表标签库的名字 -->
<name>if</name>
<!-- 该标签对应的助手类 -->
<tag-class>com.heminjie.tag.IfTag</tag-class>
<!-- 代表是一个JSP标签 -->
<body-content>JSP</body-content>
<attribute>
<name>test</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
<tag>
<!-- 代表标签库的名字 -->
<name>set</name>
<!-- 该标签对应的助手类 -->
<tag-class>com.heminjie.tag.SetTag</tag-class>
<!-- 代表是一个JSP标签 -->
<body-content>JSP</body-content>
<attribute>
<name>var</name>
<required>true</required>
<rtexprvalue>false</rtexprvalue>
</attribute>
<attribute>
<name>value</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
<tag>
<!-- 代表标签库的名字 -->
<name>out</name>
<!-- 该标签对应的助手类 -->
<tag-class>com.heminjie.tag.OutTag</tag-class>
<!-- 代表是一个JSP标签 -->
<body-content>JSP</body-content>
<attribute>
<name>value</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
</taglib>
我们使用测试一下:
<l:if test="true">true</l:if>
<l:if test="false">false</l:if>
结果
我们发现没有问题!
接着我们继续来试一下Set以及Out标签的定义:
Set标签类:这个我们就是拿到一个存储数据的地方:pageContext
package com.heminjie.tag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyTagSupport;
/**
* 数据标签、存储数据
* 作用:pagecontext,request,session,application,(servletContext)
*
* 要存储数据,以链值的方式进行储存,分析得来该标签有2个属性
* @author Administrator
*
*/
public class SetTag extends BodyTagSupport{
private String var;
private Object value;
public String getVar() {
return var;
}
public void setVar(String var) {
this.var = var;
}
public Object getValue() {
return value;
}
public void setValue(Object value) {
this.value = value;
}
@Override
public int doStartTag() throws JspException {
// 要存储数据,以键值得方式进行存储,分析得来标签有2个属性
pageContext.setAttribute(var, value);
return super.doStartTag();
}
}
我们继续定义标签类:
Out标签类:这个我们主要就是拿到一个输出流:pageContext.getOut();
package com.heminjie.tag;
import java.io.IOException;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.JspWriter;
import javax.servlet.jsp.tagext.BodyTagSupport;
/**
* servlet中后台代码前台输出内容 :out.print
* 将数据输出到前台,首先拿到输出流
* @author Administrator
*
*/
public class OutTag extends BodyTagSupport{
private Object value;
public Object getValue() {
return value;
}
public void setValue(Object value) {
this.value = value;
}
@Override
public int doStartTag() throws JspException {
JspWriter out = pageContext.getOut();
try {
out.print(value);
} catch (IOException e) {
e.printStackTrace();
}
return super.doStartTag();
}
}
我们写出来运行一下
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="http://jsp.hmj.cn" prefix="z"%> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Insert title here</title> </head> <body> <!-- 1.有标签体的情况下,默认会调用助手类的doStartTag、doAfterBody、doEndTag方法 2.如果将doStartTag的返回值为skip——body,那么doafterbody就不会调用执行(路线1) 3.如果将dostartag的返回值改为EVAL_BODY_INCLUDE,那么doafterbody就会执行(路线2) 4.如果将doAfterBody的返回值改为EVAL_BODY_AGAIN,那么会一直调用doafterbody,进入循环(路线3) --> <z:demo1>xx</z:demo1> <z:if test="true">true</z:if> <z:if test="false">false</z:if> <z:set var="name" value="hmj"></z:set> <z:out value="${name}"></z:out> </body> </html>
ok,我的分享就到这里了