JSP自定义标签的页面解析生命周期及线程安全

问题描述:

使用JSP自定义标签时,发现标签内的私有变量在页面第一次请求之后就不会发生变化了。而且不同的页面同样的标签变量值不一样。

问题猜想:

原以为JSP自定义标签在JSP编译的时候会被new出来,但是看样子JSP在编译的时候用了缓存池来存放解析后的标签的对象。并且这个缓存池针对同一个页面是共享的。

问题原因跟踪:

查看JSP编译成的servlet代码,发现标签的实例果然不是new出来的,而是这么创建的:

com.xxx.tag.XXTag _jspx_th_xxx_005fxx_005f0 = (com.xxx.tag.XXTag) _005fjspx_005ftagPool_005xxx_005fxx_0026_005ftitle.get(com.xxx.tag.XXTag.class);

是从一个缓存池里取出来的,再往前跟踪缓存池的创建和销毁:

public void _jspInit() {...._005fjspx_005ftagPool_005xxx_005fxx_0026_005ftitle = org.apache.jasper.runtime.TagHandlerPool.getTagHandlerPool(getServletConfig());....}public void _jspDestroy() { ....._005fjspx_005ftagPool_005xxx_005fxx_0026_005ftitle.release();.....}

把当前servlet的config作为参数,得到了一个标签缓存池的实例,也就是说对于同一个servlet标签缓存池是共享的。证明了之前的猜想。

既然实例共享,那么很自然的就想到了tomcat是怎么做到线程安全的呢?

关于标签的线程安全可以参考下面的链接,不再赘述:

结论:

虽然使用标签时可以不用担心线程安全,但是在使用标签内的属性变量时要特别小心,在标签解析完时要注意将一些不必要的属性变量重置为默认值,否则很容易“一劳永逸”。

P.s:JSP自定义标签生命周期http://blog.csdn.net/xuejianxinokok/archive/2010/06/29/5702155.aspx

相关推荐