J2EE_05 快速入门 JSP自定义标签

目录

一、JSP 自定义标签

创建"Hello"标签

访问标签体

自定义标签属性

二、详细介绍

1. 标签语言特点

2. 自定义标签的开发及使用步骤(浏览器使用:google/firefox)

三、案例示例+实践应用:

     ①  package com.web.entity包下:

        ②package com.web.tag包下:


一、JSP 自定义标签

        自定义标签是用户定义的JSP语言元素。当JSP页面包含一个自定义标签时将被转化为servlet,标签转化为对被 称为tag handler的对象的操作,即当servlet执行时Web container调用那些操作。

        JSP标签扩展可以让你创建新的标签并且可以直接插入到一个JSP页面。 JSP 2.0规范中引入Simple Tag Handlers来编写这些自定义标记。

        你可以继承SimpleTagSupport类并重写的doTag()方法来开发一个最简单的自定义标签。


创建"Hello"标签

        接下来,我们想创建一个自定义标签叫作<ex:Hello>,标签格式为:

<ex:Hello />

        要创建自定义的JSP标签,你首先必须创建处理标签的Java类。所以,让我们创建一个HelloTag类,如下所示:

package com.runoob;

import javax.servlet.jsp.tagext.*;
import javax.servlet.jsp.*;
import java.io.*;

public class HelloTag extends SimpleTagSupport {

  public void doTag() throws JspException, IOException {
    JspWriter out = getJspContext().getOut();
    out.println("Hello Custom Tag!");
  }
}

        以下代码重写了doTag()方法,方法中使用了getJspContext()方法来获取当前的JspContext对象,并将"Hello Custom Tag!"传递给JspWriter对象。

        编译以上类,并将其复制到环境变量CLASSPATH目录中。最后创建如下标签库:<Tomcat安装目录>webapps\ROOT\WEB-INF\custom.tld。

<taglib>
  <tlib-version>1.0</tlib-version>
  <jsp-version>2.0</jsp-version>
  <short-name>Example TLD</short-name>
  <tag>
    <name>Hello</name>
    <tag-class>com.runoob.HelloTag</tag-class>
    <body-content>empty</body-content>
  </tag>
</taglib>

接下来,我们就可以在JSP文件中使用Hello标签:

<%@ taglib prefix="ex" uri="WEB-INF/custom.tld"%>
<html>
  <head>
    <title>A sample custom tag</title>
  </head>
  <body>
    <ex:Hello/>
  </body>
</html>

以上程序输出结果为:

Hello Custom Tag!

访问标签体

你可以像标准标签库一样在标签中包含消息内容。如我们要在我们自定义的Hello中包含内容,格式如下:

<ex:Hello>
   This is message body
</ex:Hello>

我们可以修改标签处理类文件,代码如下:

package com.runoob;

import javax.servlet.jsp.tagext.*;
import javax.servlet.jsp.*;
import java.io.*;

public class HelloTag extends SimpleTagSupport {

   StringWriter sw = new StringWriter();
   public void doTag()
      throws JspException, IOException
    {
       getJspBody().invoke(sw);
       getJspContext().getOut().println(sw.toString());
    }

}

接下来我们需要修改TLD文件,如下所示:

<taglib>
  <tlib-version>1.0</tlib-version>
  <jsp-version>2.0</jsp-version>
  <short-name>Example TLD with Body</short-name>
  <tag>
    <name>Hello</name>
    <tag-class>com.runoob.HelloTag</tag-class>
    <body-content>scriptless</body-content>
  </tag>
</taglib>

现在我们可以在JSP使用修改后的标签,如下所示:

<%@ taglib prefix="ex" uri="WEB-INF/custom.tld"%>
<html>
  <head>
    <title>A sample custom tag</title>
  </head>
  <body>
    <ex:Hello>
        This is message body
    </ex:Hello>
  </body>
</html>

以上程序输出结果如下所示:

This is message body

自定义标签属性

你可以在自定义标准中设置各种属性,要接收属性,值自定义标签类必须实现setter方法, JavaBean 中的setter方法如下所示:

package com.runoob;

import javax.servlet.jsp.tagext.*;
import javax.servlet.jsp.*;
import java.io.*;

public class HelloTag extends SimpleTagSupport {

   private String message;

   public void setMessage(String msg) {
      this.message = msg;
   }

   StringWriter sw = new StringWriter();

