JSP标签

一.特点:

a,组成:

1,开始标签

2,标签体

3,结束标签

组成形式:<开始标签 属性="属性值">标签体</结束标签>

b,分类

1,空标签: br、hr...

2,ui标签:input、table...

3,控制标签:if、foreach...

4数据标签:set标签、out标签...

二,自定义标签开发

概念: 在jsp开发时,当jsp内置标签以及JSTL标签库都满足不了需求时,可以根据自己的需求来自定义标签

意义:根据自己的需求来调整,动态的实现一些功能,从而提高开发效率


步骤:

1.创建一个标签助手类(继承BodyTagSupport),重写doStartTag、doAfterBodyTag、doEndTag。

2.创建标签库描述文件(tld),添加自定义标签的配置(tld文件必须保存到WEB-INF目录或其子目录)

3,通过taglib引入自定义标签库

标签库描述文件:

<?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>c</short-name>
  <uri>xyy</uri>

  <validator>
    <description>
        Provides core validation features for JSTL tags.
    </description>
    <validator-class>
        org.apache.taglibs.standard.tlv.JstlCoreTLV
    </validator-class>
  </validator>

  <tag>
  
    <name>if</name>
    <tag-class>com.xyy.xyy1</tag-class>
    <body-content>JSP</body-content>
    <attribute>
    <!-- 属性名 -->
        <name>text</name>
    <!-- 是否必须 -->
        <required>true</required>
    <!-- 是否支持EI表达式 -->
        <rtexprvalue>false</rtexprvalue>
    </attribute>
  </tag>

  

</taglib>

引入自定义标签库: 

 三,标签库生命周期

 

 

路线1:doStarTag-->SKIP_BODY-->doEndTag

路线2:doStartTag-->EVAL_BODY_INCLUDE-->doAfterBody-->EVAL PAGE-->doEndTag

线路3:doStartTag-->EVAL_BODY_INCLUDE-->doAfterBody-->EVAL_BODY_AGAIN-->doAfterBody...(循环)-->满足条件后doEndTag
 


返回值:

1.SKIP_BODY:跳过主体

2.EVAL_BODY_INCLUDE:计算标签主体内容并输出

3.EVAL_BODY_AGAIN:在计算主体一次

4.EVAL_PAGE:计算页面的后续部分

5.SKIP_PAGE:跳过页面的后续部分

 


package com.xyy;
 
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyTagSupport;
/**
	三种路线(根据生命周期图整理出来的)
  1.dostartTag...skipBody...doEndTag
  2.dostartTag...EVAL_BODY_INCLUDE...doAfterBody...EVAL_PAGE...doEndTag
  3.dostartTag...EVAL BODN INCLUDE...doAfterBody...EVAL_BODY_AGAIN...doAfterBody...EVAL_BODY_AGAIN...doAfterBody...EVAL_BODY_AGAIN...doEndTag
 */
 
public class Dome01 extends BodyTagSupport{
 
	//开始标签
	public int doStartTag() throws JspException {
		System.out.println("Dome1...doStartTag...进来了");
		return EVAL_BODY_INCLUDE;
	}
	
	//结束标签
	public int doEndTag() throws JspException {
		System.out.println("Dome1...doEndTag...进来了");
		return super.doEndTag();
	}
	
}
public class Dome02 extends BodyTagSupport{
 
	//开始标签
	public int doStartTag() throws JspException {
		System.out.println("Dome2...doStartTag...进来了");
		return EVAL_BODY_INCLUDE;
	}
	
	//标签体
	@Override
	public int doAfterBody() throws JspException {
		System.out.println("Dome2...doAfterBody...进来了");
		return EVAL_PAGE;
	}
	
	//结束标签
	public int doEndTag() throws JspException {
		System.out.println("Dome2...doEndTag...进来了");
		return super.doEndTag();
	}
	
}
public class Dome03 extends BodyTagSupport{
 
	//开始标签
	public int doStartTag() throws JspException {
		System.out.println("Dome3...doStartTag...进来了");
		return EVAL_BODY_INCLUDE;
	}
	
	//标签体
	@Override
	public int doAfterBody() throws JspException {
		System.out.println("Dome3...doAfterBody...进来了");
		return EVAL_BODY_AGAIN;
	}
	
	//结束标签
	public int doEndTag() throws JspException {
		System.out.println("Dome3...doEndTag...进来了");
		return super.doEndTag();
	}
	
}

四,案例实践

if标签:

助手类

package com.xyy;

import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyTagSupport;

public class xyysIF extends BodyTagSupport{
	 private boolean text;

	public boolean isText() {
		return text;
	}

	public void setText(boolean text) {
		this.text = text;
	}


	@Override
	public int doStartTag() throws JspException {
		// TODO Auto-generated method stub\
		return text==true? EVAL_BODY_INCLUDE:SKIP_BODY ;
	}
	
	
	@Override
	public int doEndTag() throws JspException {
		// TODO Auto-generated method stub
		return super.doEndTag();
	}
	
	
}

