jsp自定义标签 线程安全

系统 1833 0

 

转自: http://klcwt.iteye.com/blog/749652

 

我们在编写自定义标签的时候设置属性如下

Java代码     收藏代码
  1. public   class  InputTag  extends  TagSupport {  
  2.   
  3.      private   static   final   long  serialVersionUID = 1L;  
  4.   
  5.      private  String onclick;  
  6.   
  7.      private  String style;  
  8.   
  9.      private  String styleClass;  
  10.   
  11.      private  String value;  
  12.   
  13.      private  String id;  

 

在页面上如果同时使用两个标签:

Html代码     收藏代码
  1. < h3:input   type = "button"   onclick = "myFun()"   name = "name"   id = "id"   
  2.          style = "style"   styleClass = "styleClass"   value = "中国人"   url = "url"   
  3.          pid = "pid"   isValidated = "true" >   
  4.     中国人  
  5.      </ h3:input >   
  6.       
  7.      < h3:input   type = "button"   onclick = "myFun()"   name = "name"   id = "id"   
  8.          style = "style"   styleClass = "styleClass"   value = "美国人"   url = "url"   
  9.          pid = "pid"   isValidated = "true" >   
  10.      </ h3:input >   

 

从后台发现打印的InpuTag都是 同一个对象!

 

发现这个后,我十分担心线程安全问题!比如这些getType();setType(); !

 

于是就看了下jsp生成的Servlet源代码

Java代码     收藏代码
  1. out.write( "\t<body>\r\n" );  
  2.   out.write( "\t\t" );  
  3.     //调用InputTag   
  4.    if  (_jspx_meth_h3_005finput_005f0(_jspx_page_context))  
  5.      return ;  
  6.   out.write( "\r\n" );  
  7.   out.write( "\t\t\r\n" );  
  8.   out.write( "\t\t" );  
  9.    //调用InputTag   
  10.    if  (_jspx_meth_h3_005finput_005f1(_jspx_page_context))  
  11.      return ;  
  12.   out.write( "\r\n" );  
  13.   out.write( "\t</body>\r\n" );  

 

再接着看_jspx_meth_h3_005finput_005f0方法

Java代码     收藏代码
  1.    private   boolean  _jspx_meth_h3_005finput_005f0(PageContext _jspx_page_context)  
  2.            throws  Throwable {  
  3.     PageContext pageContext = _jspx_page_context;  
  4.     JspWriter out = _jspx_page_context.getOut();  
  5.      //  h3:input   
  6.    <span style= "color: #ff0000;" > tag.InputTag <span style= "color: #0000ff;" >_jspx_th_h3_005finput_005f0</span> = (tag.InputTag) _005fjspx_005ftagPool_005fh3_005finput_0026_005fvalue_005furl_005ftype_005fstyleClass_005fstyle_005fpid_005fonclick_005fname_005fisValidated_005fid.get(tag.InputTag. class );</span>  
  7.     _jspx_th_h3_005finput_005f0.setPageContext(_jspx_page_context);  
  8.     _jspx_th_h3_005finput_005f0.setParent( null );  
  9.      // /button2.jsp(12,2) name = type type = null reqTime = true required = false fragment = false deferredValue = false expectedTypeName = null deferredMethod = false methodSignature = null   
  10.     _jspx_th_h3_005finput_005f0.setType( "button" );  
  11.      // /button2.jsp(12,2) name = onclick type = java.lang.String reqTime = false required = true fragment = false deferredValue = false expectedTypeName = null deferredMethod = false methodSignature = null    _jspx_th_h3_005finput_005f0.setPid("pid");   
  12.      // /button2.jsp(12,2) name = isValidated type = null reqTime = true required = false fragment = false deferredValue = false expectedTypeName = null deferredMethod = false methodSignature = null   
  13.     _jspx_th_h3_005finput_005f0.setIsValidated( "true" );  
  14.      int  _jspx_eval_h3_005finput_005f0 = _jspx_th_h3_005finput_005f0.doStartTag();  
  15.      if  (_jspx_eval_h3_005finput_005f0 != javax.servlet.jsp.tagext.Tag.SKIP_BODY) {  
  16.        do  {  
  17.         out.write( "\r\n" );  
  18.         out.write( "\t\t中国人\r\n" );  
  19.         out.write( "\t\t" );  
  20.          int  evalDoAfterBody = _jspx_th_h3_005finput_005f0.doAfterBody();  
  21.          if  (evalDoAfterBody != javax.servlet.jsp.tagext.BodyTag.EVAL_BODY_AGAIN)  
  22.            break ;  
  23.       }  while  ( true );  
  24.     }  
  25.      if  (_jspx_th_h3_005finput_005f0.doEndTag() == javax.servlet.jsp.tagext.Tag.SKIP_PAGE) {  
  26.       <span style= "color: #ff0000;" >_005fjspx_005ftagPool_005fh3_005finput_0026_005fvalue_005furl_005ftype_005fstyleClass_005fstyle_005fpid_005fonclick_005fname_005fisValidated_005fid.reuse(_jspx_th_h3_005finput_005f0);</span>  
  27.        return   true ;  
  28.     }  
  29. <span style= "color: #ff0000;" >    _005fjspx_005ftagPool_005fh3_005finput_0026_005fvalue_005furl_005ftype_005fstyleClass_005fstyle_005fpid_005fonclick_005fname_005fisValidated_005fid.reuse(_jspx_th_h3_005finput_005f0);</span>  
  30.      return   false ;  
  31.   }  

最关键就是这句了,看他如何获得自定义标签对象: tag.InputTag _jspx_th_h3_005finput_005f0 = (tag.InputTag) _005fjspx_005ftagPool_005fh3_005finput_0026_005fvalue_005furl_005ftype_005fstyleClass_005fstyle_005fpid_005fonclick_005fname_005fisValidated_005fid.get(tag.InputTag.class);

解释下:

 _jspx_th_h3_005finput_005f0 是InputTag 的实例 也就是<h3:input.

005fjspx_005ftagPool_005fh3_005finput_0026_005fvalue_005furl_005ftype_005fstyleClass_005fstyle_005fpid_005fonclick_005fname_005fisValidated_005

是TagHandlerPool的实例

 

自定义标签是通过这个TagHandlerPool.get 来获取的!

举一反三,有借就有还TagHandlerPool.reuse用来回收这个对象!

 

 

TagHandlerPool.get 

TagHandlerPool.reuse

方法如下:

 

Java代码     收藏代码
  1. /**  
  2.   * Gets the next available tag handler from this tag handler pool,  
  3.   * instantiating one if this tag handler pool is empty.  
  4.   *  
  5.   * @param handlerClass Tag handler class  
  6.   *  
  7.   * @return Reused or newly instantiated tag handler  
  8.   *  
  9.   * @throws JspException if a tag handler cannot be instantiated  
  10.   */   
  11.   public  Tag get(Class handlerClass)  throws  JspException {  
  12. g handler =  null ;  
  13.       synchronized this  ) {  
  14.           if  (current >=  0 ) {  
  15.              handler = handlers[current--];  
  16.               return  handler;  
  17.          }  
  18.      }  
  19.   
  20.       // Out of sync block - there is no need for other threads to   
  21.       // wait for us to construct a tag for this thread.   
  22.       try  {  
  23.          Tag instance = (Tag) handlerClass.newInstance();  
  24.          AnnotationHelper.postConstruct(annotationProcessor, instance);  
  25.           return  instance;  
  26.      }  catch  (Exception e) {  
  27.           throw   new  JspException(e.getMessage(), e);  
  28.      }  
  29.  }  

 