   public void doTag()
      throws JspException, IOException
    {
       if (message != null) {
          /* 从属性中使用消息 */
          JspWriter out = getJspContext().getOut();
          out.println( message );
       }
       else {
          /* 从内容体中使用消息 */
          getJspBody().invoke(sw);
          getJspContext().getOut().println(sw.toString());
       }
   }

}

属性的名称是"message",所以setter方法​​是的setMessage()。现在让我们在TLD文件中使用的<attribute>元素添加此属性:

<taglib>
  <tlib-version>1.0</tlib-version>
  <jsp-version>2.0</jsp-version>
  <short-name>Example TLD with Body</short-name>
  <tag>
    <name>Hello</name>
    <tag-class>com.runoob.HelloTag</tag-class>
    <body-content>scriptless</body-content>
    <attribute>
       <name>message</name>
    </attribute>
  </tag>
</taglib>

现在我们就可以在JSP文件中使用message属性了,如下所示:

<%@ taglib prefix="ex" uri="WEB-INF/custom.tld"%>
<html>
  <head>
    <title>A sample custom tag</title>
  </head>
  <body>
    <ex:Hello message="This is custom tag" />
  </body>
</html>

以上实例数据输出结果为:

This is custom tag

你还可以包含以下属性:

属性描述
name定义属性的名称。每个标签的是属性名称必须是唯一的。
required指定属性是否是必须的或者可选的,如果设置为false为可选。
rtexprvalue声明在运行表达式时,标签属性是否有效。
type定义该属性的Java类类型 。默认指定为 String
description描述信息
fragment如果声明了该属性,属性值将被视为一个 JspFragment

以下是指定相关的属性实例:

.....
    <attribute>
      <name>attribute_name</name>
      <required>false</required>
      <type>java.util.Date</type>
      <fragment>false</fragment>
    </attribute>
.....

如果你使用了两个属性,修改TLD文件,如下所示:

.....
    <attribute>
      <name>attribute_name1</name>
      <required>false</required>
      <type>java.util.Boolean</type>
      <fragment>false</fragment>
    </attribute>
    <attribute>
      <name>attribute_name2</name>
      <required>true</required>
      <type>java.util.Date</type>
    </attribute>
.....

二、详细介绍

1. 标签语言特点


           <开始标签 属性="属性值">标签体</结束标签>

   空标签
           <br/><hr/>
           <开始标签></结束标签>
           <开始标签/>

2. 自定义标签的开发及使用步骤(浏览器使用:google/firefox)

        2.1 创建一个标签助手类(继承BodyTagSupport)
      标签属性必须与助手类的属性对应、且要提供对应get/set方法
      rtexprvalue

        2.2 创建标签库描述文件(tld),添加自定义标签的配置

        注:tld文件必须保存到WEB-INF目录或其子目录
      jstl标签库

        tld文件示例:

<!DOCTYPE taglib
  PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.2//EN"
   "http://java.sun.com/dtd/web-jsptaglibrary_1_2.dtd">
<!-- 标签库描述符 -->
<taglib xmlns="http://java.sun.com/JSP/TagLibraryDescriptor">
	<!-- 代表标签库的版本号 -->
	<tlib-version>1.0</tlib-version>
	<!-- 代表jsp的版本 -->
	<jsp-version>1.2</jsp-version>
	<!-- 你的标签库的简称 -->
	<short-name>test</short-name>
	<!-- 你标签库的引用uri -->
	<uri>/zking</uri>

	<tag>
		<!-- 标签名 -->
		<name>yy</name>
		<!-- 标签工具类 -->
		<tag-class>com.zking.TestTag</tag-class>
		<!-- 标签的内容类型:empty表示空标签,jsp表示可以为任何合法的JSP元素 -->
		<body-content>jsp</body-content>
		<!-- 自定义标签的属性定义,请注意一定要在标签类中提供对应的get/set方法 -->
		<attribute>
			<!-- 自定义标签的属性名称 -->
			<name>name</name>
			<!-- true表示必填 -->
			<required>true</required>
			<!-- true支持动态值,可以向值里面填jsp表达式、EL表达式,false则不支持 -->
			<rtexprvalue>false</rtexprvalue>
		</attribute>
	</tag>
</taglib>

  2.3 在JSP通过taglib指令导入标签库,并通过指定后缀访问自定义标签

