自定義標籤主要用於移除Jsp頁面中的java代碼。html
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 @Override 17 public int doEndTag() throws JspException { 18 System.out.println("調用doEndTag()方法"); 19 return 0; 20 } 21 22 @Override 23 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 @Override 39 public Tag getParent() { 40 return null; 41 } 42 43 @Override 44 public void release() { 45 System.out.println("調用release()方法"); 46 } 47 48 @Override 49 public void setPageContext(PageContext pageContext) { 50 System.out.println("setPageContext(PageContext pageContext)"); 51 this.pageContext = pageContext; 52 } 53 54 @Override 55 public void setParent(Tag arg0) { 56 57 } 58 59 }
gacl.tld文件的代碼以下:java
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>
一、使用"<%@taglib uri="標籤庫的uri" prefix="標籤的使用前綴"%>"指令引入要使用的標籤庫。web
例如:在jspTag_Test1.jsp中引用gacl標籤庫express
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>
標籤的運行效果以下:apache
從運行效果種能夠看到,使用自定義標籤就能夠將jsp頁面上的java代碼移除掉,如須要在jsp頁面上輸出客戶端的IP地址時,使用 <xdp:viewIP/>標籤就能夠代替jsp頁面上的這些代碼:tomcat
1 <% 2 //在jsp頁面中使用java代碼獲取客戶端IP地址 3 String ip = request.getRemoteAddr(); 4 out.write(ip); 5 %>
這就是開發和使用自定義標籤的好處,可讓咱們的Jsp頁面上不嵌套java代碼。服務器
JSP引擎遇到自定義標籤時,首先建立標籤處理器類的實例對象,而後按照JSP規範定義的通訊規則依次調用它的方法。
一、public void setPageContext(PageContext pc), JSP引擎實例化標籤處理器後,將調用setPageContext方法將JSP頁面的pageContext對象傳遞給標籤處理器,標籤處理器之後能夠經過這個pageContext對象與JSP頁面進行通訊。
二、public void setParent(Tag t),setPageContext方法執行完後,WEB容器接着調用的setParent方法將當前標籤的父標籤傳遞給當前標籤處理器,若是當前標籤沒有父標籤,則傳遞給setParent方法的參數值爲null。
三、public int doStartTag(),調用了setPageContext方法和setParent方法以後,WEB容器執行到自定義標籤的開始標記時,就會調用標籤處理器的doStartTag方法。
四、public int doEndTag(),WEB容器執行完自定義標籤的標籤體後,就會接着去執行自定義標籤的結束標記,此時,WEB容器會去調用標籤處理器的doEndTag方法。
五、public void release(),一般WEB容器執行完自定義標籤後,標籤處理器會駐留在內存中,爲其它請求服務器,直至中止web應用時,web容器纔會調用release方法。session
咱們在tomcat服務器的"work\Catalina\localhost\JavaWeb_JspTag_study_20140816\org\apache\jsp"目錄下能夠找到將jspTag_Test1.jsp翻譯成Servlet後的java源代碼,以下圖所示:app
打開jspTag_005fTest1_jsp.java文件,能夠看到setPageContext(PageContext pc)、setParent(Tag t)、doStartTag()、doEndTag()、release()這5個方法的調用順序和過程。jsp
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:viewIP 104 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:viewIP 2 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的自定義標籤開發,在後面的博文中會進行更加詳盡的介紹。