javaweb學習總結(二十三)——jsp自訂標籤開發入門

來源:互聯網
上載者:User

標籤:

javaweb學習總結(二十三)——jsp自訂標籤開發入門一、自訂標籤的作用

  自訂標籤主要用於移除Jsp頁面中的java代碼。

二、自訂標籤開發和使用2.1、自訂標籤開發步驟    1、編寫一個實現Tag介面的Java類(標籤處理器類)
 1 package me.gacl.web.tag; 2  3 import java.io.IOException; 4  5 import javax.servlet.http.HttpServletRequest; 6 import javax.servlet.jsp.JspException; 7 import javax.servlet.jsp.JspWriter; 8 import javax.servlet.jsp.PageContext; 9 import javax.servlet.jsp.tagext.Tag;10 11 public class ViewIPTag implements Tag {12 13     //接收傳遞進來的PageContext對象14     private PageContext pageContext;15     16     @Override17     public int doEndTag() throws JspException {18         System.out.println("調用doEndTag()方法");19         return 0;20     }21 22     @Override23     public int doStartTag() throws JspException {24         System.out.println("調用doStartTag()方法");25         HttpServletRequest request =(HttpServletRequest) pageContext.getRequest();26         JspWriter out = pageContext.getOut();27         String ip = request.getRemoteAddr();28         try {29             //這裡輸出的時候會拋出IOException異常30             out.write(ip);31         } catch (IOException e) {32             //捕獲IOException異常後繼續拋出33             throw new RuntimeException(e);34         }35         return 0;36     }37 38     @Override39     public Tag getParent() {40         return null;41     }42 43     @Override44     public void release() {45         System.out.println("調用release()方法");46     }47 48     @Override49     public void setPageContext(PageContext pageContext) {50         System.out.println("setPageContext(PageContext pageContext)");51         this.pageContext = pageContext;52     }53 54     @Override55     public void setParent(Tag arg0) {56 57     }58 59 }
  2、在WEB-INF/目錄下建立tld檔案,在tld檔案中對標籤處理器類進行描述

  

  gacl.tld檔案的代碼如下:

 1 <?xml version="1.0" encoding="UTF-8" ?> 2  3 <taglib xmlns="http://java.sun.com/xml/ns/j2ee" 4     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 5     xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-jsptaglibrary_2_0.xsd" 6     version="2.0"> 7     <!-- description用來添加對taglib(標籤庫)的描述 --> 8     <description>孤傲蒼狼開發的自訂標籤庫</description> 9     <!--taglib(標籤庫)的版本號碼 -->10     <tlib-version>1.0</tlib-version>11     <short-name>GaclTagLibrary</short-name>12     <!-- 13         為自訂標籤庫設定一個uri,uri以/開頭,/後面的內容隨便寫,如這裡的/gacl ,14         在Jsp頁面中引用標籤庫時,需要通過uri找到標籤庫15         在Jsp頁面中就要這樣引入標籤庫:<%@taglib uri="/gacl" prefix="gacl"%>16     -->17     <uri>/gacl</uri>18     19     <!--一個taglib(標籤庫)中包含多個自訂標籤,每一個自訂標籤使用一個tag標記來描述  -->20     <!-- 一個tag標記對應一個自訂標籤 -->21      <tag>22         <description>這個標籤的作用是用來輸出用戶端的IP地址</description>23         <!-- 24             為標籤處理器類配一個標籤名,在Jsp頁面中使用標籤時是通過標籤名來找到要調用的標籤處理器類的25             通過viewIP就能找到對應的me.gacl.web.tag.ViewIPTag類26          -->27         <name>viewIP</name>28         <!-- 標籤對應的處理器類-->29         <tag-class>me.gacl.web.tag.ViewIPTag</tag-class>30         <body-content>empty</body-content>31     </tag>32     33 </taglib>
2.2、在Jsp頁面中使用自訂標籤

  1、使用"<%@taglib uri="標籤庫的uri"  prefix="標籤的使用首碼"%>"指令引入要使用的標籤庫。

例如:在jspTag_Test1.jsp中引用gacl標籤庫

 1 <%@ page language="java" pageEncoding="UTF-8"%> 2 <!-- 使用taglib指令引用gacl標籤庫,標籤庫的首碼(prefix)可以隨便設定,如這裡設定成 prefix="xdp" --> 3 <%@taglib uri="/gacl"  prefix="xdp"%> 4 <!DOCTYPE HTML> 5 <html> 6   <head> 7     <title>輸出用戶端的IP</title> 8   </head> 9   10   <body>11     你的IP地址是(使用java代碼擷取輸出):12     <%13         //在jsp頁面中使用java代碼擷取用戶端IP地址14         String ip = request.getRemoteAddr();15         out.write(ip);16     %>17     <hr/>18      你的IP地址是(使用自訂標籤擷取輸出):19      <%--使用自訂標籤viewIP --%>20      <xdp:viewIP/>21   </body>22 </html>

  標籤的運行效果如下:

  

  從運行效果種可以看到,使用自訂標籤就可以將jsp頁面上的java代碼移除掉,如需要在jsp頁面上輸出用戶端的IP地址時,使用 <xdp:viewIP/>標籤就可以代替jsp頁面上的這些代碼:

1 <%2         //在jsp頁面中使用java代碼擷取用戶端IP地址3         String ip = request.getRemoteAddr();4         out.write(ip);5 %>

  這就是開發和使用自訂標籤的好處,可以讓我們的Jsp頁面上不嵌套java代碼。

三、自訂標籤的執行流程

  JSP引擎遇到自訂標籤時,首先建立標籤處理器類的執行個體對象,然後按照JSP規範定義的通訊規則依次調用它的方法。
    1、public void setPageContext(PageContext pc), JSP引擎執行個體化標籤處理器後,將調用setPageContext方法將JSP頁面的pageContext對象傳遞給標籤處理器,標籤處理器以後可以通過這個pageContext對象與JSP頁面進行通訊。
    2、public void setParent(Tag t),setPageContext方法執行完後,WEB容器接著調用的setParent方法將當前標籤的父標籤傳遞給當前標籤處理器,如果當前標籤沒有父標籤,則傳遞給setParent方法的參數值為null。
    3、public int doStartTag(),調用了setPageContext方法和setParent方法之後,WEB容器執行到自訂標籤的開始標記時,就會調用標籤處理器的doStartTag方法。
    4、public int doEndTag(),WEB容器執行完自訂標籤的標籤體後,就會接著去執行自訂標籤的結束標記,此時,WEB容器會去調用標籤處理器的doEndTag方法。
    5、public void release(),通常WEB容器執行完自訂標籤後,標籤處理器會駐留在記憶體中,為其它請求伺服器,直至停止web應用時,web容器才會調用release方法。

   我們在tomcat伺服器的"work\Catalina\localhost\JavaWeb_JspTag_study_20140816\org\apache\jsp"目錄下可以找到將jspTag_Test1.jsp翻譯成Servlet後的java原始碼,如所示:

  開啟jspTag_005fTest1_jsp.java檔案,可以看到setPageContext(PageContext pc)、setParent(Tag t)、doStartTag()、doEndTag()、release()這5個方法的調用順序和過程。

  jspTag_005fTest1_jsp.java的代碼如下:

  1 package org.apache.jsp;  2   3 import javax.servlet.*;  4 import javax.servlet.http.*;  5 import javax.servlet.jsp.*;  6   7 public final class jspTag_005fTest1_jsp extends org.apache.jasper.runtime.HttpJspBase  8     implements org.apache.jasper.runtime.JspSourceDependent {  9  10   private static final JspFactory _jspxFactory = JspFactory.getDefaultFactory(); 11  12   private static java.util.List _jspx_dependants; 13  14   static { 15     _jspx_dependants = new java.util.ArrayList(1); 16     _jspx_dependants.add("/WEB-INF/gacl.tld"); 17   } 18  19   private org.apache.jasper.runtime.TagHandlerPool _005fjspx_005ftagPool_005fxdp_005fviewIP_005fnobody; 20  21   private javax.el.ExpressionFactory _el_expressionfactory; 22   private org.apache.AnnotationProcessor _jsp_annotationprocessor; 23  24   public Object getDependants() { 25     return _jspx_dependants; 26   } 27  28   public void _jspInit() { 29     _005fjspx_005ftagPool_005fxdp_005fviewIP_005fnobody = org.apache.jasper.runtime.TagHandlerPool.getTagHandlerPool(getServletConfig()); 30     _el_expressionfactory = _jspxFactory.getJspApplicationContext(getServletConfig().getServletContext()).getExpressionFactory(); 31     _jsp_annotationprocessor = (org.apache.AnnotationProcessor) getServletConfig().getServletContext().getAttribute(org.apache.AnnotationProcessor.class.getName()); 32   } 33  34   public void _jspDestroy() { 35     _005fjspx_005ftagPool_005fxdp_005fviewIP_005fnobody.release(); 36   } 37  38   public void _jspService(HttpServletRequest request, HttpServletResponse response) 39         throws java.io.IOException, ServletException { 40  41     PageContext pageContext = null; 42     HttpSession session = null; 43     ServletContext application = null; 44     ServletConfig config = null; 45     JspWriter out = null; 46     Object page = this; 47     JspWriter _jspx_out = null; 48     PageContext _jspx_page_context = null; 49  50  51     try { 52       response.setContentType("text/html;charset=UTF-8"); 53       pageContext = _jspxFactory.getPageContext(this, request, response, 54                   null, true, 8192, true); 55       _jspx_page_context = pageContext; 56       application = pageContext.getServletContext(); 57       config = pageContext.getServletConfig(); 58       session = pageContext.getSession(); 59       out = pageContext.getOut(); 60       _jspx_out = out; 61  62       out.write("\r\n"); 63       out.write("<!-- 引用gacl標籤庫,標籤庫的首碼(prefix)可以隨便設定,如這裡設定成 prefix=\"gacl\" -->\r\n"); 64       out.write("\r\n"); 65       out.write("<!DOCTYPE HTML>\r\n"); 66       out.write("<html>\r\n"); 67       out.write("  <head>\r\n"); 68       out.write("    <title>輸出用戶端的IP</title>\r\n"); 69       out.write("  </head>\r\n"); 70       out.write("  \r\n"); 71       out.write("  <body>\r\n"); 72       out.write("    你的IP地址是(使用java代碼擷取輸出):\r\n"); 73       out.write("    "); 74  75         //在jsp頁面中使用java代碼擷取用戶端IP地址 76         String ip = request.getRemoteAddr(); 77         out.write(ip); 78      79       out.write("\r\n"); 80       out.write("    <hr/>\r\n"); 81       out.write("     你的IP地址是(使用自訂標籤擷取輸出):"); 82       if (_jspx_meth_xdp_005fviewIP_005f0(_jspx_page_context)) 83         return; 84       out.write("\r\n"); 85       out.write("  </body>\r\n"); 86       out.write("</html>\r\n"); 87     } catch (Throwable t) { 88       if (!(t instanceof SkipPageException)){ 89         out = _jspx_out; 90         if (out != null && out.getBufferSize() != 0) 91           try { out.clearBuffer(); } catch (java.io.IOException e) {} 92         if (_jspx_page_context != null) _jspx_page_context.handlePageException(t); 93       } 94     } finally { 95       _jspxFactory.releasePageContext(_jspx_page_context); 96     } 97   } 98  99   private boolean _jspx_meth_xdp_005fviewIP_005f0(PageContext _jspx_page_context)100           throws Throwable {101     PageContext pageContext = _jspx_page_context;102     JspWriter out = _jspx_page_context.getOut();103     //  xdp:viewIP104     me.gacl.web.tag.ViewIPTag _jspx_th_xdp_005fviewIP_005f0 = (me.gacl.web.tag.ViewIPTag) _005fjspx_005ftagPool_005fxdp_005fviewIP_005fnobody.get(me.gacl.web.tag.ViewIPTag.class);105     _jspx_th_xdp_005fviewIP_005f0.setPageContext(_jspx_page_context);106     _jspx_th_xdp_005fviewIP_005f0.setParent(null);107     int _jspx_eval_xdp_005fviewIP_005f0 = _jspx_th_xdp_005fviewIP_005f0.doStartTag();108     if (_jspx_th_xdp_005fviewIP_005f0.doEndTag() == javax.servlet.jsp.tagext.Tag.SKIP_PAGE) {109       _005fjspx_005ftagPool_005fxdp_005fviewIP_005fnobody.reuse(_jspx_th_xdp_005fviewIP_005f0);110       return true;111     }112     _005fjspx_005ftagPool_005fxdp_005fviewIP_005fnobody.reuse(_jspx_th_xdp_005fviewIP_005f0);113     return false;114   }115 }

   下面重點分析一下上述代碼中標紅色的那個 private boolean _jspx_meth_xdp_005fviewIP_005f0(PageContext _jspx_page_context)方法中的代碼

  ①、這裡是執行個體化一個viewIP標籤處理器類me.gacl.web.tag.ViewIPTag的對象

1  //  xdp:viewIP2     me.gacl.web.tag.ViewIPTag _jspx_th_xdp_005fviewIP_005f0 = (me.gacl.web.tag.ViewIPTag) _005fjspx_005ftagPool_005fxdp_005fviewIP_005fnobody.get(me.gacl.web.tag.ViewIPTag.class);

  ②、執行個體化標籤處理器後,調用setPageContext方法將JSP頁面的pageContext對象傳遞給標籤處理器

1 _jspx_th_xdp_005fviewIP_005f0.setPageContext(_jspx_page_context);

  ③、setPageContext方法執行完後,接著調用的setParent方法將當前標籤的父標籤傳遞給當前標籤處理器,如果當前標籤沒有父標籤,則傳遞給setParent方法的參數值為null

1 _jspx_th_xdp_005fviewIP_005f0.setParent(null);

  ④、調用了setPageContext方法和setParent方法之後,WEB容器執行到自訂標籤的開始標記時,就會調用標籤處理器的doStartTag方法

1 int _jspx_eval_xdp_005fviewIP_005f0 = _jspx_th_xdp_005fviewIP_005f0.doStartTag();

  ⑤、WEB容器執行完自訂標籤的標籤體後,就會接著去執行自訂標籤的結束標記,此時,WEB容器會去調用標籤處理器的doEndTag方法

1 if (_jspx_th_xdp_005fviewIP_005f0.doEndTag() == javax.servlet.jsp.tagext.Tag.SKIP_PAGE)

  這就是自訂標籤的執行流程。

  這裡以一個入門級的案例來講解javaweb的自訂標籤開發,在後面的博文中會進行更加詳盡的介紹。

javaweb學習總結(二十三)——jsp自訂標籤開發入門

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.