3. 标签生命周期
  
 

 流程A:
                                      SKIP_BODY
          3.1  实例化标签助手类->doStartTag()------------->doEndTag()
       //主要用开发简单标签

  流程B:
                                      EVAL_BODY_INCLUDE         SKIP_BODY
          3.2  实例化标签助手类->doStartTag()------------->doAfterBody---------------->doEndTag()...
                                                              EVAL_BODY_AGAIN

  •           SKIP_BODY:跳过主体
  •           EVAL_BODY_INCLUDE:计算标签主体内容并[输出]
  •           EVAL_PAGE:   计算页面的后续部分
  •           SKIP_PAGE:    跳过页面的后续部分
  •           EVAL_BODY_AGAIN:   再计算主体一次

三、案例示例+实践应用:

        要求:自定义test标签、out输出标签、if条件标签、forEach循环标签、自定义deptList数据标签,select标签

        

 

         我们首先创建一个web项目,然后创建如上所示的包与文件,接着我们再导入两个相关的jstl的jar包,我们一步一步来,jar包的导入过程我就跳过了。

        接下来我们在WEB-INF文件夹下创建mytag.tld的文件,我们来具体的配置想要自定义的标签

<!DOCTYPE taglib
  PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.2//EN"
   "http://java.sun.com/dtd/web-jsptaglibrary_1_2.dtd">