Java代码     收藏代码
  1. /**  
  2.  * Adds the given tag handler to this tag handler pool, unless this tag  
  3.  * handler pool has already reached its capacity, in which case the tag  
  4.  * handler's release() method is called.  
  5.  *  
  6.  * @param handler Tag handler to add to this tag handler pool  
  7.  */   
  8. public   void  reuse(Tag handler) {  
  9.      synchronized this  ) {  
  10.          if  (current < (handlers.length -  1 )) {  
  11.             handlers[++current] = handler;  
  12.              return ;  
  13.         }  
  14.     }  
  15.      // There is no need for other threads to wait for us to release   
  16.     handler.release();  
  17.      if  (annotationProcessor !=  null ) {  
  18.          try  {  
  19.             AnnotationHelper.preDestroy(annotationProcessor, handler);  
  20.         }  catch  (Exception e) {  
  21.             log.warn( "Error processing preDestroy on tag instance of "    
  22.                     + handler.getClass().getName(), e);  
  23.         }  
  24.     }  
  25. }  

 

 

现在就明白了所有的tag对象都是有一个对象池来维护的,一是方便了重用,而是做到了线程同步。

 

 总结:自定义标签是线程安全的,同时也是可重用的!


 

同时又有另一个疑问

 synchronized( this ) {
            if (current >= 0) {
                handler = handlers[current--];
                return handler;
            }
        }

感觉这种方法可能只能在一个页面上共享,另一个页面上的tag估计是不能共享的!

 

后来看了下生成的servelt代码

如下:

Java代码     收藏代码
  1. public   void  _jspInit() {  
  2.     _tagpool = org.apache.jasper.runtime.TagHandlerPool.getTagHandlerPool(getServletConfig());  
  3.     _el_expressionfactory = _jspxFactory.getJspApplicationContext(getServletConfig().getServletContext()).getExpressionFactory();  
  4.     _jsp_annotationprocessor = (org.apache.AnnotationProcessor) getServletConfig().getServletContext().getAttribute(org.apache.AnnotationProcessor. class .getName());  
  5.   }  

 可以看到_tagpool 是根据ServletConifg来生成的

TagHandlerPool.getTagHandlerPool代码如下

Java代码     收藏代码
  1. public   static  TagHandlerPool getTagHandlerPool( ServletConfig config) {  
  2.        TagHandlerPool result= null ;  
  3.   
  4.        String tpClassName=getOption( config, OPTION_TAGPOOL,  null );  
  5.         if ( tpClassName !=  null  ) {  
  6.             try  {  
  7.                Class c=Class.forName( tpClassName );  
  8.                result=(TagHandlerPool)c.newInstance();  
  9.            }  catch  (Exception e) {  
  10.                e.printStackTrace();  
  11.                result= null ;  
  12.            }  
  13.        }  
  14.         if ( result== null  ) result= new  TagHandlerPool();  
  15.        result.init(config);  
  16.   
  17.         return  result;  
  18.    }  

 

jsp自定义标签 线程安全


更多文章、技术交流、商务合作、联系博主

微信扫码或搜索:z360901061

微信扫一扫加我为好友

QQ号联系: 360901061

您的支持是博主写作最大的动力,如果您喜欢我的文章,感觉我的文章对您有帮助,请用微信扫描下面二维码支持博主2元、5元、10元、20元等您想捐的金额吧,狠狠点击下面给点支持吧,站长非常感激您!手机微信长按不能支付解决办法:请将微信支付二维码保存到相册,切换到微信,然后点击微信右上角扫一扫功能,选择支付二维码完成支付。

【本文对您有帮助就好】

您的支持是博主写作最大的动力,如果您喜欢我的文章,感觉我的文章对您有帮助,请用微信扫描上面二维码支持博主2元、5元、10元、自定义金额等您想捐的金额吧,站长会非常 感谢您的哦!!!

发表我的评论
最新评论 总共0条评论