轉自:
http://blog.csdn.net/imnol/archive/2007/08/15/1745288.aspx
一 Servlet
1、Servlet、ServletConfig:前者對應一個Servlet類,后者對應的是在web.xml中的配置信息
(1)Servlet:
init(ServletConfig):只再類加載并實例化后會被調用一次。
service(ServletRequest,ServletResponse):會被調用多次
destroy():只有一次
getServletConfig()
getServletInfo()
(2)ServletCionfig
getServletName()
getInitParameter(String):---<servlet><init-param></init-param></servlet>
getInitParameterNames():
getServletContext
注意ServletContext中也有getInitParameter(String),它對應的是:<context-param>中嵌套的元素
這個<context-param>和<servlet>標簽是同級別的,初始化的是整個應用程序
2、兩個類:
GenericServlet/HttpServlet:
GenericServlet實現Servlet和ServletConfig,HttpServlet繼承自GenericServlet,而且它是和HTTP協議相關的。
(1)GenericServlet:具有一個無參的init方法,方便子類的覆蓋。
(2)HttpServlet:具有兩個service方法,并且有7個do方法。重寫的時候一般重寫do方法
3、Servlet配置
指的是web.xml中的信息,
<servlet>
<servlet-name></servlet-name>
<servlet-class></servlet-class>
<init-param></init-param>
</servlet>
//上面的配置對應的可以說就是ServletConfig中的東西
<servlet-mapping>
<servlet-name></servlet-name>
<url-pattern></url-pattern>
</servlet-mapping>
4、WEB應用程序文件夾的結構
--WEB-INF
|
|------------web.xml
|------------lib文件夾 JAR文件
|------------classes 類文件
|------------tags TagFile標簽文件
--靜態資源與JSP文件
5、請求和響應:
(1)請求ServletRequest->HttpServletRequest
getParameter(String)、
getHeader()、
getParameterValues(String)、用于復選框
getParameterNames、
getRemoteAddr、得到遠程地址
getLocalAddr、得到本地地址
getLocale、得到本地化信息
getSession/getCookies
getRequestDispatcher
(2)響應ServletResponse->HttpServletResponse
sendRedirect
sendError
setHeader/addHeader/setIntHeader
getWriter/getOutputStream :這兩個方法絕對不可以同時被調用!!!!!
setContentType/setCharacterEncoding
6、請求的轉發和響應的重定向之間的區別要搞清楚
(1)轉發:
A、三種轉發的方法:
RequestDispatcher.forward/include
<jsp:include>
<jsp:forward> 等價于RequestDispatcher.forward();return;
也就是說使用RequestDispatcher.forward()后,下面的代碼仍然要執行,但是<jsp:forward>執行后,下面的代碼就不執行了。
pageContext.forward/include
B、如何得到RequestDispatcher:
request.getRequestDispatcher
ServletContext.getRequestDispatcher
RequestDispatcher的請求轉發、Jsp頁面中指令元素include、以及ServletContext中的getResourceAsStream(String path)
(用來讀出文件夾中的路徑資源)都可以訪問WEB-INF文件夾,
比如為了防止用戶訪問一些受保護的頁面(比如控制器Servlet),把它們放到WEB-INF中去,請求轉發的時候可以去訪問。
有的時候需要讀取WEB-INF中的一些配置資源,需要ServletContext中的getResourceAsStream方法。
而且有的網站的各個頁面具有相同的頭圖片和尾圖片,這些東東就可以放到WEB-INF文件夾中用page元素的include指令來包含進來。
注意如果使用ServletContext的RequestDispatcher,要從A應用程序轉發請求到B,必須設定A的crossContext值為true!
(2)重定向
response.sendRedirect:比如在注冊結束后轉到下一個頁面的時候,一定要使用重定向改變瀏覽器的URL地址!
請求的轉發不可能脫離Tomcat服務器的范圍,如果想要脫離本服務器的話,只能通過響應重定向的方式。重定向要生成一個臨時的響應,(響應
一旦生成請求就結束了)瀏覽器接到這個臨時的響應后不顯示任何東西而是發送請求去找新的地址。
二、JSP
1、模板和元素:
對于JSP來說,它是不可執行的,必須翻譯成Servlet才能執行,必須要容器特殊處理的叫做元素,
直接打印到輸出流中去的是模板
元素分為以下幾種:
1腳本元素
2指令元素
3動作元素
1腳本元素:
(1)腳本片斷:翻譯后原封不動的放到service方法里面
(2)腳本聲明:放在類里面,但是是在service方法外面,
(3)腳本表達式:原封不動的放到out.print里面去,腳本表達式不可以加分號。
內置對象絕對不可以在腳本聲明中去用!因為它們的作用是在service方法里面的。
2指令元素:
Page指令:
include指令<%@include file=""%>,包含的這個文件一定是按照純文本的格式去讀取,讀取文件的時候就有編碼的問題,這時候就是pageEncoding
的設置問題了。
taglib指令
3頁面亂碼:pageEncoding/contentType
前者設置jsp頁面讀取的形式,后者設置一個響應的報頭,告訴瀏覽器以什么編碼格式去顯示
如果沒有設置pageEncoding而是只設置了contentType的話,那么pageEncoding要受contentType影響,反之亦然。
也就是兩者只設一者的話都會按照一個編碼去顯示!
無論請求還是響應,遞交的時候默認都是按照iso-8859-1去解碼的。一定要保證讀寫和顯示時候的編碼都是一致的
4 動作元素:
<jsp:useBean>:主要作用:開放一個腳本變量并且向作用域里面存一個屬性
<jsp:setProperty>
<jsp:getProperty>
<jsp:forward>
<jsp:include>
<jsp:param>
前六個比較重要!
<jsp:invoke>
<jsp:doBody>
<jsp:plugin>
<jsp:fallback>
<jsp:params>
三、過濾器和監聽器
1、Filter/FilterConfig
init(FilterConfig)
doFilter(ServletRequest,ServletResponse,FilterChain)
destroy()
<filter>
<filter-name>
<filter-class>
</filter>
<filter-mapping>
<filter-name>
<url-pattern>|<servlet-name>
</filter-mapping>
url-pattern相同的過濾器處于一個過濾器鏈上,執行的順序完全按照web.xml中的先后順序進行。
2、監聽器->ServletContext/HttpSession/ServletRequest
(1)生命周期
ServletContextListener:初始化(從BBS中讀取全部討論區并存儲在應用程序的作用域中或是將全部封殺的IP讀出來存儲)和銷毀
HttpSessionListener/HttpSessionActivationListener
ServletRequestListener
(2)屬性的增刪改
一般是AttributeListener去作的
<listener>
<listener-class>包名.類名</listener-class>
</listener>
四、四個作用域和九個內置對象
1、頁面作用域對應的是pageContext,而不是page
2、內置對象:
pageContext/request/session/application
out->是JspWriter的實例,它有緩存,而PrintWriter沒有緩存,在頁面結束的時候一定會通過響應生成PrintWriter去寫出緩存的內容!
如果緩存滿了的話,要看:如果<%@page autoFlush="true"%>,則會自動刷新,如果是false,會拋出異常的!response.getWriter()
返回的是PrintWriter,不是JspWriter。
exception->isErrorPage=true的時候才有用
session:兩種跟蹤機制:一種是通過客戶端的cookie存儲,另外一種是將sessionId存在服務器端,
config->ServletConfig
page->Object->this
response:
五、EL表達式語言:
${}
1、常量、函數、變量的表達式
2、變量是存儲在某一作用域中的同名的屬性值
3、函數是在tld文件中聲明的引用一個類的公共且靜態的方法
4、11個內置對象
(1)作用域:pageScope/requestScope/sessionScope/applicationScope
(2)和頁面通信:pageContext
(3)請求參數的:param/paramValues
(4)報頭:header/headerValues
(5)Cookies/initParam ---<context-param>
5、點操作符和[]操作符和empty
對javaBean,是調用getter方法,
對map,是調用get(Object)方法,
對作用域,是調用getAttribute(String)
對pageContext,也是調用getter方法,可以得到所有的內置對象
對param,是去取得參數
對header,是去拿報頭
對initParam,是去拿初始化參數
[]操作符主要是針對數組
六 自定義標簽
1、Tag接口(傳統接口系列)
JspTag
(1)Tag : doStartTag(返回值是SKIP_BODY/EVAL_BODY_INCLUDE)/doEndTag(返回SKIP_PAGE或是EVAL_PAGE)
(2)IterationTag->TagSupport
doAfterBody:EVAL_BODY_AGAIN/SKIP_BODY
(3)BodyTag->BodyTagSupport
doInitBody
doAfterBody:EVAL_BODY_BUFFERED->BodyContent:pageContext里面有一個pushBody,用來將體包起來,這時候再用getOut方法得到
的不再是JspWriter了,而是bodyContent了。BodyContent的生成是調用了pushBody方法,這里面又涉及一個getEnclosingWriter(),它
是BodyContent的一個方法,可以得到它里面包的輸出流JspWriter,但是要注意要是進行了一次以上的pushBody但是沒有進行popBody的話,那么
拿出來的就是里面包的BodyContent了,但是由于BodyContent是JspWriter的子類,所以也是符合方法聲明的
2、SimpleTag接口(簡單接口系列)
setJspContext()
setParent()
setJspBody(JspFragment) 注意JspFragment是標簽體,里面絕對不可以有腳本元素的!注冊的時候content元素中要么設置為empty,
要么設置為scriptless,也就是簡單標簽的標簽體是不可以有腳本元素的!
doTag()
JspFragment.invoke(null):直接弄到輸出流中去
JspFragment.invoke(StringWriter out):將輸出流弄到StringWriter中去,然后調用out.toString可以將體變成字符串的形式
3、Tag File標簽文件
.tag
可以放在/WEB-INF/tags里面或者/META-INF/tags里面也可以
至于標簽文件的tld可以放在WEB-INF中也可以放在META-INF中去,都是沒有問題的
標簽文件描述的是標簽處理類,jsp描述的是Servlet,二者都是不可運行的,注意標簽文件中有<%@tag%>,里面可以設置不少標簽的屬性
還有<%@attribute %>,<%@taglib%>和<%@include%>也可以用,屬性可以是JspFragment
比如:<my:first>
<jsp:attribute>
</jsp:attribute>
<jsp:body>
</jsp:body>
</my:first>
<jsp:invoke name=var>運算結果就放在了var里面
<jsp:doBody> 處理體
七、JSTL
一共5個庫,我們說了core、sql、i18n、fn標記庫(就是表達式語言的函數庫)
我們沒有講xml的標簽庫
八、國際化與漢字的編碼
ResourceBundle
兩種形式:
1、ListResourceBundle
2、寫一個Properties文件,里面有一個nativetoacsii命令進行,參照前面的筆記轉碼
基名和擴展名:起名字都是要有標準的
Locale:本地化對象
getBundle方法可以通過傳入基名和本地化對象來處理
<fmt:bundle>
<fmt:setBundle>
<fmt:message>
漢字編碼:GB2312/GB13000(GBK)/GB18030
Unicode UCS/UTF-8