<!-- 标签库描述符 -->
<taglib xmlns="http://java.sun.com/JSP/TagLibraryDescriptor">
	<!-- 代表标签库的版本号 -->
	<tlib-version>1.0</tlib-version>
	<!-- 代表jsp的版本 -->
	<jsp-version>1.2</jsp-version>
	<!-- 你的标签库的简称 -->
	<short-name>y</short-name>
	<!-- 你标签库的引用uri -->
	<uri>/yutianfu</uri>
	
	<!-- 自定义name标签 -->
	<tag>
		<!-- 标签名 -->
		<name>test</name>
		<!-- 标签工具类 -->
		<tag-class>com.web.tag.TestTag</tag-class>
		<!-- 标签的内容类型:empty表示空标签,jsp表示可以为任何合法的JSP元素 -->
		<body-content>jsp</body-content>
		<!-- 自定义标签的属性定义,请注意一定要在标签类中提供对应的get/set方法 -->
		<attribute>
			<!-- 自定义标签的属性名称 -->
			<name>name</name>
			<!-- true表示必填 -->
			<required>true</required>
			<!-- true支持动态值,可以向值里面填jsp表达式、EL表达式,false则不支持 -->
			<rtexprvalue>false</rtexprvalue>
		</attribute>
	</tag>
	
	<!-- 自定义out标签 -->
	<tag>
		<!-- 标签名 -->
		<name>out</name>
		<!-- 标签工具类 -->
		<tag-class>com.web.tag.OutTag</tag-class>
		<!-- 标签的内容类型:empty表示空标签,jsp表示可以为任何合法的JSP元素 -->
		<body-content>jsp</body-content>
		<!-- 自定义标签的属性定义,请注意一定要在标签类中提供对应的get/set方法 -->
		<attribute>
			<!-- 自定义标签的属性名称 -->
			<name>value</name>
			<!-- true表示必填 -->
			<required>true</required>
			<!-- true支持动态值,可以向值里面填jsp表达式、EL表达式,false则不支持 -->
			<rtexprvalue>true</rtexprvalue>
		</attribute>
	</tag>
	
	<!-- 自定义if标签 -->
	<tag>
		<!-- 标签名 -->
		<name>if</name>
		<!-- 标签工具类 -->
		<tag-class>com.web.tag.IfTag</tag-class>
		<!-- 标签的内容类型:empty表示空标签,jsp表示可以为任何合法的JSP元素 -->
		<body-content>jsp</body-content>
		<!-- 自定义标签的属性定义,请注意一定要在标签类中提供对应的get/set方法 -->
		<attribute>
			<!-- 自定义标签的属性名称 -->
			<name>test</name>
			<!-- true表示必填 -->
			<required>true</required>
			<!-- true支持动态值,可以向值里面填jsp表达式、EL表达式,false则不支持 -->
			<rtexprvalue>true</rtexprvalue>
		</attribute>
	</tag>
	
	<!-- 自定义forEach标签 -->
	<tag>
		<!-- 标签名 -->
		<name>forEach</name>
		<!-- 标签工具类 -->
		<tag-class>com.web.tag.ForEachTag</tag-class>
		<!-- 标签的内容类型:empty表示空标签,jsp表示可以为任何合法的JSP元素 -->
		<body-content>jsp</body-content>
		<!-- 自定义标签的属性定义,请注意一定要在标签类中提供对应的get/set方法 -->
		<attribute>
			<!-- 自定义标签的属性名称 -->
			<name>items</name>
			<!-- true表示必填 -->
			<required>true</required>
			<!-- true支持动态值,可以向值里面填jsp表达式、EL表达式,false则不支持 -->
			<rtexprvalue>true</rtexprvalue>
		</attribute>
		<attribute>
			<!-- 自定义标签的属性名称 -->
			<name>var</name>
			<!-- true表示必填 -->
			<required>true</required>
			<!-- true支持动态值,可以向值里面填jsp表达式、EL表达式,false则不支持 -->
			<rtexprvalue>false</rtexprvalue>
		</attribute>
	</tag>
	
	
	<!-- 自定义if标签 -->
	<tag>
		<!-- 标签名 -->
		<name>if</name>
		<!-- 标签工具类 -->
		<tag-class>com.web.tag.IfTag</tag-class>
		<!-- 标签的内容类型:empty表示空标签,jsp表示可以为任何合法的JSP元素 -->
		<body-content>jsp</body-content>
		<!-- 自定义标签的属性定义,请注意一定要在标签类中提供对应的get/set方法 -->
		<attribute>
			<!-- 自定义标签的属性名称 -->
			<name>test</name>
			<!-- true表示必填 -->
			<required>true</required>
			<!-- true支持动态值,可以向值里面填jsp表达式、EL表达式,false则不支持 -->
			<rtexprvalue>true</rtexprvalue>
		</attribute>
	</tag>
	
	<tag>
		<!-- 标签名 -->
		<name>dept</name>
		<!-- 标签工具类 -->
		<!-- 利用反射机制Class.forName() -->
		<tag-class>com.web.tag.DeptTag</tag-class>
		<!-- 标签的内容类型:empty表示空标签,jsp表示可以为任何合法的JSP元素 -->
		<body-content>empty</body-content>
		<!-- 自定义标签的属性定义,请注意一定要在标签类中提供对应的get/set方法 -->
		<attribute>
			<!-- 自定义标签的属性名称 -->
			<name>var</name>
			<!-- true表示必填 -->
			<required>true</required>
			<!-- true支持动态值,可以向值里面填jsp表达式、EL表达式,false则不支持 -->
			<rtexprvalue>false</rtexprvalue>
		</attribute>
		<attribute>
			<!-- 自定义标签的属性名称 -->
			<name>scope</name>
			<!-- true表示必填 -->
			<required>false</required>
			<!-- true支持动态值,可以向值里面填jsp表达式、EL表达式,false则不支持 -->
			<rtexprvalue>false</rtexprvalue>
		</attribute>
	</tag>
	
	<!-- 自定义select下拉框标签 -->
	<tag>
		<!-- 标签名 -->
		<name>select</name>
		<!-- 标签工具类 -->
		<tag-class>com.web.tag.SelectTag</tag-class>
		<!-- 标签的内容类型:empty表示空标签,jsp表示可以为任何合法的JSP元素 -->
		<body-content>jsp</body-content>
		<!-- 自定义标签的属性定义,请注意一定要在标签类中提供对应的get/set方法 -->
		<attribute>
			<!-- 自定义标签的属性名称 -->
			<name>items</name>
			<!-- true表示必填 -->
			<required>true</required>
			<!-- true支持动态值,可以向值里面填jsp表达式、EL表达式,false则不支持 -->
			<rtexprvalue>true</rtexprvalue>
		</attribute>
		<attribute>
			<!-- 自定义标签的属性名称 -->
			<name>optionValue</name>
			<!-- true表示必填 -->
			<required>true</required>
			<!-- true支持动态值,可以向值里面填jsp表达式、EL表达式,false则不支持 -->
			<rtexprvalue>false</rtexprvalue>
		</attribute>
		<attribute>
			<!-- 自定义标签的属性名称 -->
			<name>optionTest</name>
			<!-- true表示必填 -->
			<required>true</required>
			<!-- true支持动态值,可以向值里面填jsp表达式、EL表达式,false则不支持 -->
			<rtexprvalue>false</rtexprvalue>
		</attribute>
	</tag>
