最近在学习自定义标签,自己写的一个循环标签,明明被循环的List只有3个元素,确循环了4次,代码如下
jsp:
<%
List<String> list = new ArrayList<String>();
list.add("a1");
list.add("a2");
list.add("a3");
request.setAttribute("list", list);
%>
<Test:loop item="${list}" var="result">
<c:out value="${result}test"></c:out>
</Test:loop>
java:
package com;
import java.io.IOException;
import java.util.Iterator;
import java.util.List;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.IterationTag;
import javax.servlet.jsp.tagext.Tag;
import javax.servlet.jsp.tagext.TagSupport;
public class MyLoopTag extends TagSupport {
private Iterator<String> it;
private List<String> item;
public void setItem(List<String> item){
this.item = item;
}
private String var;
public void setVar(String var) {
this.var = var;
}
@Override
public int doEndTag() throws JspException {
System.out.println("end");
return Tag.SKIP_PAGE;
}
@Override
public int doStartTag() throws JspException {
System.out.println("start");
if(this.item!=null){
return Tag.EVAL_BODY_INCLUDE;
}
}
@Override
public int doAfterBody() throws JspException {
System.out.println("doAfterBody");
if(it!=null){
if(it.hasNext()){
String str = it.next();
System.out.println("test"+str);
this.pageContext.setAttribute(var, str);
return this.EVAL_BODY_AGAIN;
}
}
return Tag.SKIP_BODY;
}
}
预期输出结果:a1test a2test a3test
实际输出结果:test a1test a2test a3test
后来看jsp对应的源码,才找出问题,
下面是我自己写的一个jsp中标签执行的简略过程,就可以理解上面的结果了
//jsp的servlet中的方法
public void _jspService(HttpServletRequest request, HttpServletResponse response){
/***此处省略标签体前的代码***/
//执行到自定义标签时,调用一个方法
if(someMethod(pagecontext)==true){
return;
}
/***此处省略自定义标签结尾后面的代码***/
}
private boolean someMethod(pagecontext){
//获取标签类实例
SomeTag tag;
/**此处省略初始化tag属性***/
//执行doStartTag
int startR = tag.doStartTag();
if(startR != Tag.SKIP_BODY){//不等于SKIP_BODY时执行标签体内的内容
do{
/****此处省略标签体内容输出****/
int afterR = tag.doAfterBody();
if(afterR!=BodyTag.EVAL_BODY_AGAIN){
break;
}
}while(true);
}
if(Tag.SKIP_PAGE == tag.doEndTag()){
return true;
}else{
return false;
}
}
上面的代码清楚得告诉了我们标签类的方法调用过程,就很容易理解每个方法的作用和返回值的作用了。
因为doStartTag后的循环是do while,所以要达到预期的结果,修改如下
public int doStartTag() throws JspException {
System.out.println("start");
if(this.item!=null){
it = this.item.iterator();
if(it.hasNext()){
String str = it.next();
this.pageContext.setAttribute(var, str);
return Tag.EVAL_BODY_INCLUDE;
}else{
return Tag.SKIP_BODY;
}
}else{
return Tag.SKIP_BODY;
}
}