J2EE基础--自定义JSP标签

目录

思维导图

​编辑

一、标签语言的特点

二、自定义标签的开发及其使用步骤

三、标签的生命周期

接下来就来了解一下它的生命周期吧:

经过多次运行试验我们得到一些结论:

四、实践自定义标签:if、set、out

4.3 配置tld文件:

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,我的分享就到这里了

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值