jimphei學(xué)習(xí)工作室

          jimphei學(xué)習(xí)工作室

            BlogJava :: 首頁(yè) :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
            23 隨筆 :: 0 文章 :: 1 評(píng)論 :: 0 Trackbacks

          2009年8月14日 #

          import java.util.Map;

          import org.apache.velocity.app.VelocityEngine;
          import org.springframework.ui.velocity.VelocityEngineUtils;

          public class MsgBean ...{
              private VelocityEngine velocityEngine;

              private String msg;

              private Map model; // 用來(lái)保存velocity中的參數(shù)值

              private String encoding; // 編碼

              private String templateLocation; // 注入的velocity模塊

              public String getEncoding() ...{
                  return encoding;
              }

              public void setEncoding(String encoding) ...{
                  this.encoding = encoding;
              }

              public String getTemplateLocation() ...{
                  return templateLocation;
              }

              public void setTemplateLocation(String templateLocation) ...{
                  this.templateLocation = templateLocation;
              }

              public Map getModel() ...{
                  return model;
              }

              public void setModel(Map model) ...{
                  this.model = model;
              }

              public String getMsg() ...{
                  // return title;
                  // 將參數(shù)值注入到模塊后的返回值
                  return VelocityEngineUtils.mergeTemplateIntoString(velocityEngine,
                          templateLocation, encoding, model);

              }

              public void setMsg(String msg) ...{
                  this.msg = msg;
              }

              public VelocityEngine getVelocityEngine() ...{
                  return velocityEngine;
              }

              public void setVelocityEngine(VelocityEngine velocityEngine) ...{
                  this.velocityEngine = velocityEngine;
              }

          }

          <?xml version="1.0" encoding="UTF-8"?>
          <beans xmlns="http://www.springframework.org/schema/beans"
                 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
                 xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd">

           

             
           <bean id="velocityEngine" class="org.springframework.ui.velocity.VelocityEngineFactoryBean"> 
             <property name="resourceLoaderPath">
                      <value>classpath:velocity</value>
               </property>
              <property name="velocityProperties">
                           <props>
                                 <prop key="resource.loader">class</prop>
                                 <prop key="class.resource.loader.class">
                                       org.apache.velocity.runtime.resource.loader.ClasspathResourceLoader
                                 </prop>
                                 <prop key="velocimacro.library"></prop>
                                 <prop key="input.encoding">GBK</prop>
                                 <prop key="output.encoding">GBK</prop>
                                 <prop key="default.contentType">text/html; charset=GBK</prop>
                           </props>
                     </property>
          </bean>

          <bean id="msgBean" class="MsgBean">
                  <property name="templateLocation" value="test.vm"></property>
                  <property name="encoding" value="GBK"></property>
                  <property name="velocityEngine" ref="velocityEngine"></property>
          </bean>


          </beans>

          import java.io.File;
          import java.io.IOException;
          import java.util.HashMap;
          import java.util.Map;

          import org.apache.commons.io.FileUtils;
          import org.springframework.context.ApplicationContext;
          import org.springframework.context.support.ClassPathXmlApplicationContext;


          public class TestVeloctiy ...{
              public static void main(String[] args) ...{
                  // TODO Auto-generated method stub
                  ApplicationContext ctx=new ClassPathXmlApplicationContext("test3.xml");
                  MsgBean    msgBean=((MsgBean)ctx.getBean("msgBean"));
                  Map<String, String> data = new HashMap<String, String>();
                  data.put("me","yourname");
                  msgBean.setModel(data);
                  System.out.println(msgBean.getMsg());
                 
                  
                  //根據(jù)apache common IO 組件直接將內(nèi)容寫到一個(gè)文件中去.
                   File dest = new File( "test.html" );         
                    try ...{
                      FileUtils.writeStringToFile( dest, msgBean.getMsg(), "GBK" );
                  } catch (IOException e) ...{
                      // TODO Auto-generated catch block
                      e.printStackTrace();
                  }

              }
          }

          本文來(lái)自CSDN博客,轉(zhuǎn)載請(qǐng)標(biāo)明出處:http://blog.csdn.net/pengchua/archive/2008/01/17/2049490.aspx

          posted @ 2009-11-26 11:36 jimphei 閱讀(1149) | 評(píng)論 (0)編輯 收藏

          引用自:http://blog.csdn.net/axzywan/archive/2008/07/12/2643921.aspx

          取Session中的值

          <c:out value="${sessionScope.user.userId}"></c:out><br>  

          <c:out value="${user.userLoginName}"></c:out><br>    

          <s:property value="#session.user.userId"/><br>  

          ${session.user.userId}<br> 

          ${sessionScope.user.userId}<br>

          OGNL

          OGNL 是Object Graph Navigation Language 的簡(jiǎn)稱,詳細(xì)相關(guān)的信息可以參考:http://www.ognl.org 。這里我們只涉及Struts2 框架中對(duì)OGNL 的基本支持。

           

          OGNL 是一個(gè)對(duì)象,屬性的查詢語(yǔ)言。在OGNL 中有一個(gè)類型為Map 的Context (稱為上下文),在這個(gè)上下文中有一個(gè)根元素(root ),對(duì)根元素的屬性的訪問(wèn)可以直接使用屬性名字,但是對(duì)于其他非根元素屬性的訪問(wèn)必須加上特殊符號(hào)# 。

           

          在Struts2 中上下文為ActionContext ,根元素位Value Stack (值堆棧,值堆棧代表了一族對(duì)象而不是一個(gè)對(duì)象,其中Action 類的實(shí)例也屬于值堆棧的一個(gè))。ActionContext 中的內(nèi)容如下圖:

                        |

                        |--application

                        |

                        |--session

          context map---|

                         |--value stack(root)

                        |

                        |--request

                        |

                        |--parameters

                        |

                        |--attr (searches page, request, session, then application scopes)

                        |

          因?yàn)锳ction 實(shí)例被放在Value Stack 中,而Value Stack 又是根元素(root )中的一個(gè),所以對(duì)Action 中的屬性的訪問(wèn)可以不使用標(biāo)記# ,而對(duì)其他的訪問(wèn)都必須使用# 標(biāo)記。

           

          引用Action 的屬性

          <s:property value="postalCode"/>

          ActionContext 中的其他非根(root )元素的屬性可以按照如下的方式訪問(wèn):

          <s:property value="#session.mySessionPropKey"/> or

          <s:property value="#session["mySessionPropKey"]"/> or

          <s:property value="#request["mySessionPropKey"]/>

           

          Action 類可以使用ActionContext 中的靜態(tài)方法來(lái)訪問(wèn)ActionContext 。

          ActionContext.getContext().getSession().put("mySessionPropKey", mySessionObject);

           

          OGNL 與Collection (Lists ,Maps ,Sets )

           

          生成List 的語(yǔ)法為: {e1,e2,e3}.

          <s:select label="label" name="name"

          list="{'name1','name2','name3'}" value="%{'name2'}" />

          上面的代碼生成了一個(gè)HTML Select 對(duì)象,可選的內(nèi)容為: name1 ,name2 ,name3 ,默認(rèn)值為:name2 。

           

          生成Map 的語(yǔ)法為:#{key1:value1,key2:value2}.

          <s:select label="label" name="name"

          list="#{'foo':'foovalue', 'bar':'barvalue'}" />

          上面的代碼生成了一個(gè)HTML Select 對(duì)象,foo 名字表示的內(nèi)容為:foovalue ,bar 名字表示的內(nèi)容為:barvalue 。

           

          判斷一個(gè)對(duì)象是否在List 內(nèi)存在:

          <s:if test="'foo' in {'foo','bar'}">

             muhahaha

          </s:if>

          <s:else>

             boo

          </s:else>

           

          <s:if test="'foo' not in {'foo','bar'}">

             muhahaha

          </s:if>

          <s:else>

             boo

          </s:else>

           

          取得一個(gè)List 的一部分:

          ?   –   所有滿足選擇邏輯的對(duì)象

          ^   -    第一個(gè)滿足選擇邏輯的對(duì)象

          $   -    最后一個(gè)滿足選擇邏輯的對(duì)象

          例如:

          person.relatives.{? #this.gender == 'male'}

          上述代碼取得這個(gè)人(person )所有的男性(this.gender==male )的親戚(relatives)

           

           

          Lambda 表達(dá)式

           

          OGNL 支持簡(jiǎn)單的Lambda 表達(dá)式語(yǔ)法,使用這些語(yǔ)法可以建立簡(jiǎn)單的lambda 函數(shù)。

           

          例如:

          Fibonacci:

          if n==0 return 0;

          elseif n==1 return 1;

          else return fib(n-2)+fib(n-1);

          fib(0) = 0

          fib(1) = 1

          fib(11) = 89

           

          OGNL 的Lambda 表達(dá)式如何工作呢?

          Lambda 表達(dá)式必須放在方括號(hào)內(nèi)部,#this 表示表達(dá)式的參數(shù)。例如:

          <s:property value="#fib =:[#this==0 ? 0 : #this==1 ? 1 : #fib(#this-2)+#fib(#this-1)], #fib(11)" />

           

          #fib =:[#this==0 ? 0 : #this==1 ? 1 : #fib(#this-2)+#fib(#this-1)] 定義了一個(gè)Lambda 表達(dá)式,

          #fib(11) 調(diào)用了這個(gè)表達(dá)式。

           

          所以上述代碼的輸出為:89

           

          在JSP2.1 中# 被用作了JSP EL (表達(dá)式語(yǔ)言)的特殊記好,所以對(duì)OGNL 的使用可能導(dǎo)致問(wèn)題,

          一個(gè)簡(jiǎn)單的方法是禁用JSP2.1 的EL 特性,這需要修改web.xml 文件:

          <jsp-config>

              <jsp-property-group>

                <url-pattern>*.jsp</url-pattern>

                <el-ignored>true</el-ignored>

              </jsp-property-group>

          </jsp-config>

          關(guān)于EL表達(dá)式語(yǔ)言的簡(jiǎn)單總結(jié)
           

          基本語(yǔ)法

          一、EL簡(jiǎn)介
            1.語(yǔ)法結(jié)構(gòu)
              ${expression}
            2.[]與.運(yùn)算符
              EL 提供.和[]兩種運(yùn)算符來(lái)存取數(shù)據(jù)。
              當(dāng)要存取的屬性名稱中包含一些特殊字符,如.或?等并非字母或數(shù)字的符號(hào),就一定要使用 []。例如:
                  ${user.My-Name}應(yīng)當(dāng)改為${user["My-Name"] }
              如果要?jiǎng)討B(tài)取值時(shí),就可以用[]來(lái)做,而.無(wú)法做到動(dòng)態(tài)取值。例如:
                  ${sessionScope.user[data]}中data 是一個(gè)變量
            3.變量
              EL存取變量數(shù)據(jù)的方法很簡(jiǎn)單,例如:${username}。它的意思是取出某一范圍中名稱為username的變量。
              因?yàn)槲覀儾](méi)有指定哪一個(gè)范圍的username,所以它會(huì)依序從Page、Request、Session、Application范圍查找。
              假如途中找到username,就直接回傳,不再繼續(xù)找下去,但是假如全部的范圍都沒(méi)有找到時(shí),就回傳null。
              屬性范圍在EL中的名稱
                  Page         PageScope
                  Request         RequestScope
                  Session         SessionScope
                  Application     ApplicationScope
                 
          二、EL隱含對(duì)象
            1.與范圍有關(guān)的隱含對(duì)象
            與范圍有關(guān)的EL 隱含對(duì)象包含以下四個(gè):pageScope、requestScope、sessionScope 和applicationScope;
            它們基本上就和JSP的pageContext、request、session和application一樣;
            在EL中,這四個(gè)隱含對(duì)象只能用來(lái)取得范圍屬性值,即getAttribute(String name),卻不能取得其他相關(guān)信息。
           
            例如:我們要取得session中儲(chǔ)存一個(gè)屬性u(píng)sername的值,可以利用下列方法:
              session.getAttribute("username") 取得username的值,
            在EL中則使用下列方法
              ${sessionScope.username}

            2.與輸入有關(guān)的隱含對(duì)象
            與輸入有關(guān)的隱含對(duì)象有兩個(gè):param和paramValues,它們是EL中比較特別的隱含對(duì)象。
           
            例如我們要取得用戶的請(qǐng)求參數(shù)時(shí),可以利用下列方法:
              request.getParameter(String name)
              request.getParameterValues(String name)
            在EL中則可以使用param和paramValues兩者來(lái)取得數(shù)據(jù)。
              ${param.name}
              ${paramValues.name}

            3.其他隱含對(duì)象
           
            cookie
            JSTL并沒(méi)有提供設(shè)定cookie的動(dòng)作,
            例:要取得cookie中有一個(gè)設(shè)定名稱為userCountry的值,可以使用${cookie.userCountry}來(lái)取得它。

            header和headerValues
            header 儲(chǔ)存用戶瀏覽器和服務(wù)端用來(lái)溝通的數(shù)據(jù)
            例:要取得用戶瀏覽器的版本,可以使用${header["User-Agent"]}。
            另外在鮮少機(jī)會(huì)下,有可能同一標(biāo)頭名稱擁有不同的值,此時(shí)必須改為使用headerValues 來(lái)取得這些值。

            initParam
            initParam取得設(shè)定web站點(diǎn)的環(huán)境參數(shù)(Context)
            例:一般的方法String userid = (String)application.getInitParameter("userid");
              可以使用 ${initParam.userid}來(lái)取得名稱為userid

            pageContext
            pageContext取得其他有關(guān)用戶要求或頁(yè)面的詳細(xì)信息。
              ${pageContext.request.queryString}         取得請(qǐng)求的參數(shù)字符串
              ${pageContext.request.requestURL}         取得請(qǐng)求的URL,但不包括請(qǐng)求之參數(shù)字符串
              ${pageContext.request.contextPath}         服務(wù)的web application 的名稱
              ${pageContext.request.method}           取得HTTP 的方法(GET、POST)
              ${pageContext.request.protocol}         取得使用的協(xié)議(HTTP/1.1、HTTP/1.0)
              ${pageContext.request.remoteUser}         取得用戶名稱
              ${pageContext.request.remoteAddr }         取得用戶的IP 地址
              ${pageContext.session.new}             判斷session 是否為新的
              ${pageContext.session.id}               取得session 的ID
              ${pageContext.servletContext.serverInfo}   取得主機(jī)端的服務(wù)信息

          三、EL運(yùn)算符
            1.算術(shù)運(yùn)算符有五個(gè):+、-、*或$、/或div、%或mod
            2.關(guān)系運(yùn)算符有六個(gè):==或eq、!=或ne、<或lt、>或gt、<=或le、>=或ge
            3.邏輯運(yùn)算符有三個(gè):&&或and、||或or、!或not
            4.其它運(yùn)算符有三個(gè):Empty運(yùn)算符、條件運(yùn)算符、()運(yùn)算符
              例:${empty param.name}、${A?B:C}、${A*(B+C)}
           
          四、EL函數(shù)(functions)。
            語(yǔ)法:ns:function( arg1, arg2, arg3 …. argN)
            其中ns為前置名稱(prefix),它必須和taglib 指令的前置名稱一置

          ---------------------------------------------

          補(bǔ)充:

          <%@ taglib prefix="c" uri="http://java.sun.com/jstl/core_rt " %>

          FOREACH:

          <c:forEach items="${messages}"
          var="item"
          begin="0"
          end="9"
          step="1"
          varStatus="var">
          ……
          </c:forEach>


          OUT:

          <c:out value="${logininfo.username}"/>
          c:out>將value 中的內(nèi)容輸出到當(dāng)前位置,這里也就是把logininfo 對(duì)象的
          username屬性值輸出到頁(yè)面當(dāng)前位置。
          ${……}是JSP2.0 中的Expression Language(EL)的語(yǔ)法。它定義了一個(gè)表達(dá)式,
          其中的表達(dá)式可以是一個(gè)常量(如上),也可以是一個(gè)具體的表達(dá)語(yǔ)句(如forEach循環(huán)體中
          的情況)。典型案例如下:
          Ø ${logininfo.username}
          這表明引用logininfo 對(duì)象的username 屬性。我們可以通過(guò)“.”操作符引
          用對(duì)象的屬性,也可以用“[]”引用對(duì)象屬性,如${logininfo[username]}
          與${logininfo.username}達(dá)到了同樣的效果。
          “[]”引用方式的意義在于,如果屬性名中出現(xiàn)了特殊字符,如“.”或者“-”,
          此時(shí)就必須使用“[]”獲取屬性值以避免語(yǔ)法上的沖突(系統(tǒng)開發(fā)時(shí)應(yīng)盡量避免
          這一現(xiàn)象的出現(xiàn))。
          與之等同的JSP Script大致如下:
          LoginInfo logininfo =
          (LoginInfo)session.getAttribute(“logininfo”);
          String username = logininfo.getUsername();
          可以看到,EL大大節(jié)省了編碼量。
          這里引出的另外一個(gè)問(wèn)題就是,EL 將從哪里找到logininfo 對(duì)象,對(duì)于
          ${logininfo.username}這樣的表達(dá)式而言,首先會(huì)從當(dāng)前頁(yè)面中尋找之前是
          否定義了變量logininfo,如果沒(méi)有找到則依次到Request、Session、
          Application 范圍內(nèi)尋找,直到找到為止。如果直到最后依然沒(méi)有找到匹配的
          變量,則返回null.
          如果我們需要指定變量的尋找范圍,可以在EL表達(dá)式中指定搜尋范圍:
          ${pageScope.logininfo.username}
          ${requestScope.logininfo.username}
          ${sessionScope.logininfo.username}
          ${applicationScope.logininfo.username}
          在Spring 中,所有邏輯處理單元返回的結(jié)果數(shù)據(jù),都將作為Attribute 被放
          置到HttpServletRequest 對(duì)象中返回(具體實(shí)現(xiàn)可參見(jiàn)Spring 源碼中
          org.springframework.web.servlet.view.InternalResourceView.
          exposeModelAsRequestAttributes方法的實(shí)現(xiàn)代碼),也就是說(shuō)Spring
          MVC 中,結(jié)果數(shù)據(jù)對(duì)象默認(rèn)都是requestScope。因此,在Spring MVC 中,
          以下尋址方法應(yīng)慎用:
          ${sessionScope.logininfo.username}
          ${applicationScope.logininfo.username}
          Ø ${1+2}
          結(jié)果為表達(dá)式計(jì)算結(jié)果,即整數(shù)值3。
          Ø ${i>1}
          如果變量值i>1的話,將返回bool類型true。與上例比較,可以發(fā)現(xiàn)EL會(huì)自
          動(dòng)根據(jù)表達(dá)式計(jì)算結(jié)果返回不同的數(shù)據(jù)類型。
          表達(dá)式的寫法與java代碼中的表達(dá)式編寫方式大致相同。

          IF / CHOOSE:

          <c:if test="${var.index % 2 == 0}">
          *
          </c:if>
          判定條件一般為一個(gè)EL表達(dá)式。
          <c:if>并沒(méi)有提供else子句,使用的時(shí)候可能有些不便,此時(shí)我們可以通過(guò)<c:choose>
          tag來(lái)達(dá)到類似的目的:
          <c:choose>
          <c:when test="${var.index % 2 == 0}">
          *
          </c:when>
          <c:otherwise>
          !
          </c:otherwise>
          </c:choose>
          類似Java 中的switch 語(yǔ)句,<c:choose>提供了復(fù)雜判定條件下的簡(jiǎn)化處理手法。其
          中<c:when>子句類似case子句,可以出現(xiàn)多次。上面的代碼,在奇數(shù)行時(shí)輸出“*”號(hào),
          而偶數(shù)行時(shí)輸出“!”。
          ---------------------------------------------

          再補(bǔ)充:

           1    EL表達(dá)式用${}表示,可用在所有的HTML和JSP標(biāo)簽中 作用是代替JSP頁(yè)面中復(fù)雜的JAVA代碼.

                  2   EL表達(dá)式可操作常量 變量 和隱式對(duì)象. 最常用的 隱式對(duì)象有${param}和${paramValues}. ${param}表示返回請(qǐng)求參數(shù)中單個(gè)字符串的值. ${paramValues}表示返回請(qǐng)求參數(shù)的一組值.pageScope表示頁(yè)面范圍的變量.requestScope表示請(qǐng)求對(duì)象的變量. sessionScope表示會(huì)話范圍內(nèi)的變量.applicationScope表示應(yīng)用范圍的變量.

                  3   <%@  page isELIgnored="true"%> 表示是否禁用EL語(yǔ)言,TRUE表示禁止.FALSE表示不禁止.JSP2.0中默認(rèn)的啟用EL語(yǔ)言.

                  4   EL語(yǔ)言可顯示 邏輯表達(dá)式如${true and false}結(jié)果是false    關(guān)系表達(dá)式如${5>6} 結(jié)果是false     算術(shù)表達(dá)式如 ${5+5} 結(jié)果是10

                  5   EL中的變量搜索范圍是:page request session application   點(diǎn)運(yùn)算符(.)和"[ ]"都是表示獲取變量的值.區(qū)別是[ ]可以顯示非詞類的變量


          本文來(lái)自CSDN博客,轉(zhuǎn)載請(qǐng)標(biāo)明出處:http://blog.csdn.net/stonec/archive/2009/10/09/4647394.aspx

          posted @ 2009-11-23 10:53 jimphei 閱讀(284) | 評(píng)論 (0)編輯 收藏

           sitemesh應(yīng)用Decorator模式,用filter截取request和response,把頁(yè)面組件head,content,banner結(jié)合為一個(gè)完整的視圖。通常我們都是用include標(biāo)簽在每個(gè)jsp頁(yè)面中來(lái)不斷的包含各種header, stylesheet, scripts and footer,現(xiàn)在,在sitemesh的幫助下,我們可以開心的刪掉他們了。如下圖,你想輕松的達(dá)到復(fù)合視圖模式,那末看完本文吧。

          一、在WEB-INF/web.xml中copy以下filter的定義:

          <?xml version="1.0" encoding="GBK"?>
          <web-app xmlns="http://java.sun.com/xml/ns/j2ee"
          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
          xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"
          version="2.4">

          <filter>
            <filter-name>sitemesh</filter-name>
               <filter-class>com.opensymphony.module.sitemesh.filter.PageFilter</filter-class>
            </filter>

            <filter-mapping>
               <filter-name>sitemesh</filter-name>
               <url-pattern>/*</url-pattern>
            </filter-mapping>

          </web-app>

          二、copy所需sitemesh-2.3.jar到WEB-INF\lib下。(這里可以下載http://www.opensymphony.com/sitemesh/)

          三、
          建立WEB-INF/decorators.xml描述各裝飾器頁(yè)面。

          <decorators defaultdir="/decorators">
                                         <decorator name="main" page="main.jsp">
                                             <pattern>*</pattern>
                                         </decorator>
                                  </decorators>

            上面配置文件指定了裝飾器頁(yè)面所在的路徑,并指定了一個(gè)名為main的裝飾器,該裝飾器默認(rèn)裝飾web應(yīng)用根路徑下的所有頁(yè)面。

          四、
          建立裝飾器頁(yè)面 /decorators/main.jsp

        1. <%@ page contentType="text/html; charset=GBK"%>
          <%@ taglib uri="http://www.opensymphony.com/sitemesh/decorator" prefix="decorator"%> <html>
                <head>
                    <title><decorator:title default="裝飾器頁(yè)面..." /></title>
                    <decorator:head />
                </head>
               <body>
                  sitemesh的例子<hr>
                  <decorator:body />
                  <hr>chen56@msn.com
              </body>
          </html>
           

          五、建立一個(gè)的被裝飾頁(yè)面 /index.jsp(內(nèi)容頁(yè)面)

        2. <%@ page contentType="text/html; charset=GBK"%>
                                  <html>
                                       <head>
                                         <title>Agent Test</title>
                                       </head>
                                       <body>
                                         <p>本頁(yè)只有一句,就是本句.</p>
                                       </body>
                                  </html>

            最后訪問(wèn)index.jsp,將生成如下頁(yè)面:

                而且,所有的頁(yè)面也會(huì)如同index.jsp一樣,被sitemesh的filter使用裝飾模式修改成如上圖般模樣,卻不用再使用include標(biāo)簽。



          1. 裝飾器     decorator概念
                為了建立可復(fù)用的web應(yīng)用程序,一個(gè)通用的方法是建立一個(gè)分層系統(tǒng),如同下面一個(gè)普通的web應(yīng)用:
            • 前端:JSP和Servlets,或jakarta的velocity 。。。
            • 控制層框架 Controller : (Struts/Webwork)
            • 業(yè)務(wù)邏輯 Business :主要業(yè)務(wù)邏輯
            • 持久化框架 :hibernate/jdo

                可糟糕的是前端的頁(yè)面邏輯很難被復(fù)用,當(dāng)你在每一個(gè)頁(yè)面中用數(shù)之不盡的include來(lái)復(fù)用公共的header, stylesheet, scripts,footer時(shí),一個(gè)問(wèn)題出現(xiàn)了-重復(fù)的代碼,每個(gè)頁(yè)面必須用copy來(lái)復(fù)用頁(yè)面結(jié)構(gòu),而當(dāng)你需要?jiǎng)?chuàng)意性的改變頁(yè)面結(jié)構(gòu)時(shí),災(zāi)難就愛(ài)上了你。

                 sitemesh通過(guò)filter截取request和response,并給原始的頁(yè)面加入一定的裝飾(可能為header,footer...),然后把結(jié)果返回給客戶端,并且被裝飾的原始頁(yè)面并不知道sitemesh的裝飾,這也就達(dá)到了脫耦的目的。

                 據(jù)說(shuō)即將新出臺(tái)的Portlet規(guī)范會(huì)幫助我們標(biāo)準(zhǔn)的實(shí)現(xiàn)比這些更多更c(diǎn)ool的想法,但可憐的我還不懂它到底是一個(gè)什末東東,有興趣的人可以研究
            jetspeed,或JSR (Java Specification Request) 168,但我想sitemesh如此簡(jiǎn)單,我們不妨先用著。

             

            讓我們看看怎樣配置環(huán)境
                除了要copy到WEB-INF/lib中的sitemesh.jar外,還有2個(gè)文件要建立到WEB-INF/:
            • sitemesh.xml (可選)  
            • decorators.xml

            sitemesh.xml 可以設(shè)置2種信息:

            Page Parsers :負(fù)責(zé)讀取stream的數(shù)據(jù)到一個(gè)Page對(duì)象中以被SiteMesh解析和操作。(不太常用,默認(rèn)即可)

            Decorator Mappers : 不同的裝飾器種類,我發(fā)現(xiàn)2種比較有用都列在下面。一種通用的mapper,可以指定裝飾器的配置文件名,另一種可打印的裝飾器,可以允許你當(dāng)用http://localhost/aaa/a.html?printable=true方式訪問(wèn)時(shí)給出原始頁(yè)面以供打印(免得把header,footer等的花哨的圖片也搭上)

            (但一般不用建立它,默認(rèn)設(shè)置足夠了:com/opensymphony/module/sitemesh/factory/sitemesh-default.xml):

            范例:

            <sitemesh>
                 <page-parsers>
                   <parser default="true" class="com.opensymphony.module.sitemesh.parser.DefaultPageParser" />
                   <parser content-type="text/html" class="com.opensymphony.module.sitemesh.parser.FastPageParser" />
                   <parser content-type="text/html;charset=ISO-8859-1" class="com.opensymphony.module.sitemesh.parser.FastPageParser" />
                 </page-parsers>

                 <decorator-mappers>
                   <mapper class="com.opensymphony.module.sitemesh.mapper.ConfigDecoratorMapper">
                     <param name="config" value="/WEB-INF/decorators.xml" />
                   </mapper>
                     <mapper class="com.opensymphony.module.sitemesh.mapper.PrintableDecoratorMapper">
                        <param name="decorator" value="printable" />
                        <param name="parameter.name" value="printable" />
                                <param name="parameter.value" value="true" />
                     </mapper>
              
            </decorator-mappers>
            </sitemesh>

            decorators.xml :定義構(gòu)成復(fù)合視圖的所有頁(yè)面構(gòu)件的描述(主要結(jié)構(gòu)頁(yè)面,header,footer...),如下例:

            <decorators defaultdir="/decorators">
                 <decorator name="main" page="main.jsp">
                   <pattern>*</pattern>
                 </decorator>
                 <decorator name="printable" page="printable.jsp" role="customer" webapp="aaa" />
            </decorators>
            • defaultdir: 包含裝飾器頁(yè)面的目錄
            • page : 頁(yè)面文件名
            • name : 別名
            • role : 角色,用于安全
            • webapp : 可以另外指定此文件存放目錄
            • Patterns : 匹配的路徑,可以用*,那些被訪問(wèn)的頁(yè)面需要被裝飾。

             

            最重要的是寫出裝飾器本身(也就是那些要復(fù)用頁(yè)面,和結(jié)構(gòu)頁(yè)面)。
                其實(shí),重要的工作就是制作裝飾器頁(yè)面本身(也就是包含結(jié)構(gòu)和規(guī)則的頁(yè)面),然后把他們描述到decorators.xml中。

                讓我們來(lái)先看一看最簡(jiǎn)單的用法:其實(shí)最常用也最簡(jiǎn)單的用法就是我們的hello例子,面對(duì)如此眾多的技術(shù),我想只要達(dá)到功能點(diǎn)到為止即可,沒(méi)必要去研究太深(除非您有更深的需求)。

            <%@ page contentType="text/html; charset=GBK"%>
                                        <%@ taglib uri="http://www.opensymphony.com/sitemesh/decorator" prefix="decorator" %>
                                        <html>
                                             <head>
                                               <title><decorator:title default="裝飾器頁(yè)面..." /></title>
                                               <decorator:head />
                                             </head>
                                             <body>
                                               sitemesh的例子<hr>
                                               <decorator:body />
                                               <hr>chen56@msn.com
                                             </body>
                                        </html>
                                        

            我們?cè)谘b飾器頁(yè)面只用了2個(gè)標(biāo)簽:

            <decorator:title default="裝飾器頁(yè)面..." />       : 把請(qǐng)求的原始頁(yè)面的title內(nèi)容插入到<title></title>中間。

            <decorator:body /> : 把請(qǐng)求的原始頁(yè)面的body內(nèi)的全部?jī)?nèi)容插入到相應(yīng)位置。

            然后我們?cè)赿ecorator.xml中加入以下描述即可:

            <decorator name="main" page="main.jsp">
                   <pattern>*</pattern>
            </decorator>

            這樣,請(qǐng)求的所有頁(yè)面都會(huì)被重新處理,并按照main.jsp的格式重新展現(xiàn)在你面前。

             

            讓我們看看更多的用法。(抄襲sitemesh文檔)
            以下列著全部標(biāo)簽:
            Decorator Tags Page Tags
            被用于建立裝飾器頁(yè)面. 被用于從原始內(nèi)容頁(yè)面訪問(wèn)裝飾器.
            <decorator:head />
            <decorator:body />
            <decorator:title />
            <decorator:getProperty />
            <decorator:usePage />
            <page:applyDecorator />
            <page:param
             

            <decorator:head />

            插入原始頁(yè)面(被包裝頁(yè)面)的head標(biāo)簽中的內(nèi)容(不包括head標(biāo)簽本身)。

            <decorator:body />

            插入原始頁(yè)面(被包裝頁(yè)面)的body標(biāo)簽中的內(nèi)容。

            <decorator:title [ default="..." ] />

            插入原始頁(yè)面(被包裝頁(yè)面)的title標(biāo)簽中的內(nèi)容,還可以添加一個(gè)缺省值。

            例:

            /decorator/main.jsp中 (裝飾器頁(yè)面): <title><decorator:title default="卻省title-hello"     /> - 附加標(biāo)題</title>

            /aaa.jsp中 (原始頁(yè)面):<title>aaa頁(yè)面</title>

            訪問(wèn)/aaa.jsp的結(jié)果:<title>aaa頁(yè)面 - 附加標(biāo)題</title>

            <decorator:getProperty property="..." [ default="..." ] [ writeEntireProperty="..." ]/>

            在標(biāo)簽處插入原始頁(yè)面(被包裝頁(yè)面)的原有的標(biāo)簽的屬性中的內(nèi)容,還可以添加一個(gè)缺省值。

            sitemesh文檔中的例子很好理解:
            The decorator: <body bgcolor="white"<decorator:getProperty property="body.onload" writeEntireProperty="true" />>
            The undecorated page: <body onload="document.someform.somefield.focus();">
            The decorated page: <body bgcolor="white" onload="document.someform.somefield.focus();">

            注意,writeEntireProperty="true"會(huì)在插入內(nèi)容前加入一個(gè)空格。

            <decorator:usePage id="..." />
            象jsp頁(yè)面中的<jsp:useBean>標(biāo)簽一樣,可以使用被包裝為一個(gè)Page對(duì)象的頁(yè)面。 (懶的用)

            例:可用<decorator:usePage id="page" /><%=page.getTitle()%>達(dá)到<decorator:title/>的訪問(wèn)結(jié)果。

            <page:applyDecorator name="..." [ page="..." title="..." ] >
            <page:param name="..."> ... </page:param>
            <page:param name="..."> ... </page:param>
            </page:applyDecorator>

            應(yīng)用包裝器到指定的頁(yè)面上,一般用于被包裝頁(yè)面中主動(dòng)應(yīng)用包裝器。這個(gè)標(biāo)簽有點(diǎn)不好理解,我們來(lái)看一個(gè)例子:

            包裝器頁(yè)面 /decorators/panel.jsp:<p><decorator:title /></p>     ... <p><decorator:body /></p>
                 并且在decorators.xml中有<decorator name="panel" page="panel.jsp"/>

            一個(gè)公共頁(yè)面,即將被panel包裝:/public/date.jsp:  
                 ... <%=new java.util.Date()%>     ...<decorator:getProperty property="myEmail" />

            被包裝頁(yè)面 /page.jsp :
                 <title>page的應(yīng)用</title>
                 .....  

                 <page:applyDecorator name="panel" page="/_public/date.jsp" >
                   <page:param name="myEmail"> chen_p@neusoft.com </page:param>
                 </page:applyDecorator>

            最后會(huì)是什末結(jié)果呢?除了/page.jsp會(huì)被默認(rèn)的包裝頁(yè)面包裝上header,footer外,page.jsp頁(yè)面中還內(nèi)嵌了date.jsp頁(yè)面,并且此date.jsp頁(yè)面還會(huì)被panel.jsp包裝為一個(gè)title加body的有2段的頁(yè)面,第1段是date.jsp的title,第2段是date.jsp的body內(nèi)容。

            另外,page:applyDecorator中包含的page:param標(biāo)簽所聲明的屬性值還可以在包裝頁(yè)面中用decorator:getProperty標(biāo)簽訪問(wèn)到。


            可打印的界面裝飾
                 前面說(shuō)過(guò)有1種可打印的裝飾器,可以允許你當(dāng)用http://localhost/aaa/a.html?printable=true方式訪問(wèn)時(shí),應(yīng)用其他的裝飾器(自己指定),給出原始頁(yè)面以供打印(免得把header,footer等的花哨的圖片也搭上)。

            讓我們來(lái)看一看怎樣實(shí)現(xiàn)他:

            1.首先在WEB-INFO/sitemesh.xml中設(shè)置:
                 <mapper class="com.opensymphony.module.sitemesh.mapper.PrintableDecoratorMapper">
                   <param name="decorator" value="printable" />
                   <param name="parameter.name" value="printable" />
                   <param name="parameter.value" value="true" />
                 </mapper>
            這樣就可以通過(guò)?printable=true來(lái)使用名為printable的裝飾器,而不是用原來(lái)的裝飾器。

            2.在WEB-INFO/decorators.xml中定義相應(yīng)的printable裝飾器
                 <decorator name="printable" page="printable.jsp"/>

            3.最后編寫printable裝飾器/decorators/printable.jsp

            <%@ taglib uri="sitemesh-decorator" prefix="decorator" %>
            <html>
            <head>
                 <title><decorator:title /></title>
                 <decorator:head />
            </head>
            <body>

                 <h1><decorator:title /></h1>
                 <p align="right"><i>(printable version)</i></p>

                 <decorator:body />

            </body>
            </html>

            這樣就可以讓一個(gè)原始頁(yè)面通過(guò)?printable=true開關(guān)來(lái)切換不同的裝飾器頁(yè)面。

             

            中文問(wèn)題
            由于sitemesh內(nèi)部所使用的缺省字符集為iso-8859-1,直接使用會(huì)產(chǎn)生亂碼,我們可以通過(guò)以下方法糾正之:
            • 方法1:可以在您所用的application server的配置文件中找一找,有沒(méi)有設(shè)置encoding或charset的項(xiàng)目,然后設(shè)成gbk或gb2312即可
            • 方法2:這也是我們一直使用的方法。
              1.在每一個(gè)jsp頁(yè)里設(shè)置: <%@ page contentType="text/html; charset=gbk"%> 來(lái)告訴server你所要求的字符集。
              2.在每個(gè)jsp頁(yè)的head中定義:<META HTTP-EQUIV="content-type" CONTENT="text/html; charset=gbk"> 來(lái)告訴瀏覽器你所用的字符集。
            總結(jié):使用sitemesh最通常的途徑:

            1.配置好環(huán)境,

            2.在WEB-INFO/decroators.xml中描述你將建立的包裝器。

            3.開發(fā)在decroators.xml中描述的包裝器,最好存放在/_decorators目錄下

            4.ok ,可以看看辛勤的成果了 :)

            posted @ 2009-11-22 09:55 jimphei 閱讀(197) | 評(píng)論 (0)編輯 收藏

            package com.yaday.test;

            import java.io.StringWriter;
            import java.util.Properties;

            import org.apache.velocity.Template;
            import org.apache.velocity.VelocityContext;
            import org.apache.velocity.app.Velocity;
            import org.apache.velocity.app.VelocityEngine;
            import org.junit.Test;

            public class VelocityTest {
                @Test 
            public void testVelocity(){
                    
            try {
                        VelocityEngine ve 
            = new VelocityEngine();
                        ve.setProperty(
            "resource.loader" , "class");
                        ve.setProperty(
            "class.resource.loader.class""org.apache.velocity.runtime.resource.loader.ClasspathResourceLoader");
                        ve.init();
                        Template template
            =null;
                        template
            =ve.getTemplate("velocity/first.vm");

                        VelocityContext context
            =new VelocityContext();
                        context.put(
            "name"new String("jimphei"));
                        
                    
                        

                        StringWriter sw
            =new StringWriter();
                        template.merge(context, sw);
                        System.out.println(sw.toString());
                    }
             catch (Exception e) {
                        
            // TODO Auto-generated catch block
                        e.printStackTrace();
                    }

                }

            }

            posted @ 2009-11-10 15:15 jimphei 閱讀(548) | 評(píng)論 (0)編輯 收藏

            Linux+Apache+PHP+MySQL是一個(gè)低成本效率高而又穩(wěn)定的WEB Server,但是我們絕大部分開發(fā)都是在Windows環(huán)境下完成開發(fā),然后移植到Linux或者Unix下。現(xiàn)在依據(jù)個(gè)人體驗(yàn)來(lái)說(shuō)明一下Windows XP+I(xiàn)IS下安裝Apache2+PHP 5。沒(méi)有IIS安裝就更加簡(jiǎn)單,除去IIS相關(guān)步驟就可以了。

            一、關(guān)閉IIS,如果不關(guān)閉IIS安裝Apache會(huì)出錯(cuò)。apache整合tomcat配置

            關(guān)閉IIS有兩種方法,任意一種都可以:

            1. 控制面板--性能和維護(hù)--管理工具--服務(wù)中,關(guān)閉IIS Admin服務(wù)。
              控制面板--性能和維護(hù)--管理工具--服務(wù)中,關(guān)閉IIS Admin服務(wù)
            2. 在開始--運(yùn)行中直接輸入如下代碼,或者先輸入cmd,在彈出的窗口中輸入也行net stop iisadmin上述命令關(guān)閉了iis相關(guān)的所有服務(wù)器,比如web sites 、smtp等。net stop iisadmin /y避免輸入上面那個(gè)命令后需要在輸入y如果用net stop w3svc只是關(guān)閉一個(gè)站點(diǎn)3w服務(wù)器,但是如果是多個(gè)web站點(diǎn)就不行。

            如果開啟IIS可以在控制面板中找到interet信息服務(wù)打開網(wǎng)站服務(wù)的方法,也可以用命名,net start w3svc都可以。注意如果直接在服務(wù)中打開IIS Admin服務(wù)或者運(yùn)動(dòng)net start iisadmin,是可以打開IIS Admin服務(wù),但是3w服務(wù)沒(méi)有打開,所以依舊需要用上面的方法打開3w服務(wù),因?yàn)樵诖蜷_IIS Admin服務(wù)沒(méi)有打開3w服務(wù),但是打開3w服務(wù)肯定就打開了IIS Admin服務(wù)。

            二、安裝Apache2。

            ps,Apache 2不能在Windows 95上運(yùn)行;在Windows 98上勉強(qiáng)能夠運(yùn)行,但不能作為服務(wù)使用。從4.3版本開始,PHP也不再支持Windows 95。所以,你的Windows操作系統(tǒng)必須是Windows NT、2000或者XP。

            1. Apache可以到http://www.apache.org/dyn/closer.cgi/httpd/binaries/win32/下載
            2. 對(duì)于本機(jī)開發(fā)Network Domain,ServerName都填入localhost就可以了,填入email地址即可。
              安裝apache時(shí)需要填入的信息
            3. 上圖中的單項(xiàng)選擇,對(duì)于初學(xué)者來(lái)說(shuō),不管Apache的服務(wù)是否使用80單口,建議都選第一個(gè),這樣就直接把Apache注冊(cè)為系統(tǒng)服務(wù),穩(wěn)定方便。然后下一步選擇Typical。
            4. 安裝路徑一般會(huì)默認(rèn)為c:\Programme Files\Apache Group改成c:\web或者其他符合8.3格式的名稱,這樣以來(lái)以后每次輸入Apache安裝路徑不用加引號(hào),并且Apache安裝時(shí)會(huì)自動(dòng)生成Apache2文件夾,所以文件會(huì)安裝到c:\web\apache2,這樣以后也可以把PHP,MySQL都安裝到web下便于幾種管理。
            5. 由于Apache&IIS都默認(rèn)WEB服務(wù)端口是80,所以其中一個(gè)必須修改其端口,一般改成8080
              修改IIS端口直接在IIS管理工具中就可以了。可以在控制面板中找,或者在運(yùn)行中輸入inetmgr
              修改Apache端口,通過(guò)開始-所有程序-Apache-Configure Apache Server打開httpd.conf文件,
              找到 #Listen 12.34.56.78:80   #是注釋符號(hào)
                  Listen 80  改成  Listen 8080
                  然后找到  ServerName localhost:80   改成  ServerName localhost:8080  即可
            6. 在瀏覽器中輸入localhost,如果修改了端口就輸入localhost:8080能夠看到apache頁(yè)面,就說(shuō)明安裝成功了。

            ps[2005.9.29].利用apache的proxy模塊實(shí)現(xiàn)隱藏iis的端口

            1. 按照上面的方法,apache使用默認(rèn)端口80,修改iis使用端口為8080,當(dāng)然你也可以采用其他的合理端口。
            2. 修改apache的http.conf文件,去掉下面兩行代碼前的注釋符號(hào)#,啟動(dòng)代理模塊
              LoadModule proxy_module modules/mod_proxy.so
                  LoadModule proxy_http_module modules/mod_proxy_http.so
            3. 在該文件添加上如下兩行代碼,使輸入http://localhost/iis/轉(zhuǎn)向http://localhost:8080
              ProxyPass /iis/ http://127.0.0.1:8080/
                  ProxyPassReverse /iis http://127.0.0.1:8080

              這樣就可以在瀏覽器中輸入localhost訪問(wèn)apache,輸入localhost/iis/訪問(wèn)iis了而隱藏了8080端口

            4. 另外,可以通過(guò)設(shè)置虛擬主機(jī)來(lái)訪問(wèn)apache或者iis
              <VirtualHost *:80>
                  ServerAdmin kavenyan@163.com
                  DocumentRoot E:/www/dancewithnet
                  ServerName www.dancewithnet.com
                  ServerAlias dancewithnet.com
                  DefaultLanguage zh-CN
                  AddDefaultCharset UTF-8
                  </VirtualHost>
                  <VirtualHost *:80>
                  ServerAdmin kavenyan@163.com
                  ServerName iis.dancewithnet.com
                  DefaultLanguage zh-CN
                  AddDefaultCharset GB2312
                  ProxyPass / http://127.0.0.1:8080/  or http://服務(wù)器ip:8080/
                  ProxyPassReverse / http://127.0.0.1:8080/   or http://服務(wù)器ip:8080/
                  </VirtualHost>

              這樣就可以使用www.dancewithnet.com訪問(wèn)apache,iis.dancewithnet.com訪問(wèn)iss,而隱藏了8080端口

              三、配置PHP環(huán)境

              1. www.php.net上下載php5的zip安裝包,將其文件解壓放到c:\web\php5中即可

                ps, Apache 2可采取2種方式來(lái)運(yùn)行PHP程序:通過(guò)一個(gè)CGI接口來(lái)運(yùn)行(外部調(diào)用Php.exe),或者使用PHP的DLL文件在Apache的內(nèi)部運(yùn)行。后一種方式的速度較快。所以,針對(duì)每個(gè)版本的PHP,都會(huì)提供2個(gè)Windows二進(jìn)制發(fā)行包。較小的是.msi包,它會(huì)安裝CGI可執(zhí)行程序Php.exe,但其中拿掉了通過(guò)Apache DLL來(lái)運(yùn)行PHP腳本所需的模塊。較大的.zip包則包含了所有這些東西

              2. 最好是無(wú)論使用何種接口(CGI 或者 SAPI)都確保 php5ts.dll 可用,因此必須將此文件放到 Windows 路徑中。最好的位置是 Windows 的 system 目錄(%windir%\System):
                c:\\winnt\\system32 for Windows NT/2000
                        或者
                        c:\\winnt40\\system32 for Windows NT/2000 服務(wù)器版
                        c:\\windows\\system32 for Windows XP

                ps,也有把php文件中所有的dll文件都拷到%windir%\System中的,那樣的配置和我介紹的方法稍微有點(diǎn)不同,但是我覺(jué)得那樣比較雜亂,就不再說(shuō)明,有興趣的朋友可以自己研究。

              3. 接著實(shí)設(shè)定有效的PHP 配置文件,php.ini。壓縮包中包括兩個(gè) ini 文件,php.ini-dist 和 php.ini-recommended。建議使用 php.ini-recommended,因?yàn)榇宋募?duì)默認(rèn)設(shè)置作了性能和安全上的優(yōu)化。將選擇的 ini 文件拷貝到 PHP 能夠找到的目錄下并改名為 php.ini。PHP 默認(rèn)在 Windows 目錄(%WINDIR% 或 %SYSTEMROOT% )下搜索 php.ini:
                c:\\winnt 或 c:\\winnt40  for Windows NT/2000 服務(wù)器版
                        c:\windows  for Windows XP
                        
              4. 停止Apache,打開httpd.conf進(jìn)行編輯。
                如果是使用CGI二進(jìn)制文件的形式來(lái)使用php,添入代碼如下(注意代碼間的空格):

                 

                ScriptAlias /php/ "c:/web/php5/"
                        AddType application/x-httpd-php .php
                        Action application/x-httpd-php "/php5/php.exe"
                        

                如果作為模塊(推薦這種方式),添加代碼如下:

                LoadModule php5_module "c:/web/php5/php5apache2.dll"
                        AddType application/x-httpd-php .php
                        
              5. 保存httpd.conf,啟動(dòng)Apache

              四、測(cè)試PHP

              1. 編寫文件index.php放入C:\web\Apache2\htdocs中,代碼如下:
                測(cè)試PHP安裝是否成功的代碼
              2. 在瀏覽中輸入http://localhost/index.php效果如下,則說(shuō)明安裝成功:
                php安裝成功出現(xiàn)的頁(yè)面
            posted @ 2009-11-04 10:24 jimphei 閱讀(346) | 評(píng)論 (0)編輯 收藏

            這個(gè)是mysql版本不同的問(wèn)題

            posted @ 2009-11-03 15:50 jimphei 閱讀(189) | 評(píng)論 (0)編輯 收藏

            1、后臺(tái)ajax在應(yīng)用(特別是提交中文時(shí)要用encodeURI(encodeURI(typename))提交,然后后臺(tái)用URLDecoder.decode(strtypename, "utf-8")取值。
            2、java-fckeditor在應(yīng)用與配置。
            3、jquery的應(yīng)用。
            4、二級(jí)目錄與多級(jí)目錄的學(xué)習(xí)。
            5、驗(yàn)證碼生成技術(shù)。
            posted @ 2009-11-03 09:35 jimphei 閱讀(148) | 評(píng)論 (0)編輯 收藏

            Struts2+JQuery+JSON集成


            細(xì)節(jié)部分我就不多講了,因?yàn)槲乙膊粫?huì),就講講我是如何調(diào)試出來(lái)我的第一個(gè)JSON使用的吧

            采用的框架有:Struts2 、 JQuery 、 JSON


            按著步驟來(lái)吧:


             1.新建一個(gè)Web工程


            導(dǎo)入包列表:

             


             目錄結(jié)構(gòu)如圖:

             


             2.建立實(shí)體類User

            package model;


            public class User


            private String name;

            private int age;

             //省略相應(yīng)的get和set方法
             


             3.建立Action JsonAction

            public class JsonAction extends ActionSupport{

            private static final long serialVersionUID =

             7044325217725864312L;


            private User user;

            //用于記錄返回結(jié)果

            private String result;


            //省略相應(yīng)的get和set方法


            @SuppressWarnings("static-access")


            public String execute() throws Exception {


            //將要返回的user實(shí)體對(duì)象進(jìn)行json處理

            JSONObject jo = JSONObject.fromObject(this.user);

            //打印一下,格式如下

            //{"name":"風(fēng)達(dá)","age":23}

            System.out.println(jo);


            //調(diào)用json對(duì)象的toString方法轉(zhuǎn)換為字符串然后賦值給result

            this.result = jo.toString();


            return this.SUCCESS;

            }


            }
             


             4.建立struts.xml文件

            <?xml version="1.0" encoding="UTF-8"?>

            <!DOCTYPE struts PUBLIC

                "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"

                "http://struts.apache.org/dtds/struts-2.0.dtd">


            <struts>

            <constant name="struts.i18n.encoding" value="UTF-8"></constant>

            <package name="ttttt" extends="json-default">

            <action name="jsonAction" class="action.JsonAction">

            <result type="json" >

            <!-- 因?yàn)橐獙eslut的值返回給客戶端,所以這樣設(shè)置 -->

            <!-- root的值對(duì)應(yīng)要返回的值的屬性 -->

            <param name="root">

            result

            </param>

            </result>

            </action>

            </package>

            </struts>

             


             5.編寫index.jsp文件

            <%@ page language="java" pageEncoding="UTF-8"%>

            <%@ taglib prefix="s" uri="/struts-tags"%>

            <%

            String path = request.getContextPath();

            String basePath = request.getScheme() + "://"

            + request.getServerName() + ":" + request.getServerPort()

            + path + "/";

            %>


            <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">

            <html>

            <head>

            <base href="<%=basePath%>">


            <title>My JSP 'index.jsp' starting page</title>

            <meta http-equiv="pragma" content="no-cache">

            <meta http-equiv="cache-control" content="no-cache">

            <meta http-equiv="expires" content="0">

            <meta http-equiv="keywords" content="keyword1,keyword2,keyword3">

            <meta http-equiv="description" content="This is my page">


            <!-- basePath用來(lái)獲取js文件的絕對(duì)路徑 -->

            <script type="text/javascript" src="<%=basePath%>js/jquery.js"></script>

            <script type="text/javascript" src="<%=basePath%>js/index.js"></script>

            <s:head theme="ajax" />

            </head>


            <body>

            <div id="result">

            </div>

            <s:form name="userForm" action="" method="post">

            <s:textfield label="用戶名" name="user.name" />

            <s:textfield label="年齡" name="user.age" />

            <button>

            提交

            </button>

            </s:form>


            </body>

            </html>

             


             6.在WebRoot目錄下建立js文件件,將jquery.js文件放到文件夾下,然后再建立文件index.js


            $(document).ready(function() {


            // 直接把onclick事件寫在了JS中

            $("button").click(function() {

            // 序列化表單的值

            var params = $("input").serialize();


            $.ajax({


            // 后臺(tái)處理程序

            url : "jsonAction.action",


            // 數(shù)據(jù)發(fā)送方式

            type : "post",


            // 接受數(shù)據(jù)格式

            dataType : "json",


            // 要傳遞的數(shù)據(jù)

            data : params,


            // 回傳函數(shù)

            success : update_page


            });

            });


            });

            function update_page(result) {

            var json = eval( "("+result+")" );

            var str = "姓名:" + json.name + "<br />"; str += "年齡:"

            + json.age + "<br />";

            $("#result").html(str);


            }
             


             7.運(yùn)行前效果:

             

            要的是效果,布局就不整了

             


            運(yùn)行后效果:

             

             


            網(wǎng)上相關(guān)的信息太少了,很多Struts2+JQuery+JSON的教程,點(diǎn)開鏈接之后都是那幾篇文章轉(zhuǎn)了又轉(zhuǎn),遇到問(wèn)題真的很想找到有用的信息,或許是我太笨了,找不到,或許就是網(wǎng)上相關(guān)的信息就很少。這個(gè)實(shí)例很簡(jiǎn)單是不是,但是為了調(diào)試出這個(gè)程序,我費(fèi)了一天的時(shí)間。


            上面的實(shí)例成功了,但是問(wèn)題又出來(lái)了

            視圖類型僅僅設(shè)置了json

            那么輸入校驗(yàn)出錯(cuò)的時(shí)候怎么顯示?


            本文來(lái)自CSDN博客,轉(zhuǎn)載請(qǐng)標(biāo)明出處:http://blog.csdn.net/fengda2870/archive/2009/04/06/4052527.aspx

            posted @ 2009-09-30 12:52 jimphei 閱讀(830) | 評(píng)論 (0)編輯 收藏

            今天看到某人寫的分頁(yè)類,結(jié)果發(fā)現(xiàn)批人家的人不少,沒(méi)有必要,好的東西吸收學(xué)習(xí),感覺(jué)不實(shí)用可以不用,何必發(fā)帖子挖苦人家。我前段時(shí)間也自己設(shè)計(jì)了一個(gè)分頁(yè)的方法,絕對(duì)是自己想到的,如果網(wǎng)上有一樣的,說(shuō)明大家都思考了,有可取度,提供給大家參考。
            首先寫了一個(gè)分頁(yè)的類,其實(shí)只有主要屬性的setter和getter方法
            /**
            * 分頁(yè)類
            * @author qinglin876
            *
            */
            public class Pagination {
            private int start;
            //一次取得的數(shù)量
            private int size;
            //要取得頁(yè)數(shù)
            private int currentPage = 1;
            //總的記錄頁(yè)數(shù)
            private int totalPage = 0;
            //總的記錄條數(shù)
            private int totalRecord;
            public int getTotalRecord() {
            return totalRecord;
            }
            public void setTotalRecord(int totalRecord) {
            this.totalRecord = totalRecord;
            }
            public Pagination(){

            }
            public Pagination(int size){
            this.size = size;
            }
            public int getSize() {
            return size;
            }
            public void setSize(int size) {
            this.size = size;
            }
            public int getStart() {
            return start;
            }
            public void setStart(int start) {
            this.start = start;
            }
            public int getCurrentPage() {
            return currentPage;
            }
            public void setCurrentPage(int currentPage) {
            this.currentPage = currentPage;
            }
            public int getTotalPage() {
            return totalPage;
            }
            public void setTotalPage(int totalPage) {
            this.totalPage = totalPage;
            }

            }

            另外pagination.jsp由pagination類填充

            <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>

            <%@ taglib prefix="s" uri="/struts-tags"%>
            <SCRIPT type="text/javascript">

                  function trim(str){
            return str.replace(/(^\s*)|(\s*$)/g, "");
            }

            function selectPage(input){

            var value = trim(input.value);
            if(value == ""){
            return;
            }

            if(/\d+/.test(value)){

            input.form.submit();
            return;
            }
            alert("請(qǐng)輸入正確的頁(yè)數(shù)");
            input.focus();

            }
               
            </SCRIPT>
            <div class="pagech">

            <s:if test="pagination.totalPage != 0">
            <s:url action="%{#request.url}" id="first">
            <s:param name="pagination.currentPage" value="1"></s:param>
            </s:url>
            <s:url action="%{#request.url}" id="next"  >
            <s:param name="pagination.currentPage"
            value="pagination.currentPage+1">
            </s:param>
            </s:url>
            <s:url action="%{#request.url}" id="prior" >
            <s:param name="pagination.currentPage"
            value="pagination.currentPage-1"></s:param>
            </s:url>
            <s:url action="%{#request.url}" id="last">
            <s:param name="pagination.currentPage" value="pagination.totalPage"></s:param>
            </s:url>
            <s:if test="pagination.currentPage == 1">
            <span class="current">首頁(yè)</span>
            <span class="current">上一頁(yè)</span>
            </s:if>
            <s:else>
            <s:a href="%{first}">首頁(yè)</s:a>
            <s:a href="%{prior}">上一頁(yè)</s:a>
            </s:else>
            <s:if
            test="pagination.currentPage == pagination.totalPage || pagination.totalPage == 0">
            <span class="current">下一頁(yè)</span>
            <span class="current">末頁(yè)</span>
            </s:if>
            <s:else>
            <s:a href="%{next}">下一頁(yè)</s:a>&nbsp;&nbsp;
                              <s:a href="%{last}">末頁(yè)</s:a>
            </s:else>
            <span class="jumplabel">跳轉(zhuǎn)到</span>
            <s:form action="%{#request.url}" theme="simple"
            cssStyle="display:inline">
            <s:hidden name="pagination.totalPage" value="%{pagination.totalPage}"></s:hidden>
            <input type="text" name="pagination.currentPage" size="2"
            onblur="selectPage(this)" />
            </s:form>

            <span class="jumplabel">頁(yè)</span>
            <span class="jumplabel">共<s:property
            value="pagination.totalRecord" />條</span>
            <span class="jumplabel">當(dāng)前是第<s:property
            value="pagination.currentPage" />/<s:property value="pagination.totalPage"/>頁(yè)</span>


            </s:if>

            </div>

            用的時(shí)候,在頁(yè)面include進(jìn)去,注意上面的"%{#request.url}",即是在struts2的action里面有一個(gè)setter和getter方法,下面看action中的某個(gè)方法
            public String showNotices() throws Exception{

            if(tip != null){
            tip = new String(tip.getBytes("iso8859-1"),"utf-8");
            }
            if(notices == null)
            this.notices = new ArrayList<Notice>();
            int size = Integer.valueOf(this.getText("per_page_notice_size"));
            if (pagination == null) {
            pagination = new Pagination(size);
            }
            pagination.setSize(size);
            if (pagination.getCurrentPage() <= 0) {
            pagination.setCurrentPage(1);
            }
            if (pagination.getTotalPage() != 0
            && pagination.getCurrentPage() > pagination.getTotalPage()) {
            pagination.setCurrentPage(pagination.getTotalPage());
            }
            url = "goto_showNotices.action"; this.notices.addAll(this.noticeDAO.showAll(pagination));
            if(this.notices.size() == 0 && pagination.getCurrentPage() != 1){
            pagination.setCurrentPage(pagination.getCurrentPage()-1);
            this.notices.addAll(this.noticeDAO.showAll(pagination));
            }
            return "success";
            }

            在上面的this.noticeDAO.showAll(pagination))中填充pagination,具體如下
            /*
            * 顯示所有的通告
            * @see com.qinglin.dao.NoticeDAO#showAll(com.qinglin.util.Pagination)
            */
            @SuppressWarnings("unchecked")
            public List<Notice> showAll(final Pagination pagination) {
            String hql = "from Notice as n";
            this.getHibernateTemplate().setCacheQueries(true);
            int totalRecord = ((Long) this.getSession().createQuery(
            "select count(*) " + hql).uniqueResult()).intValue();
            int totalPage = totalRecord % pagination.getSize() == 0 ? totalRecord
            / pagination.getSize() : totalRecord / pagination.getSize() + 1;
            pagination.setTotalRecord(totalRecord);
            pagination.setTotalPage(totalPage);
            hql += " order by n.add_date desc";
            final String hql1 = hql;

            return (List<Notice>) this.getHibernateTemplate().execute(
            new HibernateCallback() {
            public Object doInHibernate(Session session)
            throws HibernateException, SQLException {
            Query query = session.createQuery(hql1);
            query.setFirstResult((pagination.getCurrentPage() - 1)
            * pagination.getSize());
            query.setMaxResults(pagination.getSize());
            return query.list();
            }
            });
            }


            基本上就這些,當(dāng)然請(qǐng)求的action里面需要設(shè)置pagination的setter和getter方法
            這個(gè)分頁(yè)方法特點(diǎn)是簡(jiǎn)單,只需在action中指明請(qǐng)求的url,用某種方法填充pagination,在顯示的頁(yè)面包含pagination.jsp即可。



            package com.shop.bean;

            import java.util.List;

            public class PageView <T> {

            private int currentPage = 1;

            private long totalPage = 1;

            private long totalRecord = 1;

            private List <T> records;

            private int firstIndex = 1;

            private PageIndex pageIndex;

            private int maxResult = 12;

            public PageView(int currentPage, int maxResult) {
            this.currentPage = currentPage;
            this.maxResult = maxResult;
            this.firstIndex = currentPage * maxResult;
            }

            public int getCurrentPage() {
            return currentPage;
            }

            public void setCurrentPage(int currentPage) {
            this.currentPage = currentPage;
            }

            public void setQueryResult(QueryResult <T> qr){
            setTotalRecord(qr.getTotal());
            setRecords(qr.getDatas());
            }

            public long getTotalPage() {
            return totalPage;
            }

            public void setTotalPage(long totalPage) {
            this.totalPage = totalPage;
            this.pageIndex = WebTool.getPageIndex(this.maxResult, this.currentPage, this.totalPage);
            }

            public long getTotalRecord() {
            return totalRecord;
            }

            public void setTotalRecord(long totalRecord) {
            this.totalRecord = totalRecord;
            setTotalPage(totalRecord / this.maxResult == 0 ? totalRecord / this.maxResult : totalRecord / this.maxResult + 1);
            }

            public List <T> getRecords() {
            return records;
            }

            public void setRecords(List <T> records) {
            this.records = records;
            }

            public int getFirstIndex() {
            return firstIndex;
            }
            public PageIndex getPageIndex() {
            return pageIndex;
            }

            public void setPageIndex(PageIndex pageIndex) {
            this.pageIndex = pageIndex;
            }

            public int getMaxResult() {
            return maxResult;
            }

            public void setMaxResult(int maxResult) {
            this.maxResult = maxResult;
            }

            public void setFirstIndex(int firstIndex) {
            this.firstIndex = firstIndex;
            }
            }


            畫面的代碼:
            <s:iterator value="#request.pageView.pageIndex.pageList">
                  <s:if test="#request.pageView.currentPage == 4"> <b> <font color="#FFFFFF">第 <s:property/>頁(yè) </font> </b> </s:if>
                <s:if test="#request.pageView.currentPage != 4"> <a href="javascript:topage( <s:property/>)" class="a03">第 <s:property/>頁(yè) </a> </s:if>
            </s:iterator>

            action中的代碼:
            Map <String, Object> request = (Map <String, Object>)ActionContext.getContext().get("request");
            request.put("pageView", pageView);




            <s:iterator value="#request.pageView.pageIndex.pageList">中="#request.pageView.pageIndex.pageList值能正常獲取,可是  <s:if test="#request.pageView.currentPage == 4"> 中的="#request.pageView.currentPage值獲取不到正確的值,這是什么原因???
            問(wèn)題補(bǔ)充:
              <s:iterator value="#request.pageView.pageIndex.pageList">
                <s:if test="{#request.pageView.currentPage == 4}"><b><font color="#FFFFFF">第<s:property/>頁(yè)</font></b></s:if>
                <s:if test="{#request.pageView.currentPage != 4}"><a href="javascript:topage(<s:property/>)" class="a03">第<s:property/>頁(yè)</a></s:if>
            </s:iterator>
            posted @ 2009-09-22 12:03 jimphei 閱讀(211) | 評(píng)論 (0)編輯 收藏

            在地址欄輸入javascript:document.body.contentEditable='true'; document.designMode='on'; void 0就可以編輯網(wǎng)頁(yè)了,哈哈
            posted @ 2009-09-05 21:01 jimphei 閱讀(123) | 評(píng)論 (0)編輯 收藏

            posted @ 2009-08-27 15:41 jimphei 閱讀(52) | 評(píng)論 (0)編輯 收藏

            1 定義頭和根元素
                  部署描述符文件就像所有XML文件一樣,必須以一個(gè)XML頭開始。這個(gè)頭聲明可以使用的XML版本并給出文件的字符編碼。

                  DOCYTPE聲明必須立即出現(xiàn)在此頭之后。這個(gè)聲明告訴服務(wù)器適用的servlet規(guī)范的版本(如2.2或2.3)并指定管理此文件其余部分內(nèi)容的語(yǔ)法的DTD(Document Type Definition,文檔類型定義)。

                  所有部署描述符文件的頂層(根)元素為web-app。請(qǐng)注意,XML元素不像HTML,他們是大小寫敏感的。因此,web-App和WEB-APP都是不合法的,web-app必須用小寫。

                   2 部署描述符文件內(nèi)的元素次序

                   XML元素不僅是大小寫敏感的,而且它們還對(duì)出現(xiàn)在其他元素中的次序敏感。例如,XML頭必須是文件中的第一項(xiàng),DOCTYPE聲明必須是第二項(xiàng),而web-app元素必須是第三項(xiàng)。在web-app元素內(nèi),元素的次序也很重要。服務(wù)器不一定強(qiáng)制要求這種次序,但它們?cè)试S(實(shí)際上有些服務(wù)器就是這樣做的)完全拒絕執(zhí)行含有次序不正確的元素的Web應(yīng)用。這表示使用非標(biāo)準(zhǔn)元素次序的web.xml文件是不可移植的。

                  下面的列表給出了所有可直接出現(xiàn)在web-app元素內(nèi)的合法元素所必需的次序。例如,此列表說(shuō)明servlet元素必須出現(xiàn)在所有servlet-mapping元素之前。請(qǐng)注意,所有這些元素都是可選的。因此,可以省略掉某一元素,但不能把它放于不正確的位置。

            icon icon元素指出IDE和GUI工具用來(lái)表示W(wǎng)eb應(yīng)用的一個(gè)和兩個(gè)圖像文件的位置。
            display-name display-name元素提供GUI工具可能會(huì)用來(lái)標(biāo)記這個(gè)特定的Web應(yīng)用的一個(gè)名稱。
            description description元素給出與此有關(guān)的說(shuō)明性文本。
            context-param context-param元素聲明應(yīng)用范圍內(nèi)的初始化參數(shù)。
            filter 過(guò)濾器元素將一個(gè)名字與一個(gè)實(shí)現(xiàn)javax.servlet.Filter接口的類相關(guān)聯(lián)。
            filter-mapping 一旦命名了一個(gè)過(guò)濾器,就要利用filter-mapping元素把它與一個(gè)或多個(gè)servlet或JSP頁(yè)面相關(guān)聯(lián)。
            listener servlet API的版本2.3增加了對(duì)事件監(jiān)聽程序的支持,事件監(jiān)聽程序在建立、修改和刪除會(huì)話或servlet環(huán)境時(shí)得到通知。Listener元素指出事件監(jiān)聽程序類。
            servlet 在向servlet或JSP頁(yè)面制定初始化參數(shù)或定制URL時(shí),必須首先命名servlet或JSP頁(yè)面。Servlet元素就是用來(lái)完成此項(xiàng)任務(wù)的。
            servlet-mapping 服務(wù)器一般為servlet提供一個(gè)缺省的URL:http://host/webAppPrefix/servlet/ServletName。但是,常常會(huì)更改這個(gè)URL,以便servlet可以訪問(wèn)初始化參數(shù)或更容易地處理相對(duì)URL。在更改缺省URL時(shí),使用servlet-mapping元素。
            session-config 如果某個(gè)會(huì)話在一定時(shí)間內(nèi)未被訪問(wèn),服務(wù)器可以拋棄它以節(jié)省內(nèi)存??赏ㄟ^(guò)使用HttpSession的setMaxInactiveInterval方法明確設(shè)置單個(gè)會(huì)話對(duì)象的超時(shí)值,或者可利用session-config元素制定缺省超時(shí)值。
            mime-mapping 如果Web應(yīng)用具有想到特殊的文件,希望能保證給他們分配特定的MIME類型,則mime-mapping元素提供這種保證。
            welcom-file-list welcome-file-list元素指示服務(wù)器在收到引用一個(gè)目錄名而不是文件名的URL時(shí),使用哪個(gè)文件。
            error-page error-page元素使得在返回特定HTTP狀態(tài)代碼時(shí),或者特定類型的異常被拋出時(shí),能夠制定將要顯示的頁(yè)面。
            taglib taglib元素對(duì)標(biāo)記庫(kù)描述符文件(Tag Libraryu Descriptor file)指定別名。此功能使你能夠更改TLD文件的位置,而不用編輯使用這些文件的JSP頁(yè)面。
            resource-env-ref resource-env-ref元素聲明與資源相關(guān)的一個(gè)管理對(duì)象。
            resource-ref resource-ref元素聲明一個(gè)資源工廠使用的外部資源。
            security-constraint security-constraint元素制定應(yīng)該保護(hù)的URL。它與login-config元素聯(lián)合使用
            login-config 用login-config元素來(lái)指定服務(wù)器應(yīng)該怎樣給試圖訪問(wèn)受保護(hù)頁(yè)面的用戶授權(quán)。它與sercurity-constraint元素聯(lián)合使用。
            security-role security-role元素給出安全角色的一個(gè)列表,這些角色將出現(xiàn)在servlet元素內(nèi)的security-role-ref元素的role-name子元素中。分別地聲明角色可使高級(jí)IDE處理安全信息更為容易。
            env-entry env-entry元素聲明Web應(yīng)用的環(huán)境項(xiàng)。
            ejb-ref ejb-ref元素聲明一個(gè)EJB的主目錄的引用。
            ejb-local-ref ejb-local-ref元素聲明一個(gè)EJB的本地主目錄的應(yīng)用。

                  3 分配名稱和定制的UL

                  在web.xml中完成的一個(gè)最常見(jiàn)的任務(wù)是對(duì)servlet或JSP頁(yè)面給出名稱和定制的URL。用servlet元素分配名稱,使用servlet-mapping元素將定制的URL與剛分配的名稱相關(guān)聯(lián)。

                  3.1 分配名稱

                  為了提供初始化參數(shù),對(duì)servlet或JSP頁(yè)面定義一個(gè)定制URL或分配一個(gè)安全角色,必須首先給servlet或JSP頁(yè)面一個(gè)名稱。可通過(guò)servlet元素分配一個(gè)名稱。最常見(jiàn)的格式包括servlet-name和servlet-class子元素(在web-app元素內(nèi)),如下所示:

            <servlet>
            <servlet-name>Test</servlet-name>
            <servlet-class>moreservlets.TestServlet</servlet-class>
            </servlet>

                  這表示位于WEB-INF/classes/moreservlets/TestServlet的servlet已經(jīng)得到了注冊(cè)名Test。給servlet一個(gè)名稱具有兩個(gè)主要的含義。首先,初始化參數(shù)、定制的URL模式以及其他定制通過(guò)此注冊(cè)名而不是類名引用此servlet。其次,可在URL而不是類名中使用此名稱。因此,利用剛才給出的定義,URL http://host/webAppPrefix/servlet/Test 可用于 http://host/webAppPrefix/servlet/moreservlets.TestServlet 的場(chǎng)所。

             

            請(qǐng)記住:XML元素不僅是大小寫敏感的,而且定義它們的次序也很重要。例如,web-app元素內(nèi)所有servlet元素必須位于所有servlet-mapping元素(下一小節(jié)介紹)之前,而且還要位于5.6節(jié)和5.11節(jié)討論的與過(guò)濾器或文檔相關(guān)的元素(如果有的話)之前。類似地,servlet的servlet-name子元素也必須出現(xiàn)在servlet-class之前。5.2節(jié)"部署描述符文件內(nèi)的元素次序"將詳細(xì)介紹這種必需的次序。
                  例如,程序清單5-1給出了一個(gè)名為TestServlet的簡(jiǎn)單servlet,它駐留在moreservlets程序包中。因?yàn)榇藄ervlet是扎根在一個(gè)名為deployDemo的目錄中的Web應(yīng)用的組成部分,所以TestServlet.class放在deployDemo/WEB-INF/classes/moreservlets中。程序清單5-2給出將放置在deployDemo/WEB-INF/內(nèi)的web.xml文件的一部分。此web.xml文件使用servlet-name和servlet-class元素將名稱Test與TestServlet.class相關(guān)聯(lián)。圖5-1和圖5-2分別顯示利用缺省URL和注冊(cè)名調(diào)用TestServlet時(shí)的結(jié)果。

            程序清單5-1 TestServlet.java
            package moreservlets;

            import java.io.*;
            import javax.servlet.*;
            import javax.servlet.http.*;

            /** Simple servlet used to illustrate servlet naming
            * and custom URLs.
            * <P>
            * Taken from More Servlets and JavaServer Pages
            * from Prentice Hall and Sun Microsystems Press,
            * http://www.moreservlets.com/.
            * © 2002 Marty Hall; may be freely used or adapted.
            */

            public class TestServlet extends HttpServlet {
            public void doGet(HttpServletRequest request,
            HttpServletResponse response)
            throws ServletException, IOException {
            response.setContentType("text/html");
            PrintWriter out = response.getWriter();
            String uri = request.getRequestURI();
            out.println(ServletUtilities.headWithTitle("Test Servlet") +
            "<BODY BGCOLOR=\"#FDF5E6\">\n" +
            "<H2>URI: " + uri + "</H2>\n" +
            "</BODY></HTML>");
            }
            }

            程序清單5-2 web.xml(說(shuō)明servlet名稱的摘錄)
            <?xml version="1.0" encoding="ISO-8859-1"?>
            <!DOCTYPE web-app
            PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
            "http://java.sun.com/dtd/web-app_2_3.dtd">

            <web-app>
            <!-- … -->
            <servlet>
            <servlet-name>Test</servlet-name>
            <servlet-class>moreservlets.TestServlet</servlet-class>
            </servlet>
            <!-- … -->
            </web-app>

                  3.2 定義定制的URL

                   大多數(shù)服務(wù)器具有一個(gè)缺省的serlvet URL:
            http://host/webAppPrefix/servlet/packageName.ServletName。雖然在開發(fā)中使用這個(gè)URL很方便,但是我們常常會(huì)希望另一個(gè)URL用于部署。例如,可能會(huì)需要一個(gè)出現(xiàn)在Web應(yīng)用頂層的URL(如,http://host/webAppPrefix/Anyname),并且在此URL中沒(méi)有servlet項(xiàng)。位于頂層的URL簡(jiǎn)化了相對(duì)URL的使用。此外,對(duì)許多開發(fā)人員來(lái)說(shuō),頂層URL看上去比更長(zhǎng)更麻煩的缺省URL更簡(jiǎn)短。

            事實(shí)上,有時(shí)需要使用定制的URL。比如,你可能想關(guān)閉缺省URL映射,以便更好地強(qiáng)制實(shí)施安全限制或防止用戶意外地訪問(wèn)無(wú)初始化參數(shù)的servlet。如果你禁止了缺省的URL,那么你怎樣訪問(wèn)servlet呢?這時(shí)只有使用定制的URL了。
                   為了分配一個(gè)定制的URL,可使用servlet-mapping元素及其servlet-name和url-pattern子元素。Servlet-name元素提供了一個(gè)任意名稱,可利用此名稱引用相應(yīng)的servlet;url-pattern描述了相對(duì)于Web應(yīng)用的根目錄的URL。url-pattern元素的值必須以斜杠(/)起始。

                   下面給出一個(gè)簡(jiǎn)單的web.xml摘錄,它允許使用URL http://host/webAppPrefix/UrlTest而不是http://host/webAppPrefix/servlet/Test或
            http://host/webAppPrefix/servlet/moreservlets.TestServlet。請(qǐng)注意,仍然需要XML頭、DOCTYPE聲明以及web-app封閉元素。此外,可回憶一下,XML元素出現(xiàn)地次序不是隨意的。特別是,需要把所有servlet元素放在所有servlet-mapping元素之前。

            <servlet>
            <servlet-name>Test</servlet-name>
            <servlet-class>moreservlets.TestServlet</servlet-class>
            </servlet>
            <!-- ... -->
            <servlet-mapping>
            <servlet-name>Test</servlet-name>
            <url-pattern>/UrlTest</url-pattern>
            </servlet-mapping>

                    URL模式還可以包含通配符。例如,下面的小程序指示服務(wù)器發(fā)送所有以Web應(yīng)用的URL前綴開始,以..asp結(jié)束的請(qǐng)求到名為BashMS的servlet。

            <servlet>
            <servlet-name>BashMS</servlet-name>
            <servlet-class>msUtils.ASPTranslator</servlet-class>
            </servlet>
            <!-- ... -->
            <servlet-mapping>
            <servlet-name>BashMS</servlet-name>
            <url-pattern>/*.asp</url-pattern>
            </servlet-mapping>

                   3.3 命名JSP頁(yè)面

                   因?yàn)镴SP頁(yè)面要轉(zhuǎn)換成sevlet,自然希望就像命名servlet一樣命名JSP頁(yè)面。畢竟,JSP頁(yè)面可能會(huì)從初始化參數(shù)、安全設(shè)置或定制的URL中受益,正如普通的serlvet那樣。雖然JSP頁(yè)面的后臺(tái)實(shí)際上是servlet這句話是正確的,但存在一個(gè)關(guān)鍵的猜疑:即,你不知道JSP頁(yè)面的實(shí)際類名(因?yàn)橄到y(tǒng)自己挑選這個(gè)名字)。因此,為了命名JSP頁(yè)面,可將jsp-file元素替換為servlet-calss元素,如下所示:

            <servlet>
            <servlet-name>Test</servlet-name>
            <jsp-file>/TestPage.jsp</jsp-file>
            </servlet>

                  命名JSP頁(yè)面的原因與命名servlet的原因完全相同:即為了提供一個(gè)與定制設(shè)置(如,初始化參數(shù)和安全設(shè)置)一起使用的名稱,并且,以便能更改激活JSP頁(yè)面的URL(比方說(shuō),以便多個(gè)URL通過(guò)相同頁(yè)面得以處理,或者從URL中去掉.jsp擴(kuò)展名)。但是,在設(shè)置初始化參數(shù)時(shí),應(yīng)該注意,JSP頁(yè)面是利用jspInit方法,而不是init方法讀取初始化參數(shù)的。

                  例如,程序清單5-3給出一個(gè)名為TestPage.jsp的簡(jiǎn)單JSP頁(yè)面,它的工作只是打印出用來(lái)激活它的URL的本地部分。TestPage.jsp放置在deployDemo應(yīng)用的頂層。程序清單5-4給出了用來(lái)分配一個(gè)注冊(cè)名PageName,然后將此注冊(cè)名與http://host/webAppPrefix/UrlTest2/anything 形式的URL相關(guān)聯(lián)的web.xml文件(即,deployDemo/WEB-INF/web.xml)的一部分。

            程序清單5-3 TestPage.jsp
            <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
            <HTML>
            <HEAD>
            <TITLE>
            JSP Test Page
            </TITLE>
            </HEAD>
            <BODY BGCOLOR="#FDF5E6">
            <H2>URI: <%= request.getRequestURI() %></H2>
            </BODY>
            </HTML>

            程序清單5-4 web.xml(說(shuō)明JSP頁(yè)命名的摘錄)
            <?xml version="1.0" encoding="ISO-8859-1"?>
            <!DOCTYPE web-app
            PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
            "http://java.sun.com/dtd/web-app_2_3.dtd">

            <web-app>
            <!-- ... -->
            <servlet>
            <servlet-name>PageName</servlet-name>
            <jsp-file>/TestPage.jsp</jsp-file>
            </servlet>
            <!-- ... -->
            <servlet-mapping>
            <servlet-name> PageName </servlet-name>
            <url-pattern>/UrlTest2/*</url-pattern>
            </servlet-mapping>
            <!-- ... -->
            </web-app>

            4 禁止激活器servlet
                  對(duì)servlet或JSP頁(yè)面建立定制URL的一個(gè)原因是,這樣做可以注冊(cè)從init(servlet)或jspInit(JSP頁(yè)面)方法中讀取得初始化參數(shù)。但是,初始化參數(shù)只在是利用定制URL模式或注冊(cè)名訪問(wèn)servlet或JSP頁(yè)面時(shí)可以使用,用缺省URL http://host/webAppPrefix/servlet/ServletName 訪問(wèn)時(shí)不能使用。因此,你可能會(huì)希望關(guān)閉缺省URL,這樣就不會(huì)有人意外地調(diào)用初始化servlet了。這個(gè)過(guò)程有時(shí)稱為禁止激活器servlet,因?yàn)槎鄶?shù)服務(wù)器具有一個(gè)用缺省的servlet URL注冊(cè)的標(biāo)準(zhǔn)servlet,并激活缺省的URL應(yīng)用的實(shí)際servlet。

                   有兩種禁止此缺省URL的主要方法:

            在每個(gè)Web應(yīng)用中重新映射/servlet/模式。
            全局關(guān)閉激活器servlet。

                  重要的是應(yīng)該注意到,雖然重新映射每個(gè)Web應(yīng)用中的/servlet/模式比徹底禁止激活servlet所做的工作更多,但重新映射可以用一種完全可移植的方式來(lái)完成。相反,全局禁止激活器servlet完全是針對(duì)具體機(jī)器的,事實(shí)上有的服務(wù)器(如ServletExec)沒(méi)有這樣的選擇。下面的討論對(duì)每個(gè)Web應(yīng)用重新映射/servlet/ URL模式的策略。后面提供在Tomcat中全局禁止激活器servlet的詳細(xì)內(nèi)容。

                  4.1 重新映射/servlet/URL模式

                  在一個(gè)特定的Web應(yīng)用中禁止以http://host/webAppPrefix/servlet/ 開始的URL的處理非常簡(jiǎn)單。所需做的事情就是建立一個(gè)錯(cuò)誤消息servlet,并使用前一節(jié)討論的url-pattern元素將所有匹配請(qǐng)求轉(zhuǎn)向該servlet。只要簡(jiǎn)單地使用:

            <url-pattern>/servlet/*</url-pattern> 

                  作為servlet-mapping元素中的模式即可。

                  例如,程序清單5-5給出了將SorryServlet servlet(程序清單5-6)與所有以http://host/webAppPrefix/servlet/ 開頭的URL相關(guān)聯(lián)的部署描述符文件的一部分。

            程序清單5-5 web.xml(說(shuō)明JSP頁(yè)命名的摘錄)
            <?xml version="1.0" encoding="ISO-8859-1"?>
            <!DOCTYPE web-app
            PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
            "http://java.sun.com/dtd/web-app_2_3.dtd">

            <web-app>
            <!-- ... -->
            <servlet>
            <servlet-name>Sorry</servlet-name>
            <servlet-class>moreservlets.SorryServlet</servlet-class>
            </servlet>
            <!-- ... -->
            <servlet-mapping>
            <servlet-name> Sorry </servlet-name>
            <url-pattern>/servlet/*</url-pattern>
            </servlet-mapping>
            <!-- ... -->
            </web-app>

            程序清單5-6 SorryServlet.java
            package moreservlets;

            import java.io.*;
            import javax.servlet.*;
            import javax.servlet.http.*;

            /** Simple servlet used to give error messages to
            * users who try to access default servlet URLs
            * (i.e., http://host/webAppPrefix/servlet/ServletName)
            * in Web applications that have disabled this
            * behavior.
            * <P>
            * Taken from More Servlets and JavaServer Pages
            * from Prentice Hall and Sun Microsystems Press,
            * http://www.moreservlets.com/.
            * © 2002 Marty Hall; may be freely used or adapted.
            */

            public class SorryServlet extends HttpServlet {
            public void doGet(HttpServletRequest request,
            HttpServletResponse response)
            throws ServletException, IOException {
            response.setContentType("text/html");
            PrintWriter out = response.getWriter();
            String title = "Invoker Servlet Disabled.";
            out.println(ServletUtilities.headWithTitle(title) +
            "<BODY BGCOLOR=\"#FDF5E6\">\n" +
            "<H2>" + title + "</H2>\n" +
            "Sorry, access to servlets by means of\n" +
            "URLs that begin with\n" +
            "http://host/webAppPrefix/servlet/\n" +
            "has been disabled.\n" +
            "</BODY></HTML>");
            }

            public void doPost(HttpServletRequest request,
            HttpServletResponse response)
            throws ServletException, IOException {
            doGet(request, response);
            }
            }

            4.2 全局禁止激活器:Tomcat
                  Tomcat 4中用來(lái)關(guān)閉缺省URL的方法與Tomcat 3中所用的很不相同。下面介紹這兩種方法:

                  1.禁止激活器: Tomcat 4

                  Tomcat 4用與前面相同的方法關(guān)閉激活器servlet,即利用web.xml中的url-mapping元素進(jìn)行關(guān)閉。不同之處在于Tomcat使用了放在install_dir/conf中的一個(gè)服務(wù)器專用的全局web.xml文件,而前面使用的是存放在每個(gè)Web應(yīng)用的WEB-INF目錄中的標(biāo)準(zhǔn)web.xml文件。

                  因此,為了在Tomcat 4中關(guān)閉激活器servlet,只需在install_dir/conf/web.xml中簡(jiǎn)單地注釋出/servlet/* URL映射項(xiàng)即可,如下所示:

            <!--
            <servlet-mapping>
            <servlet-name>invoker</servlet-name>
            <url-pattern>/servlet/*</url-pattern>
            </servlet-mapping>
            -->

                  再次提醒,應(yīng)該注意這個(gè)項(xiàng)是位于存放在install_dir/conf的Tomcat專用的web.xml文件中的,此文件不是存放在每個(gè)Web應(yīng)用的WEB-INF目錄中的標(biāo)準(zhǔn)web.xml。

                  2.禁止激活器:Tomcat3

                  在Apache Tomcat的版本3中,通過(guò)在install_dir/conf/server.xml中注釋出InvokerInterceptor項(xiàng)全局禁止缺省servlet URL。例如,下面是禁止使用缺省servlet URL的server.xml文件的一部分。

            <!--
            <RequsetInterceptor
            className="org.apache.tomcat.request.InvokerInterceptor"
            debug="0" prefix="/servlet/" />
            -->

                  5 初始化和預(yù)裝載servlet與JSP頁(yè)面

                  這里討論控制servlet和JSP頁(yè)面的啟動(dòng)行為的方法。特別是,說(shuō)明了怎樣分配初始化參數(shù)以及怎樣更改服務(wù)器生存期中裝載servlet和JSP頁(yè)面的時(shí)刻。

                  5.1 分配servlet初始化參數(shù)

                  利用init-param元素向servlet提供初始化參數(shù),init-param元素具有param-name和param-value子元素。例如,在下面的例子中,如果initServlet servlet是利用它的注冊(cè)名(InitTest)訪問(wèn)的,它將能夠從其方法中調(diào)用getServletConfig().getInitParameter("param1")獲得"Value 1",調(diào)用getServletConfig().getInitParameter("param2")獲得"2"。

            <servlet>
            <servlet-name>InitTest</servlet-name>
            <servlet-class>moreservlets.InitServlet</servlet-class>
            <init-param>
            <param-name>param1</param-name>
            <param-value>value1</param-value>
            </init-param>
            <init-param>
            <param-name>param2</param-name>
            <param-value>2</param-value>
            </init-param>
            </servlet>

                   在涉及初始化參數(shù)時(shí),有幾點(diǎn)需要注意:

            返回值。GetInitParameter的返回值總是一個(gè)String。因此,在前一個(gè)例子中,可對(duì)param2使用Integer.parseInt獲得一個(gè)int。
            JSP中的初始化。JSP頁(yè)面使用jspInit而不是init。JSP頁(yè)面還需要使用jsp-file元素代替servlet-class。
            缺省URL。初始化參數(shù)只在通過(guò)它們的注冊(cè)名或與它們注冊(cè)名相關(guān)的定制URL模式訪問(wèn)Servlet時(shí)可以使用。因此,在這個(gè)例子中,param1和param2初始化參數(shù)將能夠在使用URL http://host/webAppPrefix/servlet/InitTest 時(shí)可用,但在使用URL http://host/webAppPrefix/servlet/myPackage.InitServlet 時(shí)不能使用。

             
            web.xml 詳解二
            例如,程序清單5-7給出一個(gè)名為InitServlet的簡(jiǎn)單servlet,它使用init方法設(shè)置firstName和emailAddress字段。程序清單5-8給出分配名稱InitTest給servlet的web.xml文件。
            程序清單5-7 InitServlet.java
            package moreservlets;

            import java.io.*;
            import javax.servlet.*;
            import javax.servlet.http.*;

            /** Simple servlet used to illustrate servlet
            * initialization parameters.
            * <P>
            * Taken from More Servlets and JavaServer Pages
            * from Prentice Hall and Sun Microsystems Press,
            * http://www.moreservlets.com/.
            * © 2002 Marty Hall; may be freely used or adapted.
            */

            public class InitServlet extends HttpServlet {
            private String firstName, emailAddress;

            public void init() {
            ServletConfig config = getServletConfig();
            firstName = config.getInitParameter("firstName");
            emailAddress = config.getInitParameter("emailAddress");
            }

            public void doGet(HttpServletRequest request,
            HttpServletResponse response)
            throws ServletException, IOException {
            response.setContentType("text/html");
            PrintWriter out = response.getWriter();
            String uri = request.getRequestURI();
            out.println(ServletUtilities.headWithTitle("Init Servlet") +
            "<BODY BGCOLOR=\"#FDF5E6\">\n" +
            "<H2>Init Parameters:</H2>\n" +
            "<UL>\n" +
            "<LI>First name: " + firstName + "\n" +
            "<LI>Email address: " + emailAddress + "\n" +
            "</UL>\n" +
            "</BODY></HTML>");
            }
            }

            程序清單5-8 web.xml(說(shuō)明初始化參數(shù)的摘錄)
            <?xml version="1.0" encoding="ISO-8859-1"?>
            <!DOCTYPE web-app
            PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
            "http://java.sun.com/dtd/web-app_2_3.dtd">

            <web-app>
            <!-- ... -->
            <servlet>
            <servlet-name>InitTest</servlet-name>
            <servlet-class>moreservlets.InitServlet</servlet-class>
            <init-param>
            <param-name>firstName</param-name>
            <param-value>Larry</param-value>
            </init-param>
            <init-param>
            <param-name>emailAddress</param-name>
            <param-value>Ellison@Microsoft.com</param-value>
            </init-param>
            </servlet>
            <!-- ... -->
            </web-app>

             

            5.2 分配JSP初始化參數(shù)
                  給JSP頁(yè)面提供初始化參數(shù)在三個(gè)方面不同于給servlet提供初始化參數(shù)。

                  1)使用jsp-file而不是servlet-class。因此,WEB-INF/web.xml文件的servlet元素如下所示:

            <servlet>
            <servlet-name>PageName</servlet-name>
            <jsp-file>/RealPage.jsp</jsp-file>
            <init-param>
            <param-name>...</param-name>
            <param-value>...</param-value>
            </init-param>
            ...
            </servlet>

                   2)幾乎總是分配一個(gè)明確的URL模式。對(duì)servlet,一般相應(yīng)地使用以http://host/webAppPrefix/servlet/ 開始的缺省URL。只需記住,使用注冊(cè)名而不是原名稱即可。這對(duì)于JSP頁(yè)面在技術(shù)上也是合法的。例如,在上面給出的例子中,可用URL http://host/webAppPrefix/servlet/PageName 訪問(wèn)RealPage.jsp的對(duì)初始化參數(shù)具有訪問(wèn)權(quán)的版本。但在用于JSP頁(yè)面時(shí),許多用戶似乎不喜歡應(yīng)用常規(guī)的servlet的URL。此外,如果JSP頁(yè)面位于服務(wù)器為其提供了目錄清單的目錄中(如,一個(gè)既沒(méi)有index.html也沒(méi)有index.jsp文件的目錄),則用戶可能會(huì)連接到此JSP頁(yè)面,單擊它,從而意外地激活未初始化的頁(yè)面。因此,好的辦法是使用url-pattern(5.3節(jié))將JSP頁(yè)面的原URL與注冊(cè)的servlet名相關(guān)聯(lián)。這樣,客戶機(jī)可使用JSP頁(yè)面的普通名稱,但仍然激活定制的版本。例如,給定來(lái)自項(xiàng)目1的servlet定義,可使用下面的servlet-mapping定義:

            <servlet-mapping>
            <servlet-name>PageName</servlet-name>
            <url-pattern>/RealPage.jsp</url-pattern>
            </servlet-mapping>

                  3)JSP頁(yè)使用jspInit而不是init。自動(dòng)從JSP頁(yè)面建立的servlet或許已經(jīng)使用了inti方法。因此,使用JSP聲明提供一個(gè)init方法是不合法的,必須制定jspInit方法。

                  為了說(shuō)明初始化JSP頁(yè)面的過(guò)程,程序清單5-9給出了一個(gè)名為InitPage.jsp的JSP頁(yè)面,它包含一個(gè)jspInit方法且放置于deployDemo Web應(yīng)用層次結(jié)構(gòu)的頂層。一般,http://host/deployDemo/InitPage.jsp 形式的URL將激活此頁(yè)面的不具有初始化參數(shù)訪問(wèn)權(quán)的版本,從而將對(duì)firstName和emailAddress變量顯示null。但是,web.xml文件(程序清單5-10)分配了一個(gè)注冊(cè)名,然后將該注冊(cè)名與URL模式/InitPage.jsp相關(guān)聯(lián)。

            程序清單5-9 InitPage.jsp
            <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
            <HTML>
            <HEAD><TITLE>JSP Init Test</TITLE></HEAD>
            <BODY BGCOLOR="#FDF5E6">
            <H2>Init Parameters:</H2>
            <UL>
            <LI>First name: <%= firstName %>
            <LI>Email address: <%= emailAddress %>
            </UL>
            </BODY></HTML>
            <%!
            private String firstName, emailAddress;

            public void jspInit() {
            ServletConfig config = getServletConfig();
            firstName = config.getInitParameter("firstName");
            emailAddress = config.getInitParameter("emailAddress");
            }
            %>

            程序清單5-10 web.xml(說(shuō)明JSP頁(yè)面的init參數(shù)的摘錄)
            <?xml version="1.0" encoding="ISO-8859-1"?>
            <!DOCTYPE web-app
            PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
            "http://java.sun.com/dtd/web-app_2_3.dtd">

            <web-app>
            <!-- ... -->
            <servlet>
            <servlet-name>InitPage</servlet-name>
            <jsp-file>/InitPage.jsp</jsp-file>
            <init-param>
            <param-name>firstName</param-name>
            <param-value>Bill</param-value>
            </init-param>
            <init-param>
            <param-name>emailAddress</param-name>
            <param-value>gates@oracle.com</param-value>
            </init-param>
            </servlet>
            <!-- ... -->
            <servlet-mapping>
            <servlet-name> InitPage</servlet-name>
            <url-pattern>/InitPage.jsp</url-pattern>
            </servlet-mapping>
            <!-- ... -->
            </web-app>

            5.3 提供應(yīng)用范圍內(nèi)的初始化參數(shù)
                  一般,對(duì)單個(gè)地servlet或JSP頁(yè)面分配初始化參數(shù)。指定的servlet或JSP頁(yè)面利用ServletConfig的getInitParameter方法讀取這些參數(shù)。但是,在某些情形下,希望提供可由任意servlet或JSP頁(yè)面借助ServletContext的getInitParameter方法讀取的系統(tǒng)范圍內(nèi)的初始化參數(shù)。

                  可利用context-param元素聲明這些系統(tǒng)范圍內(nèi)的初始化值。context-param元素應(yīng)該包含param-name、param-value以及可選的description子元素,如下所示:

            <context-param>
            <param-name>support-email</param-name>
            <param-value>blackhole@mycompany.com</param-value>
            </context-param>

                  可回憶一下,為了保證可移植性,web.xml內(nèi)的元素必須以正確的次序聲明。但這里應(yīng)該注意,context-param元素必須出現(xiàn)任意與文檔有關(guān)的元素(icon、display-name或description)之后及filter、filter-mapping、listener或servlet元素之前。

                  5.4 在服務(wù)器啟動(dòng)時(shí)裝載servlet

                  假如servlet或JSP頁(yè)面有一個(gè)要花很長(zhǎng)時(shí)間執(zhí)行的init(servlet)或jspInit(JSP)方法。例如,假如init或jspInit方法從某個(gè)數(shù)據(jù)庫(kù)或ResourceBundle查找產(chǎn)量。這種情況下,在第一個(gè)客戶機(jī)請(qǐng)求時(shí)裝載servlet的缺省行為將對(duì)第一個(gè)客戶機(jī)產(chǎn)生較長(zhǎng)時(shí)間的延遲。因此,可利用servlet的load-on-startup元素規(guī)定服務(wù)器在第一次啟動(dòng)時(shí)裝載servlet。下面是一個(gè)例子。

            <servlet>
            <servlet-name> … </servlet-name>
            <servlet-class> … </servlet-class> <!-- Or jsp-file -->
            <load-on-startup/>
            </servlet>

                  可以為此元素體提供一個(gè)整數(shù)而不是使用一個(gè)空的load-on-startup。想法是服務(wù)器應(yīng)該在裝載較大數(shù)目的servlet或JSP頁(yè)面之前裝載較少數(shù)目的servlet或JSP頁(yè)面。例如,下面的servlet項(xiàng)(放置在Web應(yīng)用的WEB-INF目錄下的web.xml文件中的web-app元素內(nèi))將指示服務(wù)器首先裝載和初始化SearchServlet,然后裝載和初始化由位于Web應(yīng)用的result目錄中的index.jsp文件產(chǎn)生的servlet。

            <servlet>
            <servlet-name>Search</servlet-name>
            <servlet-class>myPackage.SearchServlet</servlet-class> <!-- Or jsp-file -->
            <load-on-startup>1</load-on-startup>
            </servlet>
            <servlet>
            <servlet-name>Results</servlet-name>
            <servlet-class>/results/index.jsp</servlet-class> <!-- Or jsp-file -->
            <load-on-startup>2</load-on-startup>
            </servlet>

                   6 聲明過(guò)濾器

                   servlet版本2.3引入了過(guò)濾器的概念。雖然所有支持servlet API版本2.3的服務(wù)器都支持過(guò)濾器,但為了使用與過(guò)濾器有關(guān)的元素,必須在web.xml中使用版本2.3的DTD。
            過(guò)濾器可截取和修改進(jìn)入一個(gè)servlet或JSP頁(yè)面的請(qǐng)求或從一個(gè)servlet或JSP頁(yè)面發(fā)出的相應(yīng)。在執(zhí)行一個(gè)servlet或JSP頁(yè)面之前,必須執(zhí)行第一個(gè)相關(guān)的過(guò)濾器的doFilter方法。在該過(guò)濾器對(duì)其FilterChain對(duì)象調(diào)用doFilter時(shí),執(zhí)行鏈中的下一個(gè)過(guò)濾器。如果沒(méi)有其他過(guò)濾器,servlet或JSP頁(yè)面被執(zhí)行。過(guò)濾器具有對(duì)到來(lái)的ServletRequest對(duì)象的全部訪問(wèn)權(quán),因此,它們可以查看客戶機(jī)名、查找到來(lái)的cookie等。為了訪問(wèn)servlet或JSP頁(yè)面的輸出,過(guò)濾器可將響應(yīng)對(duì)象包裹在一個(gè)替身對(duì)象(stand-in object)中,比方說(shuō)把輸出累加到一個(gè)緩沖區(qū)。在調(diào)用FilterChain對(duì)象的doFilter方法之后,過(guò)濾器可檢查緩沖區(qū),如有必要,就對(duì)它進(jìn)行修改,然后傳送到客戶機(jī)。

                  例如,程序清單5-11帝國(guó)難以了一個(gè)簡(jiǎn)單的過(guò)濾器,只要訪問(wèn)相關(guān)的servlet或JSP頁(yè)面,它就截取請(qǐng)求并在標(biāo)準(zhǔn)輸出上打印一個(gè)報(bào)告(開發(fā)過(guò)程中在桌面系統(tǒng)上運(yùn)行時(shí),大多數(shù)服務(wù)器都可以使用這個(gè)過(guò)濾器)。

            程序清單5-11 ReportFilter.java
            package moreservlets;

            import java.io.*;
            import javax.servlet.*;
            import javax.servlet.http.*;
            import java.util.*;

            /** Simple filter that prints a report on the standard output
            * whenever the associated servlet or JSP page is accessed.
            * <P>
            * Taken from More Servlets and JavaServer Pages
            * from Prentice Hall and Sun Microsystems Press,
            * http://www.moreservlets.com/.
            * © 2002 Marty Hall; may be freely used or adapted.
            */

            public class ReportFilter implements Filter {
            public void doFilter(ServletRequest request,
            ServletResponse response,
            FilterChain chain)
            throws ServletException, IOException {
            HttpServletRequest req = (HttpServletRequest)request;
            System.out.println(req.getRemoteHost() +
            " tried to access " +
            req.getRequestURL() +
            " on " + new Date() + ".");
            chain.doFilter(request,response);
            }

            public void init(FilterConfig config)
            throws ServletException {
            }

            public void destroy() {}
            }

            web.xml 詳解三
            一旦建立了一個(gè)過(guò)濾器,可以在web.xml中利用filter元素以及filter-name(任意名稱)、file-class(完全限定的類名)和(可選的)init-params子元素聲明它。請(qǐng)注意,元素在web.xml的web-app元素中出現(xiàn)的次序不是任意的;允許服務(wù)器(但不是必需的)強(qiáng)制所需的次序,并且實(shí)際中有些服務(wù)器也是這樣做的。但這里要注意,所有filter元素必須出現(xiàn)在任意filter-mapping元素之前,filter-mapping元素又必須出現(xiàn)在所有servlet或servlet-mapping元素之前。
                  例如,給定上述的ReportFilter類,可在web.xml中作出下面的filter聲明。它把名稱Reporter與實(shí)際的類ReportFilter(位于moreservlets程序包中)相關(guān)聯(lián)。

            <filter>
            <filter-name>Reporter</filter-name>
            <filter-class>moresevlets.ReportFilter</filter-class>
            </filter>

                  一旦命名了一個(gè)過(guò)濾器,可利用filter-mapping元素把它與一個(gè)或多個(gè)servlet或JSP頁(yè)面相關(guān)聯(lián)。關(guān)于此項(xiàng)工作有兩種選擇。

                  首先,可使用filter-name和servlet-name子元素把此過(guò)濾器與一個(gè)特定的servlet名(此servlet名必須稍后在相同的web.xml文件中使用servlet元素聲明)關(guān)聯(lián)。例如,下面的程序片斷指示系統(tǒng)只要利用一個(gè)定制的URL訪問(wèn)名為SomeServletName的servlet或JSP頁(yè)面,就運(yùn)行名為Reporter的過(guò)濾器。

            <filter-mapping>
            <filter-name>Reporter</filter-name>
            <servlet-name>SomeServletName</servlet-name>
            </filter-mapping>

                  其次,可利用filter-name和url-pattern子元素將過(guò)濾器與一組servlet、JSP頁(yè)面或靜態(tài)內(nèi)容相關(guān)聯(lián)。例如,相面的程序片段指示系統(tǒng)只要訪問(wèn)Web應(yīng)用中的任意URL,就運(yùn)行名為Reporter的過(guò)濾器。

            <filter-mapping>
            <filter-name>Reporter</filter-name>
            <url-pattern>/*</url-pattern>
            </filter-mapping>

                  例如,程序清單5-12給出了將ReportFilter過(guò)濾器與名為PageName的servlet相關(guān)聯(lián)的web.xml文件的一部分。名字PageName依次又與一個(gè)名為TestPage.jsp的JSP頁(yè)面以及以模式http://host/webAppPrefix/UrlTest2/ 開頭的URL相關(guān)聯(lián)。TestPage.jsp的源代碼已經(jīng)JSP頁(yè)面命名的談?wù)撛谇懊娴?節(jié)"分配名稱和定制的URL"中給出。事實(shí)上,程序清單5-12中的servlet和servlet-name項(xiàng)從該節(jié)原封不動(dòng)地拿過(guò)來(lái)的。給定這些web.xml項(xiàng),可看到下面的標(biāo)準(zhǔn)輸出形式的調(diào)試報(bào)告(換行是為了容易閱讀)。

            audit.irs.gov tried to access
            http://mycompany.com/deployDemo/UrlTest2/business/tax-plan.html
            on Tue Dec 25 13:12:29 EDT 2001.

            程序清單5-12 Web.xml(說(shuō)明filter用法的摘錄)
            <?xml version="1.0" encoding="ISO-8859-1"?>
            <!DOCTYPE web-app
            PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
            "http://java.sun.com/dtd/web-app_2_3.dtd">

            <web-app>
            <filter>
            <filter-name>Reporter</filter-name>
            <filter-class>moresevlets.ReportFilter</filter-class>
            </filter>
            <!-- ... -->
            <filter-mapping>
            <filter-name>Reporter</filter-name>
            <servlet-name>PageName</servlet-name>
            </filter-mapping>
            <!-- ... -->
            <servlet>
            <servlet-name>PageName</servlet-name>
            <jsp-file>/RealPage.jsp</jsp-file>
            </servlet>
            <!-- ... -->
            <servlet-mapping>
            <servlet-name> PageName </servlet-name>
            <url-pattern>/UrlTest2/*</url-pattern>
            </servlet-mapping>
            <!-- ... -->
            </web-app>

             

             7 指定歡迎頁(yè)

                  假如用戶提供了一個(gè)像http://host/webAppPrefix/directoryName/ 這樣的包含一個(gè)目錄名但沒(méi)有包含文件名的URL,會(huì)發(fā)生什么事情呢?用戶能得到一個(gè)目錄表?一個(gè)錯(cuò)誤?還是標(biāo)準(zhǔn)文件的內(nèi)容?如果得到標(biāo)準(zhǔn)文件內(nèi)容,是index.html、index.jsp、default.html、default.htm或別的什么東西呢?

                   Welcome-file-list元素及其輔助的welcome-file元素解決了這個(gè)模糊的問(wèn)題。例如,下面的web.xml項(xiàng)指出,如果一個(gè)URL給出一個(gè)目錄名但未給出文件名,服務(wù)器應(yīng)該首先試用index.jsp,然后再試用index.html。如果兩者都沒(méi)有找到,則結(jié)果有賴于所用的服務(wù)器(如一個(gè)目錄列表)。

            <welcome-file-list>
            <welcome-file>index.jsp</welcome-file>
            <welcome-file>index.html</welcome-file>
            </welcome-file-list>

                  雖然許多服務(wù)器缺省遵循這種行為,但不一定必須這樣。因此,明確地使用welcom-file-list保證可移植性是一種良好的習(xí)慣。

                 8 指定處理錯(cuò)誤的頁(yè)面

                  現(xiàn)在我了解到,你在開發(fā)servlet和JSP頁(yè)面時(shí)從不會(huì)犯錯(cuò)誤,而且你的所有頁(yè)面是那樣的清晰,一般的程序員都不會(huì)被它們的搞糊涂。但是,是人總會(huì)犯錯(cuò)誤的,用戶可能會(huì)提供不合規(guī)定的參數(shù),使用不正確的URL或者不能提供必需的表單字段值。除此之外,其它開發(fā)人員可能不那么細(xì)心,他們應(yīng)該有些工具來(lái)克服自己的不足。

                  error-page元素就是用來(lái)克服這些問(wèn)題的。它有兩個(gè)可能的子元素,分別是:error-code和exception-type。第一個(gè)子元素error-code指出在給定的HTTP錯(cuò)誤代碼出現(xiàn)時(shí)使用的URL。第二個(gè)子元素excpetion-type指出在出現(xiàn)某個(gè)給定的Java異常但未捕捉到時(shí)使用的URL。error-code和exception-type都利用location元素指出相應(yīng)的URL。此URL必須以/開始。location所指出的位置處的頁(yè)面可通過(guò)查找HttpServletRequest對(duì)象的兩個(gè)專門的屬性來(lái)訪問(wèn)關(guān)于錯(cuò)誤的信息,這兩個(gè)屬性分別是

            javax.servlet.error.status_code和javax.servlet.error.message。

                  可回憶一下,在web.xml內(nèi)以正確的次序聲明web-app的子元素很重要。這里只要記住,error-page出現(xiàn)在web.xml文件的末尾附近,servlet、servlet-name和welcome-file-list之后即可。

                 8.1 error-code元素

                  為了更好地了解error-code元素的值,可考慮一下如果不正確地輸入文件名,大多數(shù)站點(diǎn)會(huì)作出什么反映。這樣做一般會(huì)出現(xiàn)一個(gè)404錯(cuò)誤信息,它表示不能找到該文件,但幾乎沒(méi)提供更多有用的信息。另一方面,可以試一下在www.microsoft.com、www.ibm.com 處或者特別是在www.bea.com 處輸出未知的文件名。這是會(huì)得出有用的消息,這些消息提供可選擇的位置,以便查找感興趣的頁(yè)面。提供這樣有用的錯(cuò)誤頁(yè)面對(duì)于Web應(yīng)用來(lái)說(shuō)是很有價(jià)值得。事實(shí)上rm-error-page子元素)。由form-login-page給出的HTML表單必須具有一個(gè)j_security_check的ACTION屬性、一個(gè)名為j_username的用戶名文本字段以及一個(gè)名為j_password的口令字段。

                  例如,程序清單5-19指示服務(wù)器使用基于表單的驗(yàn)證。Web應(yīng)用的頂層目錄中的一個(gè)名為login.jsp的頁(yè)面將收集用戶名和口令,并且失敗的登陸將由相同目錄中名為login-error.jsp的頁(yè)面報(bào)告。

            程序清單5-19 web.xml(說(shuō)明login-config的摘錄)
            <?xml version="1.0" encoding="ISO-8859-1"?>
            <!DOCTYPE web-app
            PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
            "http://java.sun.com/dtd/web-app_2_3.dtd">

            <web-app>
            <!-- ... -->
            <security-constraint> ... </security-constraint>
            <login-config>
            <auth-method> FORM </auth-method>
            <form-login-config>
            <form-login-page>/login.jsp</form-login-page>
            <form-error-page>/login-error.jsp</form-error-page>
            </form-login-config>
            </login-config>
            <!-- ... -->
            </web-app>

             

            9.2 限制對(duì)Web資源的訪問(wèn)

                  現(xiàn)在,可以指示服務(wù)器使用何種驗(yàn)證方法了。"了不起,"你說(shuō)道,"除非我能指定一個(gè)來(lái)收到保護(hù)的URL,否則沒(méi)有多大用處。"沒(méi)錯(cuò)。指出這些URL并說(shuō)明他們應(yīng)該得到何種保護(hù)正是security-constriaint元素的用途。此元素在web.xml中應(yīng)該出現(xiàn)在login-config的緊前面。它包含是個(gè)可能的子元素,分別是:web-resource-collection、auth-constraint、user-data-constraint和display-name。下面各小節(jié)對(duì)它們進(jìn)行介紹。

                  web-resource-collection

                  此元素確定應(yīng)該保護(hù)的資源。所有security-constraint元素都必須包含至少一個(gè)web-resource-collection項(xiàng)。此元素由一個(gè)給出任意標(biāo)識(shí)名稱的web-resource-name元素、一個(gè)確定應(yīng)該保護(hù)的URL的url-pattern元素、一個(gè)指出此保護(hù)所適用的HTTP命令(GET、POST等,缺省為所有方法)的http-method元素和一個(gè)提供資料的可選description元素組成。例如,下面的Web-resource-collection項(xiàng)(在security-constratint元素內(nèi))指出Web應(yīng)用的proprietary目錄中所有文檔應(yīng)該受到保護(hù)。

            <security-constraint>
            <web-resource-coolection>
            <web-resource-name>Proprietary</web-resource-name>
            <url-pattern>/propritary/*</url-pattern>
            </web-resource-coolection>
            <!-- ... -->
            </security-constraint>

                  重要的是應(yīng)該注意到,url-pattern僅適用于直接訪問(wèn)這些資源的客戶機(jī)。特別是,它不適合于通過(guò)MVC體系結(jié)構(gòu)利用RequestDispatcher來(lái)訪問(wèn)的頁(yè)面,或者不適合于利用類似jsp:forward的手段來(lái)訪問(wèn)的頁(yè)面。這種不勻稱如果利用得當(dāng)?shù)脑捄苡泻锰?。例如,servlet可利用MVC體系結(jié)構(gòu)查找數(shù)據(jù),把它放到bean中,發(fā)送請(qǐng)求到從bean中提取數(shù)據(jù)的JSP頁(yè)面并顯示它。我們希望保證決不直接訪問(wèn)受保護(hù)的JSP頁(yè)面,而只是通過(guò)建立該頁(yè)面將使用的bean的servlet來(lái)訪問(wèn)它。url-pattern和auth-contraint元素可通過(guò)聲明不允許任何用戶直接訪問(wèn)JSP頁(yè)面來(lái)提供這種保證。但是,這種不勻稱的行為可能讓開發(fā)人員放松警惕,使他們偶然對(duì)應(yīng)受保護(hù)的資源提供不受限制的訪問(wèn)。

                  auth-constraint

                  盡管web-resource-collention元素質(zhì)出了哪些URL應(yīng)該受到保護(hù),但是auth-constraint元素卻指出哪些用戶應(yīng)該具有受保護(hù)資源的訪問(wèn)權(quán)。此元素應(yīng)該包含一個(gè)或多個(gè)標(biāo)識(shí)具有訪問(wèn)權(quán)限的用戶類別role-name元素,以及包含(可選)一個(gè)描述角色的description元素。例如,下面web.xml中的security-constraint元素部門規(guī)定只有指定為Administrator或Big Kahuna(或兩者)的用戶具有指定資源的訪問(wèn)權(quán)。

            <security-constraint>
            <web-resource-coolection> ... </web-resource-coolection>
            <auth-constraint>
            <role-name>administrator</role-name>
            <role-name>kahuna</role-name>
            </auth-constraint>
            </security-constraint>

                  重要的是認(rèn)識(shí)到,到此為止,這個(gè)過(guò)程的可移植部分結(jié)束了。服務(wù)器怎樣確定哪些用戶處于任何角色以及它怎樣存放用戶的口令,完全有賴于具體的系統(tǒng)。

                  例如,Tomcat使用install_dir/conf/tomcat-users.xml將用戶名與角色名和口令相關(guān)聯(lián),正如下面例子中所示,它指出用戶joe(口令bigshot)和jane(口令enaj)屬于administrator和kahuna角色。

            <tomcat-users>
            <user name="joe" password="bigshot" roles="administrator,kahuna" />
            <user name="jane" password="enaj" roles="kahuna" />
            </tomcat-users>

                  user-data-constraint

                  這個(gè)可選的元素指出在訪問(wèn)相關(guān)資源時(shí)使用任何傳輸層保護(hù)。它必須包含一個(gè)transport-guarantee子元素(合法值為NONE、INTEGRAL或CONFIDENTIAL),并且可選地包含一個(gè)description元素。transport-guarantee為NONE值將對(duì)所用的通訊協(xié)議不加限制。INTEGRAL值表示數(shù)據(jù)必須以一種防止截取它的人閱讀它的方式傳送。雖然原理上(并且在未來(lái)的HTTP版本中),在INTEGRAL和CONFIDENTIAL之間可能會(huì)有差別,但在當(dāng)前實(shí)踐中,他們都只是簡(jiǎn)單地要求用SSL。例如,下面指示服務(wù)器只允許對(duì)相關(guān)資源做HTTPS連接:

            <security-constraint>
            <!-- ... -->
            <user-data-constraint>
            <transport-guarantee>CONFIDENTIAL</transport-guarantee>
            </user-data-constraint>
            </security-constraint>

            display-name

                 security-constraint的這個(gè)很少使用的子元素給予可能由GUI工具使用的安全約束項(xiàng)一個(gè)名稱。

                 9.3 分配角色名

                  迄今為止,討論已經(jīng)集中到完全由容器(服務(wù)器)處理的安全問(wèn)題之上了。但servlet以及JSP頁(yè)面也能夠處理它們自己的安全問(wèn)題。

                  例如,容器可能允許用戶從bigwig或bigcheese角色訪問(wèn)一個(gè)顯示主管人員額外緊貼的頁(yè)面,但只允許bigwig用戶修改此頁(yè)面的參數(shù)。完成這種更細(xì)致的控制的一種常見(jiàn)方法是調(diào)用HttpServletRequset的isUserInRole方法,并據(jù)此修改訪問(wèn)。

                  Servlet的security-role-ref子元素提供出現(xiàn)在服務(wù)器專用口令文件中的安全角色名的一個(gè)別名。例如,假如編寫了一個(gè)調(diào)用request.isUserInRole("boss")的servlet,但后來(lái)該servlet被用在了一個(gè)其口令文件調(diào)用角色manager而不是boss的服務(wù)器中。下面的程序段使該servlet能夠使用這兩個(gè)名稱中的任何一個(gè)。

            <servlet>
            <!-- ... -->
            <security-role-ref>
            <role-name>boss</role-name> <!-- New alias -->
            <role-link>manager</role-link> <!-- Real name -->
            </security-role-ref>
            </servlet>

                  也可以在web-app內(nèi)利用security-role元素提供將出現(xiàn)在role-name元素中的所有安全角色的一個(gè)全局列表。分別地生命角色使高級(jí)IDE容易處理安全信息。

                  10 控制會(huì)話超時(shí)

                  如果某個(gè)會(huì)話在一定的時(shí)間內(nèi)未被訪問(wèn),服務(wù)器可把它扔掉以節(jié)約內(nèi)存??衫肏ttpSession的setMaxInactiveInterval方法直接設(shè)置個(gè)別會(huì)話對(duì)象的超時(shí)值。如果不采用這種方法,則缺省的超時(shí)值由具體的服務(wù)器決定。但可利用session-config和session-timeout元素來(lái)給出一個(gè)適用于所有服務(wù)器的明確的超時(shí)值。超時(shí)值的單位為分鐘,因此,下面的例子設(shè)置缺省會(huì)話超時(shí)值為三個(gè)小時(shí)(180分鐘)。

            <session-config>
            <session-timeout>180</session-timeout>
            </session-config>

                  11 Web應(yīng)用的文檔化

                  越來(lái)越多的開發(fā)環(huán)境開始提供servlet和JSP的直接支持。例子有Borland Jbuilder Enterprise Edition、Macromedia UltraDev、Allaire JRun Studio(寫此文時(shí),已被Macromedia收購(gòu))以及IBM VisuaAge for Java等。

                  大量的web.xml元素不僅是為服務(wù)器設(shè)計(jì)的,而且還是為可視開發(fā)環(huán)境設(shè)計(jì)的。它們包括icon、display-name和discription等。
            可回憶一下,在web.xml內(nèi)以適當(dāng)?shù)卮涡蚵暶鱳eb-app子元素很重要。不過(guò),這里只要記住icon、display-name和description是web.xml的web-app元素內(nèi)的前三個(gè)合法元素即可。

                   icon

                  icon元素指出GUI工具可用來(lái)代表Web應(yīng)用的一個(gè)和兩個(gè)圖像文件??衫胹mall-icon元素指定一幅16 x 16的GIF或JPEG圖像,用large-icon元素指定一幅32 x 32的圖像。下面舉一個(gè)例子:

            <icon>
            <small-icon>/images/small-book.gif</small-icon>
            <large-icon>/images/tome.jpg</large-icon>
            </icon>

                  display-name

                  display-name元素提供GUI工具可能會(huì)用來(lái)標(biāo)記此Web應(yīng)用的一個(gè)名稱。下面是個(gè)例子。

            <display-name>Rare Books</display-name>

                  description

                  description元素提供解釋性文本,如下所示:

            <description>
            This Web application represents the store developed for
            rare-books.com, an online bookstore specializing in rare
            and limited-edition books.
            </description>

            12 關(guān)聯(lián)文件與MIME類型

                  服務(wù)器一般都具有一種讓W(xué)eb站點(diǎn)管理員將文件擴(kuò)展名與媒體相關(guān)聯(lián)的方法。例如,將會(huì)自動(dòng)給予名為mom.jpg的文件一個(gè)image/jpeg的MIME類型。但是,假如你的Web應(yīng)用具有幾個(gè)不尋常的文件,你希望保證它們?cè)诎l(fā)送到客戶機(jī)時(shí)分配為某種MIME類型。mime-mapping元素(具有extension和mime-type子元素)可提供這種保證。例如,下面的代碼指示服務(wù)器將application/x-fubar的MIME類型分配給所有以.foo結(jié)尾的文件。

            <mime-mapping>
            <extension>foo</extension>
            <mime-type>application/x-fubar</mime-type>
            </mime-mapping>

                   或許,你的Web應(yīng)用希望重載(override)標(biāo)準(zhǔn)的映射。例如,下面的代碼將告訴服務(wù)器在發(fā)送到客戶機(jī)時(shí)指定.ps文件作為純文本(text/plain)而不是作為PostScript(application/postscript)。

            <mime-mapping>
            <extension>ps</extension>
            <mime-type>application/postscript</mime-type>
            </mime-mapping>


                  13 定位TLD

                  JSP taglib元素具有一個(gè)必要的uri屬性,它給出一個(gè)TLD(Tag Library Descriptor)文件相對(duì)于Web應(yīng)用的根的位置。TLD文件的實(shí)際名稱在發(fā)布新的標(biāo)簽庫(kù)版本時(shí)可能會(huì)改變,但我們希望避免更改所有現(xiàn)有JSP頁(yè)面。此外,可能還希望使用保持taglib元素的簡(jiǎn)練性的一個(gè)簡(jiǎn)短的uri。這就是部署描述符文件的taglib元素派用場(chǎng)的所在了。Taglib包含兩個(gè)子元素:taglib-uri和taglib-location。taglib-uri元素應(yīng)該與用于JSP taglib元素的uri屬性的東西相匹配。Taglib-location元素給出TLD文件的實(shí)際位置。例如,假如你將文件chart-tags-1.3beta.tld放在WebApp/WEB-INF/tlds中?,F(xiàn)在,假如web.xml在web-app元素內(nèi)包含下列內(nèi)容。

            <taglib>
            <taglib-uri>/charts.tld</taglib-uri>
            <taglib-location>
            /WEB-INF/tlds/chart-tags-1.3beta.tld
            </taglib-location>
            </taglib>

                  給出這個(gè)說(shuō)明后,JSP頁(yè)面可通過(guò)下面的簡(jiǎn)化形式使用標(biāo)簽庫(kù)。

            <%@ taglib uri="/charts.tld" prefix="somePrefix" %>

                  14 指定應(yīng)用事件監(jiān)聽程序

                  應(yīng)用事件監(jiān)聽器程序是建立或修改servlet環(huán)境或會(huì)話對(duì)象時(shí)通知的類。它們是servlet規(guī)范的版本2.3中的新內(nèi)容。這里只簡(jiǎn)單地說(shuō)明用來(lái)向Web應(yīng)用注冊(cè)一個(gè)監(jiān)聽程序的web.xml的用法。

                  注冊(cè)一個(gè)監(jiān)聽程序涉及在web.xml的web-app元素內(nèi)放置一個(gè)listener元素。在listener元素內(nèi),listener-class元素列出監(jiān)聽程序的完整的限定類名,如下所示:

            <listener>
            <listener-class>package.ListenerClass</listener-class>
            </listener>

                  雖然listener元素的結(jié)構(gòu)很簡(jiǎn)單,但請(qǐng)不要忘記,必須正確地給出web-app元素內(nèi)的子元素的次序。listener元素位于所有的servlet元素之前以及所有filter-mapping元素之后。此外,因?yàn)閼?yīng)用生存期監(jiān)聽程序是serlvet規(guī)范的2.3版本中的新內(nèi)容,所以必須使用web.xml DTD的2.3版本,而不是2.2版本。

                  例如,程序清單5-20給出一個(gè)名為ContextReporter的簡(jiǎn)單的監(jiān)聽程序,只要Web應(yīng)用的Servlet-Context建立(如裝載Web應(yīng)用)或消除(如服務(wù)器關(guān)閉)時(shí),它就在標(biāo)準(zhǔn)輸出上顯示一條消息。程序清單5-21給出此監(jiān)聽程序注冊(cè)所需要的web.xml文件的一部分。

            程序清單5-20 ContextReporterjava
            package moreservlets;

            import javax.servlet.*;
            import java.util.*;

            /** Simple listener that prints a report on the standard output
            * when the ServletContext is created or destroyed.
            * <P>
            * Taken from More Servlets and JavaServer Pages
            * from Prentice Hall and Sun Microsystems Press,
            * http://www.moreservlets.com/.
            * © 2002 Marty Hall; may be freely used or adapted.
            */

            public class ContextReporter implements ServletContextListener {
            public void contextInitialized(ServletContextEvent event) {
            System.out.println("Context created on " +
            new Date() + ".");
            }

            public void contextDestroyed(ServletContextEvent event) {
            System.out.println("Context destroyed on " +
            new Date() + ".");
            }
            }

             

            程序清單5-21 web.xml(聲明一個(gè)監(jiān)聽程序的摘錄)
            <?xml version="1.0" encoding="ISO-8859-1"?>
            <!DOCTYPE web-app
            PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
            "http://java.sun.com/dtd/web-app_2_3.dtd"> 

            <web-app>
            <!-- ... -->
            <filter-mapping> … </filter-mapping>
            <listener>
            <listener-class>package.ListenerClass</listener-class>
            </listener>
            <servlet> ... </servlet>
            <!-- ... -->
            </web-app>

                  15 J2EE元素

                  本節(jié)描述用作J2EE環(huán)境組成部分的Web應(yīng)用的web.xml元素。這里將提供一個(gè)簡(jiǎn)明的介紹,詳細(xì)內(nèi)容可以參閱http://java.sun.com/j2ee/j2ee-1_3-fr-spec.pdf的Java 2 Plantform Enterprise Edition版本1.3規(guī)范的第5章。

                  distributable

                  distributable元素指出,Web應(yīng)用是以這樣的方式編程的:即,支持集群的服務(wù)器可安全地在多個(gè)服務(wù)器上分布Web應(yīng)用。例如,一個(gè)可分布的應(yīng)用必須只使用Serializable對(duì)象作為其HttpSession對(duì)象的屬性,而且必須避免用實(shí)例變量(字段)來(lái)實(shí)現(xiàn)持續(xù)性。distributable元素直接出現(xiàn)在discription元素之后,并且不包含子元素或數(shù)據(jù),它只是一個(gè)如下的標(biāo)志。
            <distributable />

                  resource-env-ref

                 resource-env-ref元素聲明一個(gè)與某個(gè)資源有關(guān)的管理對(duì)象。此元素由一個(gè)可選的description元素、一個(gè)resource-env-ref-name元素(一個(gè)相對(duì)于java:comp/env環(huán)境的JNDI名)以及一個(gè)resource-env-type元素(指定資源類型的完全限定的類),如下所示:

            <resource-env-ref>
            <resource-env-ref-name>
            jms/StockQueue
            </resource-env-ref-name>
            <resource-env-ref-type>
            javax.jms.Queue
            </resource-env-ref-type>
            </resource-env-ref>

                  env-entry

                  env-entry元素聲明Web應(yīng)用的環(huán)境項(xiàng)。它由一個(gè)可選的description元素、一個(gè)env-entry-name元素(一個(gè)相對(duì)于java:comp/env環(huán)境JNDI名)、一個(gè)env-entry-value元素(項(xiàng)值)以及一個(gè)env-entry-type元素(java.lang程序包中一個(gè)類型的完全限定類名,java.lang.Boolean、java.lang.String等)組成。下面是一個(gè)例子:
            <env-entry>
            <env-entry-name>minAmout</env-entry-name>
            <env-entry-value>100.00</env-entry-value>
            <env-entry-type>minAmout</env-entry-type>
            </env-entry>

                  ejb-ref

                  ejb-ref元素聲明對(duì)一個(gè)EJB的主目錄的應(yīng)用。它由一個(gè)可選的description元素、一個(gè)ejb-ref-name元素(相對(duì)于java:comp/env的EJB應(yīng)用)、一個(gè)ejb-ref-type元素(bean的類型,Entity或Session)、一個(gè)home元素(bean的主目錄接口的完全限定名)、一個(gè)remote元素(bean的遠(yuǎn)程接口的完全限定名)以及一個(gè)可選的ejb-link元素(當(dāng)前bean鏈接的另一個(gè)bean的名稱)組成。

                  ejb-local-ref
            ejb-local-ref元素聲明一個(gè)EJB的本地主目錄的引用。除了用local-home代替home外,此元素具有與ejb-ref元素相同的屬性并以相同的方式使用。


            本文來(lái)自CSDN博客,轉(zhuǎn)載請(qǐng)標(biāo)明出處:http://blog.csdn.net/tanghc1983/archive/2007/05/08/1601247.aspx

            posted @ 2009-08-16 10:25 jimphei 閱讀(79) | 評(píng)論 (0)編輯 收藏

            1、建立數(shù)據(jù)庫(kù)連接

                  首先要建立一個(gè)到想要使用的DBMS的連接。這包括兩個(gè)步驟:(1) 加載驅(qū)動(dòng)程序;(2) 創(chuàng)建連接。


               (1) 加載驅(qū)動(dòng)程序 
                     Class.forName("sun.jdbc.odbc.JdbcOdbcDriver"); //字符串為他驅(qū)動(dòng)程序的類名

              (2) 創(chuàng)建連接

                  創(chuàng)建連接的第二步是使用合適的驅(qū)動(dòng)程序連接到DBMS。如下代碼行示范了一般的用法:

                        //url字符串為使用的子協(xié)議,也就是在JDBC URL中放在jdbc:之后的東西 
                      Connection con = DriverManager.getConnection(url,"myLogin", "myPassword"); 

             2、創(chuàng)建和執(zhí)行JDBC語(yǔ)句

               Statement對(duì)象可將SQL語(yǔ)句發(fā)送到DBMS。這只要?jiǎng)?chuàng)建一個(gè)Statement對(duì)象,將想要執(zhí)行的SQL語(yǔ)句傳遞給適當(dāng)?shù)膱?zhí)行方法,

            然后執(zhí)行該Statement對(duì)象。

                  Statement stmt = con.createStatement();

             

               對(duì)于SELECT語(yǔ)句,使用的方法是executeQuery。對(duì)于創(chuàng)建或修改表的語(yǔ)句,使用的方法是executeUpdate。

               //strSQL字符串是一條DDL(數(shù)據(jù)描述語(yǔ)言)語(yǔ)句

               stmt.executeUpdate(strSQL);

            3、處理結(jié)果

               JDBC將結(jié)果集返回給ResultSet對(duì)象

               //strSQL字符串是一條查詢語(yǔ)句   

               ResultSet rs = stmt.executeQuery(strSQL);
               1)next方法
               變量rs(ResultSet的一個(gè)實(shí)例)包含了結(jié)果集中顯示的表。為訪問(wèn)名稱和單價(jià)要移動(dòng)到每一行,根據(jù)它們的類型檢索數(shù)值。
            next方法將光標(biāo)移到下一行,使那行(稱為當(dāng)前行)成為可在其上操作的行。由于光標(biāo)初始定位于ResultSet對(duì)象第一行的上面,
            所以第一次調(diào)用next方法將光標(biāo)移到第一行,使它成為當(dāng)前行。接下來(lái)調(diào)用next方法將使光標(biāo)從上至下每次移動(dòng)一行。
               2)getXXX方法
               適當(dāng)類型的getXXX方法可用于檢索列中的數(shù)值。檢索VARCHAR SQL類型數(shù)值的方法是getString。檢索該類型數(shù)值的方法是getFloat。
            每次調(diào)用next方法,下一行就成為當(dāng)前行,循環(huán)一直持續(xù)到rs中再也沒(méi)有可向前移動(dòng)的行為止。
            while (rs.next()) 
            {
                String s = rs.getString(strColumnName1 );//strColumnName1為strSQL字符串中的列名
                float n = rs.getFloat(strColumnName2)//strColumnName2為strSQL字符串中的列名
                System.out.println(s + " " + n);
            }
            JDBC使用兩種方法標(biāo)識(shí)getXXX方法檢索數(shù)值的列。一種是指定列名,這也是前面例子所做的。
            另一種是指定列索引(列的序號(hào)),1表示第一列,2表示第二列,以此類推。
               String s = rs.getString(1); float n = rs.getFloat(2);
               3)getString方法
               盡管推薦使用getString方法檢索CHAR和VARCHAR SQL類型的數(shù)據(jù),但也可能用它檢索其他基本SQL類型的數(shù)據(jù)
            (但不可用它檢索新的SQL3數(shù)據(jù)類型。本教程后面將討論SQL3類型)。用getString檢索所有數(shù)值有很多優(yōu)點(diǎn)但也有局限。
            如果用它檢索numeric類型的數(shù)據(jù),getString將把numeric值轉(zhuǎn)換成Java的String對(duì)象,這樣一來(lái),在數(shù)據(jù)要作為數(shù)字前就必須轉(zhuǎn)換回numeric類型。
            在數(shù)值一直當(dāng)成字符串的場(chǎng)合,這是無(wú)可非議的。如果想讓程序檢索除了SQL3類型之外的任何標(biāo)準(zhǔn)SQL類型,就可用getString方法。
             
            4、關(guān)閉連接
               connection.close();
             
            ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
            下面是個(gè)完整的例子:
            1)下載并安裝Microsoft JDBC (http://download.microsoft.com/download/SQLSVR2000/Install/2.2.0022/NT5XP/EN-US/setup.exe)
            2) 安裝完成Microsoft JDBC后,將安裝目錄中的lib目錄下三個(gè)jar文件msbase.jar, mssqlserver.jar, msutil.jar引入項(xiàng)目中 
            3)建立數(shù)據(jù)庫(kù)tempdb,建立表COFFEES,SQL語(yǔ)句為:
               CREATE TABLE COFFEES (COF_NAME VARCHAR(32), SUP_ID INTEGER, PRICE FLOAT, SALES INTEGER, TOTAL INTEGER) 
            4)給表COFFEES添加一些數(shù)據(jù),如:
               INSERT INTO COFFEES VALUES ('Colombian', 101, 7.99, 0, 0); 
            5)輸入以下源文件,并執(zhí)行
            import java.sql.*;
            /**
             * @author liujun TODO To change the template for this generated type comment go
             *         to Window - Preferences - Java - Code Style - Code Templates
             */
            public class JDBC_01
            {
                public static void main(String[] args)
                {
                    showTable();
                }
                public static void showTable()
                {
                    //String strDriver="sun.jdbc.odbc.JdbcOdbcDriver";
                    String strDriver = "com.microsoft.jdbc.sqlserver.SQLServerDriver";
                    //String strConnUrl="jdbc:odbc:TempDataSources";
                    String strConnUrl = "jdbc:microsoft:sqlserver://localhost:1433;DatabaseName=tempdb";
                    String strSQL = "select * from COFFEES";
                    try
                    {
                        //加載驅(qū)動(dòng)程序
                        Class.forName(strDriver);
                        //創(chuàng)建連接
                        Connection objConn = DriverManager.getConnection(strConnUrl, "sa",
                                "");
                        //創(chuàng)建Statement對(duì)象
                        Statement objStatement = objConn.createStatement();
                        //JDBC將結(jié)果集返回給ResultSet對(duì)象
                        ResultSet objSet = objStatement.executeQuery(strSQL);
                        //獲得查詢結(jié)果的列數(shù)
                        int lCloumnCount = objSet.getMetaData().getColumnCount();
                        System.out.println("查詢結(jié)果如下所示");
                        //顯示列名
                        for (int i = 1; i <= lCloumnCount; i++)
                        {
                            System.out.print(objSet.getMetaData().getColumnName(i)
                                    + "       ");
                        }
                        System.out.println();
                        //顯示結(jié)果
                        while (objSet.next())
                        {
                            //顯示COF_NAME
                            System.out.print("" + objSet.getString(1));
                            //顯示SUP_ID PRICE SALES TOTAL
                            System.out.print("        " + objSet.getInt(2));
                            //顯示 PRICE
                            System.out.print("         " + objSet.getFloat(3));
                            //顯示SALES
                            System.out.print("         " + objSet.getInt(4));
                            //顯示 TOTAL
                            System.out.print("        " + objSet.getInt(5));
                        }
                   objConn.close();
                    }
                    catch (Exception e)
                    {
                        System.out.print(e.getMessage());
                    }
                }
            }

            5、使用預(yù)備語(yǔ)句

               PreparedStatement對(duì)象是包含一條預(yù)編譯過(guò)的SQL語(yǔ)句。DBMS不必編譯就可直接運(yùn)行PreparedStatement的SQL語(yǔ)句。所以代替Statement對(duì)象一般會(huì)縮短執(zhí)行時(shí)間。盡管paredStatement對(duì)象可用于不帶參數(shù)的SQL語(yǔ)句,但在多數(shù)場(chǎng)合是用于帶參數(shù)的SQL語(yǔ)句。其用法如下:

               String strUpdateSales="UPDATE COFFEES SET SALES = ? WHERE COF_NAME LIKE ?";

               PreparedStatement updateSales = con.prepareStatement(strUpdateSales);
               updateSales.setInt(1, 75);//給上面的SQL語(yǔ)句的第一個(gè)問(wèn)號(hào)付值
               updateSales.setString(2, "Colombian");//給上面的SQL語(yǔ)句的第二個(gè)問(wèn)號(hào)付值
               updateSales.executeUpdate();//執(zhí)行UPDATE COFFEES SET SALES = 75 WHERE COF_NAME LIKE ‘Colombian’
            因此,可用該對(duì)象制作成模板實(shí)現(xiàn)一次構(gòu)造多條帶參數(shù)的SQL語(yǔ)句,簡(jiǎn)化編碼。
             
            6、executeUpdate方法的返回值
               executeQuery返回一個(gè)ResultSet對(duì)象,但executeUpdate返回的是一個(gè)整數(shù)值,指出了表中已更新的行數(shù)
            如果executeUpdate的返回值為0表明如下兩種情況:(1) 執(zhí)行的語(yǔ)句是一不影響任何行的更新語(yǔ)句;(2) 執(zhí)行的是一DLL語(yǔ)句。
             
             7、使用事務(wù)
               一個(gè)事務(wù)是作為一個(gè)單元執(zhí)行的一組語(yǔ)句(一條或多條語(yǔ)句),因此它們要么全部執(zhí)行,要么全部不執(zhí)行。要讓兩條或更多條語(yǔ)句組成
            一個(gè)事務(wù)就要禁用自動(dòng)提交模式,具體用方法con.setAutoCommit(false);一旦禁用了自動(dòng)提交模式,就沒(méi)有SQL語(yǔ)句會(huì)提交了,
            除非您顯式調(diào)用con.commit()方法,實(shí)現(xiàn)提交事務(wù)。
               使用事務(wù)可以保持?jǐn)?shù)據(jù)完整性。調(diào)用rollback方法可取消一個(gè)事務(wù),將修改的任何數(shù)據(jù)返還到以前的值。如果在執(zhí)行一個(gè)事務(wù)
            中的一條或多條語(yǔ)句時(shí)得到了SQLException。就應(yīng)該調(diào)用rollback方法取消事務(wù),從頭開始整個(gè)事務(wù)。一般要在catch塊中調(diào)用
            rollback,以避免可能使用不正確的數(shù)據(jù)。
             
            8、檢索警告
               SQLWarning對(duì)象是SQLException的子類,用于處理數(shù)據(jù)庫(kù)訪問(wèn)警告。就像異常一樣,警告并不終止程序的執(zhí)行;它們只是提醒
            用戶——發(fā)生了一些未預(yù)料的事情。Connection對(duì)象、Statement對(duì)象(包括PreparedStatement和CallableStatement對(duì)象)
            或ResultSet對(duì)象都會(huì)報(bào)告警告。這些類都有g(shù)etWarnings方法,為查看調(diào)用對(duì)象報(bào)告的第一個(gè)警告,就必須調(diào)用該方法。如果
            getWarnings返回一個(gè)警告,就可在其上調(diào)用SQLWarning方法getNextWarning取得其他警告。自動(dòng)執(zhí)行一條語(yǔ)句會(huì)清除前一條語(yǔ)句
            的警告,因此警告不會(huì)迭加。但這也表明提取一條語(yǔ)句報(bào)告的警告必須在執(zhí)行下一語(yǔ)句之前進(jìn)行。
             
            9、存儲(chǔ)過(guò)程
               存儲(chǔ)過(guò)程是一組SQL語(yǔ)句組成的邏輯單元,用于執(zhí)行特定的任務(wù)。存儲(chǔ)過(guò)程用于封裝數(shù)據(jù)庫(kù)服務(wù)器上執(zhí)行的一組操作或查詢。
            存儲(chǔ)過(guò)程可以編譯,然后使用不同的參數(shù)和結(jié)果執(zhí)行,這些參數(shù)和結(jié)果可以是輸入、輸出和輸入/輸出參數(shù)的任意組合。
               1)用SQL語(yǔ)句創(chuàng)建存儲(chǔ)過(guò)程
               如下所示,其方法同其他的DDL語(yǔ)句相同:
               String createProcedure = "create procedure SHOW_SUPPLIERS " +
                   "as " +"select SUPPLIERS.SUP_NAME, COFFEES.COF_NAME " + 
                   "from SUPPLIERS, COFFEES " +"where SUPPLIERS.SUP_ID = COFFEES.SUP_ID " +
                   "order by SUP_NAME"; 
               Statement stmt = con.createStatement();
               stmt.executeUpdate(createProcedure);
               2)從JDBC調(diào)用存儲(chǔ)過(guò)程
               第一步是創(chuàng)建一個(gè)CallableStatement對(duì)象。CallableStatement對(duì)象包含了存儲(chǔ)過(guò)程的一個(gè)調(diào)用;但它不包含存儲(chǔ)過(guò)程
            本身。接著使用連接con創(chuàng)建了存儲(chǔ)過(guò)程的一個(gè)調(diào)用?;ɡㄌ?hào)內(nèi)的那部分就是存儲(chǔ)過(guò)程的轉(zhuǎn)義語(yǔ)法。當(dāng)驅(qū)動(dòng)程序碰到
            “{call SHOW_SUPPLIERS}”時(shí),它將把這個(gè)轉(zhuǎn)義語(yǔ)法轉(zhuǎn)換成數(shù)據(jù)庫(kù)使用的本地SQL,以調(diào)用call 后面的的存儲(chǔ)過(guò)程。
            如下所示:
               CallableStatement cs = con.prepareCall("{call SHOW_SUPPLIERS}"); 
               ResultSet rs = cs.executeQuery();
               注意,用于執(zhí)行cs的方法是executeQuery,因?yàn)閏s調(diào)用的存儲(chǔ)過(guò)程包含一個(gè)查詢,執(zhí)行后產(chǎn)生一個(gè)結(jié)果集。如果存儲(chǔ)過(guò)程
            包含一條更新或一條DLL語(yǔ)句,那就要使用executeUpdate方法。但有時(shí)一個(gè)存儲(chǔ)過(guò)程包含多條SQL語(yǔ)句,因而它產(chǎn)生的不只是
            一個(gè)結(jié)果集、不只是一個(gè)更新計(jì)數(shù)或產(chǎn)生一些結(jié)果集和更新計(jì)數(shù)的組合。這樣就有多個(gè)結(jié)果集,這時(shí)就應(yīng)該使用execute方法
            來(lái)執(zhí)行CallableStatement。 CallableStatement類是PreparedStatement的子類,因此CallableStatement對(duì)象可與
            PreparedStatement對(duì)象一樣帶有輸入?yún)?shù)。此外,CallableStatement對(duì)象還可帶輸出參數(shù)或輸入/輸出參數(shù)
            posted @ 2009-08-14 10:34 jimphei 閱讀(119) | 評(píng)論 (0)編輯 收藏

            主站蜘蛛池模板: 桓台县| 辉县市| 大城县| 阿巴嘎旗| 黔西县| 怀柔区| 宿州市| 梁平县| 娄烦县| 石狮市| 运城市| 万荣县| 合山市| 郁南县| 黄山市| 天气| 南郑县| 固镇县| 武陟县| 疏勒县| 卢龙县| 龙游县| 伊宁市| 锦州市| 胶州市| 巨野县| 丁青县| 涟水县| 荔浦县| 英德市| 麟游县| 潞城市| 龙泉市| 乾安县| 成安县| 五指山市| 黄浦区| 开封县| 泰和县| 武乡县| 红河县|