</taglib>

     ①  package com.web.entity包下:

        Student.java        ——实体学生类

package com.web.entity;

import java.io.Serializable;

public class Student implements Serializable {
/**
	 * 唯一编号:
	 */
	private static final long serialVersionUID = 1L;
private int sid;
private String sname;
public Student() {

}
public Student(int sid, String sname) {
	super();
	this.sid = sid;
	this.sname = sname;
}
public int getSid() {
	return sid;
}
public void setSid(int sid) {
	this.sid = sid;
}
public String getSname() {
	return sname;
}
public void setSname(String sname) {
	this.sname = sname;
}
@Override
public String toString() {
	return "Student [sid=" + sid + ", sname=" + sname + "]";
}
}

        接下来我们要借助自定义标签帮助类,来实现我们想要的功能

        注意:所有的标签帮助类都必须继承BodyTagSupport类

        ②package com.web.tag包下:

        1.TestTag.java                实现TestTag标签帮助类        ——自定义测试

package com.web.tag;

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

/**
 * JSP自定义标签开发及使用步骤:
 * 1)创建标签助手类并继承BodyTagSupport
 * 注:标签描述文件中定义的属性与助手类的中的属性必须一致,并且提供get/set方法
 * 2)创建标签描述文件(.tld),并在其中添加自定义配置
 * 注:必须添加到WEB-INF或其子目录下
 * 3)在指定的页面中通过taglib指令导入自定义标签库
 * @author Administrator
 *
 */
public class TestTag extends BodyTagSupport {
	
	private int count=0;
	
	private Object name;

	public Object getName() {
		return name;
	}

	public void setName(Object name) {
		this.name = name;
	}

	/**
	 * doStartTag:对应<开始标签>所执行的动作
	 * 例如:对应<z:test>标签
	 * 返回值:
	 * 1)SKIP_BODY:跳过主体内容不包含在输出中
	 * 2)EVAL_BODY_INCLUDE:计算主体内容并包含在输出中
	 */
	@Override
	public int doStartTag() throws JspException {
		System.out.println("name="+this.name);
		System.out.println("doStartTag:对应<开始标签>所执行的动作");
		return EVAL_BODY_INCLUDE;
	}
	
	/**
	 * doAfterBody:介于<开始标签>标签体与<结束标签>之间所执行的动作
	 * 例如:介于<z:test>123123与</z:test>之间所执行的动作
	 * 返回值:
	 * 1)SKIP_BODY:跳过页面内容不输出(可以理解成for循环中的break)
	 * 2)EVAL_BODY_AGAIN:再次计算标签主体内容并保存在输出中(可以理解成for循环中的continue)
	 */
	@Override
	public int doAfterBody() throws JspException {
		System.out.println("doAfterBody:介于<开始标签>标签体与<结束标签>之间所执行的动作");
		//第一次 count=0 输出
		//第二次 count=1 输出
		//第三次 count=2 输出
		//第四次 count=3
		if(count<3) {
			count++;
			return EVAL_BODY_AGAIN;
		}
		return SKIP_BODY;
	}

	/**
	 * doEndTag:对应<结束标签>所执行的动作
	 * 例如:对应</z:test>标签
	 * 返回值:
	 * 1)SKIP_PAGE:跳过页面的后续内容
	 * 2)EVAL_PAGE:计算页面的后续内容
	 */
	@Override
	public int doEndTag() throws JspException {
		System.out.println("doEndTag:对应<结束标签>所执行的动作");
		return EVAL_PAGE;
	}

}

        2.OutTag.java                实现out标签的帮助类        ——模拟jstl中的out标签效果

package com.web.tag;

import java.io.IOException;

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

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.write(value.toString());
		} catch (Exception e) {
			e.printStackTrace();
		}
		return SKIP_BODY;
	}
	

	@Override
	public int doEndTag() throws JspException {
		return EVAL_PAGE;
	}
}

        3.IfTag.java                实现if标签的帮助类        ——模拟jstl中的if标签效果

package com.web.tag;

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

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 {
		//如果条件成立,则显示标签体;如果条件不成立,则跳过标签体
		if(test)
			return EVAL_BODY_INCLUDE;
		else
			return SKIP_BODY;
	}
	
	@Override
	public int doEndTag() throws JspException {
		return EVAL_PAGE;
	}
}

        4.ForeachTag.java                实现forEach标签的帮助类        ——模拟jstl中的forEach标签效果

