Struts底层工作原理

从Taglib的角度论述Struts项目到底是如何工作的?(初学者可以略过)
马克- to-win:马克 java社区:防盗版实名手机尾号: 73203。
上面我们在讲:taglib推出以后的巨大变革中,讲到:你在jsp中引入他们的一个标签儿,再按照他们的语法配置,被你引入的那个标签儿,就能够调用你编写的类。下面我们就给出一个例子来说明这件事是怎么实现的。底下例子中的struts.tld和 com.struts.GetDataFromDB.java,这两个文件都是struts这个架构公司编的。它把这两个文件,打包在一个jar包里,发布出来。各路世界的工程师到它的网站下载下来,导到自己的项目当中。比如在我自己的项目中,我编写了一个index.jsp,其中我引用了struts公司的一个标签,struts:getDataFromDB,这个标签的两个属性className和methodName规定了,只要你把自己写的类名和方法名写在这里,这个标签就能调用你的类里的方法。还规定你的这个方法返回的,一定是从数据库获得的一个字符串。之后它这个标签儿就能把你的这个返回的字符串显示在网页上。总体算一下,从数据库当中取回字符串显示在网页当中,整个这件事,我们只编写了自己的一个类 com.myself.GetDataFromDB,而且完全是按照我们自己的意志,访问的数据库。因为这个类完全是我们自己编写的。大部分的代码tld 文件和com.struts.GetDataFromDB.java文件,还是struts架构公司编写的。优点一: jsp中避免了使用脚本元素,用的是标签儿,方便维护。优点二,这里没有用我们自己编的标签儿。引入了一个新的架构struts,可以利用这个架构当中的其他很多标准特性,比如上传,国际化等。

当然这个例子没有正式在讲struts架构,而是在讲struts是如何应用taglib技术和我们的项目合作的。


例 1.2.4:

index.jsp:

<%@ page contentType="text/html; charset=GBK" %>
<%@ taglib uri="/WEB-INF/struts.tld" prefix="struts" %>
<html>
   <body>
     <struts:getDataFromDB className="com.myself.GetDataFromDB" methodName="getData"/>
   </body>
</html>


struts.tld:

<?xml version="1.0" encoding="ISO-8859-1" ?>
<!DOCTYPE taglib
     PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.1//EN"
     "http://java.sun.com/j2ee/dtds/web-jsptaglibrary_1_1.dtd">
<taglib>
     <tlibversion>1.0</tlibversion>
     <jspversion>1.1</jspversion>
     <shortname>mark-to-win</shortname>
     <tag>
       <name>getDataFromDB</name>
       <tagclass>com.struts.GetDataFromDB</tagclass>
        <bodycontent>empty</bodycontent>
        <attribute>
            <name>className</name>
        </attribute>
        <attribute>
            <name>methodName</name>
        </attribute>
     </tag>
</taglib>




com.struts.GetDataFromDB.java(这里用到了反射技术,有兴趣的同学可以网上参考。)

package com.struts;
import javax.servlet.jsp.*;
import javax.servlet.jsp.tagext.*;
import java.io.*;
import java.lang.reflect.Method;
public class GetDataFromDB extends TagSupport{
    private String className, methodName;
    public void setClassName(String className) {
        this.className = className;
    }
    public void setMethodName(String methodName) {
        this.methodName = methodName;
    }
    public int doStartTag() {
        try {
            Class clz = Class.forName(className);
            Object obj = clz.newInstance();
            Method m = obj.getClass().getDeclaredMethod(methodName);
            Object  returnV = m.invoke(obj);
            pageContext.getOut().print((String)returnV);
        } catch (Exception ex) {
            ex.printStackTrace();
        }
        return this.SKIP_BODY;
    }
}



package com.myself;
public class GetDataFromDB {
    public String getData()
    {
         //这里应该是一大段访问数据库的代码
        return "returnValue from DB";
    }
}



运行index.jsp后,浏览器中的结果是:

returnValue from DB


5)带有body的标签的例子:


如果你想正儿八经的处理标签body的话,你就得用到doAfterBody这个方法。





例 1.2.5:

<%@ page contentType="text/html; charset=GBK" %>
<%@ taglib uri="/WEB-INF/tagExampleLib.tld" prefix="greeter" %>
<html>
   <body>
     <greeter:Hello>
        Now 时间 is: <%=new java.util.Date() %> <br>
     </greeter:Hello>
     结束
   </body>
</html>




tagExampleLib.tld:


<?xml version="1.0" encoding="ISO-8859-1" ?>
<!DOCTYPE taglib
     PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.1//EN"
     "http://java.sun.com/j2ee/dtds/web-jsptaglibrary_1_1.dtd">
<taglib>
    <tlibversion>1.0</tlibversion>
    <jspversion>1.1</jspversion>
    <shortname>mark-to-win</shortname>
    <tag>
        <name>Hello</name>
        <tagclass>com.marktowin.HelloWorldTag</tagclass>
    </tag>
</taglib>



/*顺序是: 4)doStartTag
4.5) doAfterBody
5)doEndTag*/

package com.marktowin;
import javax.servlet.jsp.*;
import javax.servlet.jsp.tagext.*;
import java.io.*;

public class HelloWorldTag extends TagSupport {
    int count = 2;
    public int doStartTag() {
        System.out.println("doStartTag");
        return EVAL_BODY_INCLUDE;
    }

    public int doAfterBody() {
        System.out.println("doAfterBody.");
        while (count > 0) {
            count--;
            /* EVAL_BODY_AGAIN requires that the body of the tag is evaluated. */
            return EVAL_BODY_AGAIN;
        }
        return SKIP_BODY;
    }

    public int doEndTag() {
        System.out.println("doEndTagqqq");
        return EVAL_PAGE;
    }
}


运行jsp后,浏览器输出的结果是:

Now 时间 is: Mon May 29 16:33:53 CST 2017
Now 时间 is: Mon May 29 16:33:53 CST 2017
Now 时间 is: Mon May 29 16:33:53 CST 2017
结束


console里同时输出的结果是:

doStartTag
doAfterBody.
doAfterBody.
doAfterBody.
doEndTagqqq