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