1.认识JSP自定义标签
1.1形式
如下:
<开始标签 属性="属性值">标签体</结束标签>
1.2分类
①空标签 例如:br,hr
②UI标签 例如:input,table(没有标签体也能在网页中有输出类容的标签)
③控制标签 例如:if,foreach
④数据标签 例如:out标签
2.自定义标签的开发及使用步骤
2.1认识.tld文件
通过点击c:if标签,可以链接跳到c.tid文件中,而.tid文件就是C标签库的定义配置文件。
所以我们要想自定义标签,首先我们需要自定义该标签的定义配置文件.tld。
2.2自定义.tld文件方法
2.2.1在web-inf安全目录下新建一个File并将其命名
2.2.2修改uri标签
修改为你即将要加载的uri,例如:http://www/Kissship/com
2.2.3JSP通过taglib指令导入标签库
在这时候我们导入标签库即可找到uri为:
<%@ taglib uri="http://www/Kissship/com" prefix="k" %>
这样我们就可以使用自己定义的JSP标签库了。
例如这是我们执行<z:if test="true">true</z:if>如下:
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="http://www/Kissship/com" prefix="k" %> <!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=ISO-8859-1"> <title>Insert title here</title> </head> <body> <z:if test="true">true</z:if> </body> </html>
结果如下:
但我们要实现自定义标签还需要创建助手类。
3.创建助手类
3.1首先我们先在tag包下创建一个助手类
必须继承bodytagSupport。
实现三个方法:这三个方法分别对应着:开始标签、标签体、结束标签,即执行到对应的部分时调用对应的方法。
代码如下:
package com.Kissship.tag; import javax.servlet.jsp.JspException; import javax.servlet.jsp.tagext.BodyTagSupport; /** * 助手类 * 1.必须继承bodytagSupport * @author jj * */ public class DemoTag1 extends BodyTagSupport{ //开始标签 @Override public int doStartTag() throws JspException { System.out.println("=========doStartTag========="); /*return super.doStartTag();*/ //SKIP_BODY跳过 //return SKIP_BODY; //EVAL_BODY_INCLUDE 相当于调用默认方法 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; } }
3.2修改tag标签
我们使用的标准标签例如 if、foreach……都和.tdl文件中的<tag>标签有关。
代码如下:
4.标签生命周期
SKIP_BODY:跳过主体
EVAL_BODY_INCLUDE:计算标签主体内容并(输出)
EVAL_BODY_AGAIN:在计算主体一次
EVAL_PAGE: 计算页面的后续部分
SKIP_PAGE:跳过页面后续部分
代码如下:
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="http://www/Kissship/com" 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就不会调用执行(路线一) 3.如果将dostarttag的返回值改为eval_body_include,那么doafterbody就会调用执行(路线二) 4.如果将doafterbody的返回值改为eval_body_again,那么会一直调用doafterbody,进入循环(路线三) --> <z:demo1>NiKo的印花集</z:demo1> </body> </html>
控制台输出结果:
5.JSP标签实践
5.1z:if标签
助手类:
package com.Kissship.tag; import javax.servlet.jsp.JspException; import javax.servlet.jsp.tagext.BodyTagSupport; /** * if标签: * 分析: * 如果满足条件,就打印标签体->dpstarttag的返回值eval_body_incude * 如果不满足条件,就不输出标签体->dostarttag的返回值skip_body * 需要获取到是否满足条件的结果值,那么该标签就有一个属性,属性值是boolean * @author jj * */ 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 { // 如果满足条件,就打印标签体->dpstarttag的返回值eval_body_incude // 如果不满足条件,就不输出标签体->dostarttag的返回值skip_body return test ? EVAL_BODY_INCLUDE : SKIP_BODY; } }
tld文件:
<tag> <!-- 代表标签库标签的名字 --> <name>if</name> <!-- 该标签对应的助手类的全路径名 --> <tag-class>com.Kissship.tag.IfTag</tag-class> <!-- 代表是一个JSP标签 --> <body-content>JSP</body-content> <attribute> <!-- 该自定义JSP标签的属性名称 --> <name>test</name> <!-- 该属性是否必填 --> <required>true</required> <!-- 该属性值是否支持表达式 --> <rtexprvalue>true</rtexprvalue> </attribute> </tag>
输出结果如下:
5.2z:set标签
助手类:
package com.Kissship.tag; import javax.servlet.jsp.JspException; import javax.servlet.jsp.tagext.BodyTagSupport; /** * 数据标签:存储数据的 * 作用:pagecontext,request,session,application,(servletContext) * * 要存储数据,以键值对的方式进行存储,分析得来该标签有两个属性 * @author jj * */ 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 { pageContext.setAttribute(var, value); return super.doStartTag(); } }
tld文件:
<tag> <!-- 代表标签库标签的名字 --> <name>set</name> <!-- 该标签对应的助手类的全路径名 --> <tag-class>com.Kissship.tag.SetTag</tag-class> <!-- 代表是一个JSP标签 --> <body-content>JSP</body-content> <attribute> <!-- 该自定义JSP标签的属性名称 --> <name>var</name> <!-- 该属性是否必填 --> <required>true</required> <!-- 该属性值是否支持表达式 --> <rtexprvalue>false</rtexprvalue> </attribute> <attribute> <!-- 该自定义JSP标签的属性名称 --> <name>value</name> <!-- 该属性是否必填 --> <required>true</required> <!-- 该属性值是否支持表达式 --> <rtexprvalue>true</rtexprvalue> </attribute> </tag>
demo3:
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="http://www/Kissship/com" 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就不会调用执行(路线一) 3.如果将dostarttag的返回值改为eval_body_include,那么doafterbody就会调用执行(路线二) 4.如果将doafterbody的返回值改为eval_body_again,那么会一直调用doafterbody,进入循环(路线三) --> <%-- <z:demo1>NiKo的印花集</z:demo1> --%> <%-- <z:if test="true">true</z:if> <z:if test="false">false</z:if> --%> <z:set var="name" value="弗拉米基尔"></z:set> </body> </html>
5.3z:out标签
助手类:
package com.Kissship.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 jj * */ 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) { // TODO Auto-generated catch block e.printStackTrace(); } return super.doStartTag(); } }
tld文件:
<tag> <!-- 代表标签库标签的名字 --> <name>out</name> <!-- 该标签对应的助手类的全路径名 --> <tag-class>com.Kissship.tag.OutTag</tag-class> <!-- 代表是一个JSP标签 --> <body-content>JSP</body-content> <attribute> <!-- 该自定义JSP标签的属性名称 --> <name>value</name> <!-- 该属性是否必填 --> <required>true</required> <!-- 该属性值是否支持表达式 --> <rtexprvalue>true</rtexprvalue> </attribute> </tag>
demo3:
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="http://www/Kissship/com" 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就不会调用执行(路线一) 3.如果将dostarttag的返回值改为eval_body_include,那么doafterbody就会调用执行(路线二) 4.如果将doafterbody的返回值改为eval_body_again,那么会一直调用doafterbody,进入循环(路线三) --> <%-- <z:demo1>NiKo的印花集</z:demo1> --%> <z:if test="true">true</z:if> <z:if test="false">false</z:if> <z:set var="name" value="弗拉米基尔"></z:set> <z:out value="${name} "></z:out> </body> </html>
输出结果如下:
总结:创建自定义标签需要编写标签处理类和标签描述文件,并将其部署到特定位置,然后可以在JSP页面中使用导入的标签库来调用自定义标签。自定义标签的创建是一个比较复杂的过程,但可以提高代码的重用性和可维护性,使得Web开发更加高效。
最后J自定义JSP标签01就到这里,祝大家在敲代码的路上一路通畅!