package com.web.tag;

import java.util.Iterator;
import java.util.List;

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

public class ForeachTag extends BodyTagSupport {

	//被遍历的集合或者数组,但是在此案例中只遍历集合
	private List items;
	
	//将每次循环遍历的数据保存到指定作用域中,并以var属性来命名
	//pageContext.setAttribute(var,每次循环遍历的数据项);
	private String var;
	

	public List getItems() {
		return items;
	}

	public void setItems(List items) {
		this.items = items;
	}

	public String getVar() {
		return var;
	}

	public void setVar(String var) {
		this.var = var;
	}

	@Override
	public int doStartTag() throws JspException {
		//判断items是否为空
		if(null==items) {
			return SKIP_BODY;
		}else {
			//获取迭代器
			Iterator it = items.iterator();
			//获取迭代器中的下一个元素
			Object value = it.next();
			//将获取的第一个元素保存到page作用域中,并以var属性来命名
			//pageContext.setAttribute("l",value);
			//${l}  page -> request -> session -> application
			pageContext.setAttribute(var, value);
			//将剩余未遍历的迭代器对象保存到作用域中,留到doAfterBody中再次进行遍历
			pageContext.setAttribute("it", it);
			return EVAL_BODY_INCLUDE;
		}
	}

	@Override
	public int doAfterBody() throws JspException {
		//首页要到page作用域中获取剩余未遍历完的迭代器对象
		Iterator it = (Iterator) pageContext.getAttribute("it");
		//判断it迭代器对象的下一个元素是否存在
		//SKIP_BODY:break;
		//EVAL_BODY_AGAIN:continue;
		if(it.hasNext()) {
			//获取迭代器的下一个元素
			Object value = it.next();
			//将当前元素value保存到page作用域中并以var属性命名
			pageContext.setAttribute(var, value);
			//将剩余未遍历的迭代器对象保存到作用域中,留到doAfterBody中再次进行遍历
			pageContext.setAttribute("it", it);
			return EVAL_BODY_AGAIN;
		}else
			return SKIP_BODY;
	}

	@Override
	public int doEndTag() throws JspException {
		return EVAL_PAGE;
	}
}

        5.DeptTag.java                实现dept标签帮助类        ——自定义将结果保存域对象

package com.web.tag;

import java.util.ArrayList;
import java.util.List;

import javax.servlet.ServletContext;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.BodyTagSupport;

import com.web.utils.Student;

public class DeptTag extends BodyTagSupport {
	
	//将数据保存到page作用域中,并以var属性命名
	private String var;
	//作用域,默认page
	private String scope;
	
	@Override
	public int doStartTag() throws JspException {
		//提前通过Biz层访问数据
		/*List<Dept> lst=new ArrayList<>();
		lst.add(new Student(1,"研发部"));
		lst.add(new Student(2,"市场部"));
		lst.add(new Student(3,"人事部"));*/
		List<Student> lst=new ArrayList<>();
		lst.add(new Student(4,"玄幻"));
		lst.add(new Student(6,"神话"));
		lst.add(new Student(8,"计算机"));
		
		//获取Request对象
		HttpServletRequest request = (HttpServletRequest) pageContext.getRequest();
		//获取Session对象
		HttpSession session = request.getSession();
		//获取Application对象
		ServletContext application = pageContext.getServletContext();
		
		if(null==scope)
			pageContext.setAttribute(var, lst);
		else if(scope.equals("request"))
			request.setAttribute(var, lst);
		else if(scope.equals("session"))
			session.setAttribute(var, lst);
		else if(scope.equals("application"))
			application.setAttribute(var, lst);
		else
			pageContext.setAttribute(var, lst);
		return SKIP_BODY;
	}
	
	@Override
	public int doEndTag() throws JspException {
		return EVAL_PAGE;
	}
	
	public String getVar() {
		return var;
	}
	public void setVar(String var) {
		this.var = var;
	}
	public String getScope() {
		return scope;
	}
	public void setScope(String scope) {
		this.scope = scope;
	}
}

        6.SelectTag.java                 实现select标签帮助类        ——自定义select下拉框标签

package com.web.tag;

import java.io.IOException;
import java.lang.reflect.Field;
import java.util.List;

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

public class SelectTag extends BodyTagSupport {
	