tid文件配置


 set,out标签:

助手类

package com.xyy;
/**
 * 第一条路线
 * set标签没有标签体,需要借助JSPWriter
 */
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyTagSupport;

public class xyysSET 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 {
		// TODO Auto-generated method stub
		//通过pageContext保存
		pageContext.setAttribute(var, value);
		return SKIP_BODY;
	}

	
	@Override
	public int doEndTag() throws JspException {
		// TODO Auto-generated method stub
		return super.doEndTag();
	}
}

 

package com.xyy;
/**
 * 自定义out标签
 * 成员变量: value 存放标签对应的值  必传,支持EL表达式	  
 * 注意:无标签体,要在页面上输出内容,需要借助一个类JSPWrite。
 * 执行doStartTag,将内容输出到网页上
 */
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.JspWriter;
import javax.servlet.jsp.tagext.BodyTagSupport;

public class xyysOUT extends BodyTagSupport{

	private Object var;
	
	public Object getVar() {
		return var;
	}
	
	public void setVar(Object var) {
		this.var = var;
	}
	
	

	@Override
	public int doStartTag() throws JspException {
		// TODO Auto-generated method stub
		JspWriter out=pageContext.getOut();
		try {
			out.print(var);
		} catch (Exception e) {
			e.printStackTrace();
		}
		return SKIP_BODY;
	}
	
	
	@Override
	public int doEndTag() throws JspException {
		// TODO Auto-generated method stub
		return super.doEndTag();
	}
}

tid文件配置

 


foreach标签

助手类

package com.xyy;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;

import javax.servlet.jsp.JspException;
/**
 * foreach标签,走第二,三条路线
 * 1,最少要两个参数
 * 2,一定有标签体(doAfterBody)
 * 3,有判断条件使doAfterBody改变返回值(EVAL_PAGE还是 EVAL_BOYY_AGAIN)
 */
import javax.servlet.jsp.tagext.BodyTagSupport;

public class xyysFOREACH extends BodyTagSupport{
	private String var;
	private List<Object> items=new ArrayList<Object>();
	
	public String getVar() {
		return var;
	}
	
	public void setVar(String var) {
		this.var = var;
	}
	
	public List<Object> getItems() {
		return items;
	}
	
	public void setItems(List<Object> items) {
		this.items = items;
	}
	
	

@Override
public int doStartTag() throws JspException {
	// TODO Auto-generated method stub
	//保存迭代器,供doAfterBody中使用
	Iterator< Object> it=items.iterator();
	pageContext.setAttribute("it", it);
	return EVAL_BODY_INCLUDE;
}

@Override
	public int doAfterBody() throws JspException {
		// TODO Auto-generated method stub
	Iterator<Object> it=(Iterator<Object>) pageContext.getAttribute("it");
	if(it.hasNext()) {//判断集合是否有数据
		//在页面上,需要通过var将集合数据拿出
		pageContext.setAttribute(var, it.next());
		//保存下移状态
		pageContext.setAttribute("it", it);
		//继续循环
		return EVAL_BODY_AGAIN;
	}else {
		//结束循环
		return EVAL_PAGE;
	}
	}
	
@Override
	public int doEndTag() throws JspException {
		// TODO Auto-generated method stub
		return super.doEndTag();
	}
}

tid文件配置

 


select标签

助手类

package com.xyy;
/**
 * 目的:将所学的自定义标签的知识点用于实际项目开发
 * 不管是if/set/out/foreach标签那是别人已经具备的功能,直接用别人的就行
 * 学习自定义标签理解其底层结构,就是弥补现成C标签没有的功能
 *  以前查询下拉框
 *		<select>
 *			<option value=''>-==请选择==m</option>
 *			<option value= '1'>xyy1</option>
 *			<option value='2'>xyy2</option>
 *			<option value='3'>xyy3</option>
 *		</ select>
 *		修改回显,在这里面有大量的c:foreach. c:if到断
 *		不足之处:代码过大。以及凡是涉及到下拉框以及复选框。粗类似的代码过多
 *	目前:
 *		<z:selectc/z: select>
 *		目的:通过上述标签能够实现上述代码相同的功能
 *	分析属性:
 * 		1.需要数据源属性items,用于遍历展示的
 *   users->List<User>->id=option>value;name=option>text
 * 		2.对象key属性textKey,用于对应option>value
 * 		3.对象value属性textval。用于对应option>text
 *		4.对象默认key属性headerTextKey,用于对应默认的option>value
 *		5.对象默认value属性headerTextval,用于对应默认的option>text
 *		6.对象回显值属性selectedVal,用于判断是否数据回显选中
 *	没有标签体又需要往页面输出内容(jspWriter)
 */
 
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.List;

import javax.servlet.jsp.JspException;
import javax.servlet.jsp.JspWriter;
import javax.servlet.jsp.tagext.BodyTagSupport;


public class xyysSELECT extends BodyTagSupport{

