http://hi.baidu.com/yanhua365/blog/item/a05930ad04b777094a36d662.html
頁面中太多的自定義標(biāo)簽會不會影響性能?
太多的自定義標(biāo)簽(尤其是這種包含了一部分業(yè)務(wù)邏輯的標(biāo)簽)會不會影響性能呢?下面一段文字是我以前寫的,可以說明一些問題,但真正對性能會不會產(chǎn)生太大的影響,還要看具體的情況,如果有必要的話,做一下性能測試得出的結(jié)果才是可靠的。
我們知道,JSP歸根到底是Servlet,也就是說最終是要編譯成Servlet的。如果是在Servlet的service方法中new一個Tag處理類來處理出現(xiàn)的標(biāo)簽的話,那就意味著每一次客戶的請求都會生成新的Tag處理類,這樣對服務(wù)器的性能和資源影響都很大。而如果把Tag處理類的對象做為Servlet的一個屬性的話,又會遇到線程安全等問題。據(jù)說,不同的JSP容器有不同的處理方式,有的容器下,一個頁面上出現(xiàn)太多的自定義標(biāo)簽,響應(yīng)速度就會很慢。真擔(dān)心太多的JSP和太多的自定義標(biāo)簽會使服務(wù)器承受不了啊。--今天突然想到,可以看一下Tomcat是怎么把包含有自定義標(biāo)簽的JSP頁面編譯成Servlet的,也許就清楚自定義標(biāo)簽對性能的影響有多大了!
原來,Tomcat用了緩沖池的概念來解決。。。
比如我在JSP中用到了一個Struts標(biāo)簽<html:text property="username"></html:text>那么,在對應(yīng)的JSP編譯的Servlet中會有一個這樣的屬性private org.apache.jasper.runtime.TagHandlerPool _jspx_tagPool_html_text_property;可以看出來,這是Tomcat提供的一個標(biāo)簽處理的緩沖池,在下面會用到它。在Servlet中會生成下面這個方法:
private boolean _jspx_meth_html_text_0(javax.servlet.jsp.tagext.JspTag _jspx_th_html_form_0, PageContext pageContext)
throws Throwable {
JspWriter out = pageContext.getOut();
/* ---- html:text ---- */
org.apache.struts.taglib.html.TextTag _jspx_th_html_text_0 = (org.apache.struts.taglib.html.TextTag) _jspx_tagPool_html_text_property.get(org.apache.struts.taglib.html.TextTag.class);
_jspx_th_html_text_0.setPageContext(pageContext);
if (_jspx_th_html_form_0 instanceof javax.servlet.jsp.tagext.SimpleTag)
_jspx_th_html_text_0.setParent(new javax.servlet.jsp.tagext.TagAdapter((javax.servlet.jsp.tagext.SimpleTag) _jspx_th_html_form_0));
else
_jspx_th_html_text_0.setParent((javax.servlet.jsp.tagext.Tag) _jspx_th_html_form_0);
_jspx_th_html_text_0.setProperty("username");
int _jspx_eval_html_text_0 = _jspx_th_html_text_0.doStartTag();
if (_jspx_th_html_text_0.doEndTag() == javax.servlet.jsp.tagext.Tag.SKIP_PAGE)
return true;
_jspx_tagPool_html_text_property.reuse(_jspx_th_html_text_0);
return false;
}
可以看出,在這個方法中,對標(biāo)簽處理類做了緩沖池處理,可以大大提高性能。
當(dāng)然,_jspx_tagPool_html_text_property這個Servlet的屬性,要在初始化時賦值,在最后釋放掉。
public void _jspInit() {
_jspx_tagPool_html_text_property = org.apache.jasper.runtime.TagHandlerPool.getTagHandlerPool(getServletConfig());
}
public void _jspDestroy() {
_jspx_tagPool_html_text_property.release();
}
需要說明的是,這種緩沖池是在這個JSP生命周期的范圍內(nèi)的,也就是說,另一個JSP頁面用到了相同的標(biāo)簽,也會重新建立一個緩沖池,和這個頁面毫無關(guān)系。還有一點,就是在同一個JSP頁面中,也不是一個標(biāo)簽對應(yīng)一個緩沖池,比如說我用到了link標(biāo)簽的兩種形式,它們屬性的個數(shù)是不一樣的:
<html:link action="/test">Test with a link</html:link>
<html:link action="/test" paramId="username" paramName="haha">
那就會生成兩個緩沖池:
private org.apache.jasper.runtime.TagHandlerPool _jspx_tagPool_html_link_action;
private org.apache.jasper.runtime.TagHandlerPool _jspx_tagPool_html_link_paramName_paramId_action;
這樣,雖然服務(wù)器上會有很多這樣的緩沖池,不過相對于每次請求創(chuàng)建Tag處理類來說,這不算什么,大大提高了性能和利用了服務(wù)器資源。
我們這里說的都是Tomcat的解決方案,至于其它容器,也許有不同的解決方案。