• 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     @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 }
    复制代码

      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服务器的"workCatalinalocalhostJavaWeb_JspTag_study_20140816orgapachejsp"目录下可以找到将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("
    ");
     63       out.write("<!-- 引用gacl标签库,标签库的前缀(prefix)可以随便设置,如这里设置成 prefix="gacl" -->
    ");
     64       out.write("
    ");
     65       out.write("<!DOCTYPE HTML>
    ");
     66       out.write("<html>
    ");
     67       out.write("  <head>
    ");
     68       out.write("    <title>输出客户端的IP</title>
    ");
     69       out.write("  </head>
    ");
     70       out.write("  
    ");
     71       out.write("  <body>
    ");
     72       out.write("    你的IP地址是(使用java代码获取输出):
    ");
     73       out.write("    ");
     74 
     75         //在jsp页面中使用java代码获取客户端IP地址
     76         String ip = request.getRemoteAddr();
     77         out.write(ip);
     78     
     79       out.write("
    ");
     80       out.write("    <hr/>
    ");
     81       out.write("     你的IP地址是(使用自定义标签获取输出):");
     82       if (_jspx_meth_xdp_005fviewIP_005f0(_jspx_page_context))
     83         return;
     84       out.write("
    ");
     85       out.write("  </body>
    ");
     86       out.write("</html>
    ");
     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的自定义标签开发,在后面的博文中会进行更加详尽的介绍。

  • 相关阅读:
    BF算法(串模式匹配算法)
    python字符串 常用函数 格式化字符串 字符串替换 制表符 换行符 删除空白 国际货币格式
    python 列表
    python 循环语句
    http://www.pythontutor.com/visualize.html#mode=edit python在线检测代码
    GDI+_绘制QQ头像
    socket编程之select()
    socket编程之select()
    设置itemcontrol的item点击前后不同状态
    设置itemcontrol的item点击前后不同状态
  • 原文地址:https://www.cnblogs.com/telwanggs/p/5337676.html
Copyright © 2020-2023  润新知