	private List<Object> items=new ArrayList<Object>();//用于遍历展示
	private String textKey;//用于对应option>value
	private String textVal;//用于对应option>text
	private String headerTextKey;//用于对应默认option>value
	private String headerTextVal;//用于对应默认option>text
	private String selectedVal;//用于判断数据回显选择
	
	//定义属性美化。拓展/操作标签
	private String cssStyle;//美化
	private String id;//绑定事件...操作标签
	private String className;//美化


	private String html() throws Exception {
		StringBuffer sb=new StringBuffer();//获取可变字符串
		sb.append("<select>");
		if(headerTextKey!=null&&"".equals(headerTextVal)) {
			sb.append("<option vlaue='"+headerTextKey+"'>"+headerTextVal+"</option>");
		}
		if(items.size()>0) {
			for(Object obj:items) {
				//obj就是页面传输过来集合的类型或对象
				//通过反射获取类对象属性 getDeclaredField()获取所有(公共的、私有的)类对象属性
				Field fidname=obj.getClass().getDeclaredField(textKey);
				//打开权限 
				fidname.setAccessible(true);
				//获取textKey 对应的属性值
				//String value = fidname.get(obj).toString();
				String value=fidname.get(obj).toString();
				Field fidtext = obj.getClass().getDeclaredField(textVal);
				fidtext.setAccessible(true);
				String text = fidtext.get(obj).toString();
				//追加至<optoin>中 判断默认选中
				if(text.equals(selectedVal)) {
				sb.append("<option selected vlaue='"+value+"'>"+text+"</option>");
				}else {
					sb.append("<option vlaue='"+value+"'>"+text+"</option>");
				}
	}
		}
		sb.append("<option></option>");
		sb.append("</select>");
		return sb.toString();

	}
	@Override
	public int doStartTag() throws JspException {
		// TODO Auto-generated method stub
		JspWriter out=pageContext.getOut();
		try {
			out.print(html());
		} catch (Exception e) {
			e.printStackTrace();
		}
		return super.doStartTag();
	}
	
	

	@Override
	public int doAfterBody() throws JspException {
		// TODO Auto-generated method stub
		return super.doAfterBody();
	}
	
	@Override
	public int doEndTag() throws JspException {
		// TODO Auto-generated method stub
		return super.doEndTag();
	}
}

tid文件配置

<tag>
  <name>select</name>
  <tag-class>com.xyy.xyysSELECT</tag-class>
  <body-content>JSP</body-content>
  <attribute>
  <!--  自定义标签的成员变量名称-->
  <name>id</name>
  <!-- 该成员变量是否必传-->
  <required>false</required>
  <!--  是否支持EL表达式-->
  <rtexprvalue>false</rtexprvalue>
  </attribute>
  <attribute>
  <!--  自定义标签的成员变量名称-->
  <name>cssStyle</name>
  <!-- 该成员变量是否必传-->
  <required>false</required>
  <!--  是否支持EL表达式-->
  <rtexprvalue>false</rtexprvalue>
  </attribute>
  <attribute>
  <!--  自定义标签的成员变量名称-->
  <name>className</name>
  <!-- 该成员变量是否必传-->
  <required>false</required>
  <!--  是否支持EL表达式-->
  <rtexprvalue>false</rtexprvalue>
  </attribute>
  
  <attribute>
  <!--  自定义标签的成员变量名称-->
  <name>items</name>
  <!-- 该成员变量是否必传-->
  <required>true</required>
  <!--  是否支持EL表达式-->
  <rtexprvalue>true</rtexprvalue>
  </attribute>
  <attribute>
  <!--  自定义标签的成员变量名称-->
  <name>textKey</name>
  <!-- 该成员变量是否必传-->
  <required>true</required>
  <!--  是否支持EL表达式-->
  <rtexprvalue>false</rtexprvalue>
  </attribute>
  <attribute>
  <!--  自定义标签的成员变量名称-->
  <name>textVal</name>
  <!-- 该成员变量是否必传-->
  <required>true</required>
  <!--  是否支持EL表达式-->
  <rtexprvalue>false</rtexprvalue>
  </attribute>
  <attribute>
  <!--  自定义标签的成员变量名称-->
  <name>headerTextKey</name>
  <!-- 该成员变量是否必传-->
  <required>false</required>
  <!--  是否支持EL表达式-->
  <rtexprvalue>false</rtexprvalue>
  </attribute>
  <attribute>
  <!--  自定义标签的成员变量名称-->
  <name>headerTextVal</name>
  <!-- 该成员变量是否必传-->
  <required>false</required>
  <!--  是否支持EL表达式-->
  <rtexprvalue>false</rtexprvalue>
  </attribute>
  <attribute>
  <!--  自定义标签的成员变量名称-->
  <name>selectedVal</name>
  <!-- 该成员变量是否必传-->
  <required>false</required>
  <!--  是否支持EL表达式-->
  <rtexprvalue>true</rtexprvalue>
  </attribute>
  </tag>

测试界面:

效果: 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值