	//被遍历的集合或者数组,用于循环绑定到select下拉框中
	//Dept = [deptId=1, deptName=研发部]
	//BookType = [btId=1,btName=玄幻]
	private List items;
	//对应循环绑定对象中的指定属性名,例如:将Dept对象的deptId属性绑定到option标签的value属性
	private String optionValue;
	//对应循环绑定对象中的指定属性名,例如:将Dept对象的deptName属性绑定到option标签的标签体中
	private String optionText;
	
	@Override
	public int doStartTag() throws JspException {
		try {
			JspWriter out = pageContext.getOut();
			out.write(toHtml());
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		return SKIP_BODY;
	}
	
	private String toHtml() {
		StringBuffer sb=new StringBuffer();

		//拼接select标签
		sb.append("<select>");
		
		//请选择
		sb.append("<option value=''>---请选择---</option>");
		
		//循环遍历items
		for (Object obj : items) {
			//obj==Dept(deptId,deptName)
			//deptId -> option的value属性上
			//deptName -> option的标签体上
			String value=getObjValue(obj,this.optionValue);
			String text=getObjValue(obj,this.optionText);
			//拼接option标签
			sb.append("<option value='"+value+"'>"+text+"</option>");
		}
		
		//拼接select标签
		sb.append("</select>");
		return sb.toString();
	}
	
	private String getObjValue(Object obj,String fieldName) {
		String value=null;
		try {
			//一切与反射相关的代码都从获取类对象开始
			//获取类对象
			Class cls = obj.getClass();
			//获取所有的属性数组
			Field[] fields = cls.getDeclaredFields();
			//循环遍历所有属性
			for (Field field : fields) {
				//判断属性名与fieldName的绑定名称是否相同
				if(field.getName().toUpperCase().equals(fieldName.toUpperCase())) {
					//设置权限
					field.setAccessible(true);
					//获取数据
					value=field.get(obj).toString();
					break;
				}
			}
		} catch (Exception e) {
			e.printStackTrace();
		}
		return value;
	}
	
	@Override
	public int doEndTag() throws JspException {
		return EVAL_PAGE;
	}
	
	public List getItems() {
		return items;
	}
	public void setItems(List items) {
		this.items = items;
	}
	public String getOptionValue() {
		return optionValue;
	}
	public void setOptionValue(String optionValue) {
		this.optionValue = optionValue;
	}
	public String getOptionText() {
		return optionText;
	}
	public void setOptionText(String optionText) {
		this.optionText = optionText;
	}
}

        接下来编写WebContent目录下的index.jsp测试文件

<%@page import="com.web.entity.Student"%>
<%@page import="java.util.ArrayList"%>
<%@page import="java.util.List"%>
<%@page import="java.util.Arrays"%>
<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
    <!-- 引入jstl库 -->
    <%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
    <!-- 引入自定义标签库 -->
    <%@taglib uri="/yutianfu" prefix="y" %>
<!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>
<%
	//将一个字符串数组转换成list集合
	List<String> list = Arrays.asList(new String[]{"张三","李四","王五","zs","ls","ww"});
	//保存到作用域request中
	request.setAttribute("list", list);
	//实例化一个对象集合
	List<Student> stuList = new ArrayList<Student>();
	stuList.add(new Student(1,"张三"));
	stuList.add(new Student(2,"李四"));
	stuList.add(new Student(3,"王五"));
	stuList.add(new Student(4,"老六"));
	//System.out.println(stuList);
	//保存到作用域request中
	request.setAttribute("stuList", stuList);
%>
<h1>c标签部分</h1>
<c:out value="你好呀!"></c:out>
<c:if test="${true }">
可以
</c:if>
<c:forEach items="${list }" var="n">
${n }
</c:forEach>

<h1>自定义标签部分</h1>
<y:test name="你好哟!">
不错哟
enen
tt
cc
</y:test>
<!-- out输出标签 -->
<y:out value="可以哦">
之后的呢?
</y:out>
yy
<hr>
<!-- if判断标签 -->
<y:if test="${true }">
哈哈!
</y:if>
<hr>
<!-- forEach循环标签 -->
<y:forEach items="${list }" var="l">
${l }
</y:forEach>
<hr>
ii
<!-- 自定义dept标签 -->
<y:dept var="dept" scope="request"/>
<hr>
<!-- 自定义select标签 -->
<y:select optionTest="sname" items="${stuList }" optionValue="sid">
</y:select>
</body>
</html>
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值