一、JSP基礎語法
1、JSP簡介
JSP(Java Server Pages),其根本是一個簡化的Servlet設計,它實現了在Java中使用HTML標簽。JSP是一種動態網頁技術標准,也是JavaEE的標准。JSP和Servlet一樣,是在服務器端執行的。JSP是在Servlet技術發展之后為了讓開發者寫html標簽更方便而發展起來的技術,JSP實際上就是Servlet。
但是,人們通常把Servlet作為Web應用中的控制組件來使用,只負責響應請求產生數據,並把數據通過轉發技術帶給jsp,而把jsp技術作為數據顯示模板來使用。這樣使程序結構更清晰,可讀性和可維護性更高。
2、常用動態網站開發技術
JSP:Java平台,安全性高,適合開發大型的,企業級的,分布式的Web應用程序。如Hadoop,電子銀行,12306等
ASP.net:.Net平台,簡單易學。但是安全性以及跨平台型差。
PHP:簡單,高效,成本低開發周期端,特別適合中小型企業的Web應用開發。(LAMP:Linux+Apache+MySQL+PHP)
3、JSP頁面元素構成
3.1、JSP指令:
JSP指令(directive)是為JSP引擎二設計的,它們並不直接產生任何可見輸出,而只是告訴引擎如何處理JSP頁面中的其余部分。在JSP2.0規范中共定義了三個指令,基本語法格式為,<%@ 指令 屬性名="值" %>,如果一個指令有多個屬性,這多個屬性可以寫在一個指令中,也可以分開寫。
page指令:page指令用於定義JSP頁面的各種屬性,無論指令出現在頁面中的什么地方,它作用的都是整個JSP頁面,為了保持程序的可讀性和遵循良好的編程習慣,page指令通常放在整個JSP頁面的起始位置,一個頁面可以有多個page指令。
<%@ page language="java" contentType="text/html,ISO-8859-1" import="java.util.*,java.sql.*,java.io.*"
session="true|flase" buffer="none|8kb|sizekb" autoFlush="true|false" info="一段字符串內容"
errorPage="relative_url" isErrorpage="true|false"%> 1、language 指定JSP頁面使用的腳本語言,默認值為java 2、contentType 用來指定JSP頁面的文件類型和所采用的編碼方式, 默認值為“text/html,ISO-8859-1” 3、import 通過該屬性來引用腳本語言中使用到的類文件,導入多個類或包用逗號分隔。
JSP引擎自動導入java.lang.*;java.servlet.*;javax.servlet.jsp.*;javax.servlet.http.* 4、pageEncoding 指定JSP頁面的編碼方式, 默認值為“ISO-8859-1” 5、session,用來說明是否創建session對象,默認為true 6、buffer,用來指定out對象是否創建buffer緩沖區,並指定緩沖區大小。默認為8kb,none表示不創建緩沖區。 7、autoFlush,緩沖區是否自動刷新,默認為true,如果為false,緩沖區滿后不手動刷新會包異常。 8、info,定義一個字符串常量,使用getServletInfo方法可以打印。 9、errorPage,指定異常處理頁。也可以在web.xml中使用<error-page>元素為整個WEB應用程序設置處理頁面,
其中的<exception-type>子元素指定異常類的完全限定名,<location>元素指定以"/"開頭的錯誤處理頁面的路徑。
如果設置了某個JSP頁面的errorPage屬性,那么在web.xml文件中這是的錯誤處理將不對該頁面起作用。
include指令:用於引入其他JSP頁面,如果使用include指令引入了其他JSP頁面,那么JSP引擎將把這兩個JSP翻譯成一個Servlet,所以include指令引入通常也成為靜態引入。
<%@ include file=“被包含組件的絕對URL或相對URL"%>
被引入的文件必須遵循JSP語法。被引入的文件可以是任意擴展名,即使其擴展名是html,JSP引擎也會按照處理jsp頁面的方式處理它里面的內容,為了見名知意,JSP規范建議使用.jspf(JSP fragments)作為靜態引入文件的擴展名。由於使用include指明將會涉及到2個JSP頁面,並會把JSO翻譯成一個Servlet,所以這兩個JSP頁面的指令不能沖突(pageEncoding和導包除外)
taglib指令:使用標簽庫定義顯得自定義標簽,在JSP頁面中啟用定制行為。
3.2、表達式:
在JSP頁面中執行的表達式 <%=表達式%>,注意表達式不以分號結束。例如,當前時間:<%= new java.util.Date() %>
3.3、腳本片段:
在JSP頁面中插入多行java代碼 <% Java代碼 %>,JSP引擎在翻譯JSP頁面時,會將JSP腳本片段中的Java代碼原封不動的放到Servlet的_jspServlet方法中,所以,<% %>中的Java代碼必須嚴格遵循java語法。
3.4、聲明:
在JSP頁面中定義變量或者方法 <%! Java代碼 %>,聲明中的Java代碼被翻譯到_jspService方法的外面,屬於類,所以可以聲明成員變量並初始化,也可以聲明方法或定義方法,同時還可以聲明靜態代碼塊。
JSP隱式對象的作用范圍僅限於Servlet的_jspService方法,所以在JSP聲明中不能使用這些隱式對象。
<%! static { System.out.println("loading Servlet!"); } private int globalVar = 0; public void jspInit() { System.out.println("initializing jsp!"); } %> <%! public void jspDestroy() { System.out.println("destroying jsp!"); } %>
3.5、注釋(3種方式):
HTML注釋:<!--html注釋,且客戶端可見-->
JSP注釋:<%--JSP注釋,客戶端不可見--%>
JSP腳本注釋:即java注釋 //單行 , /*多行 */
3.6、靜態內容:
<%@ page language="java" import="java.util.*" contentType="text/html; charset=utf-8"%> <html> <head> <title>index.jsp page</title> </head> <body> <h1>大家好</h1> <hr> <!-- 我是HTML注釋,在客戶端可見 --> <%-- 我是JSP注釋,在客戶端不可見 --%> <%! String s = "張三"; //聲明了一個字符串變量 int add(int x,int y) //聲明了一個返回整型的函數,實現兩個整數的求和。 { return x+y; } %> <% //單行注釋 /*多行注釋*/ out.println("大家好"); %> <br> 你好,<%=s %><br> x+y=<%=add(10,5) %><br> </body> </html>
4、JSP運行原理及生命周期:

用戶第一次請求Tomcat會將jsp文件編程成一個Servlet的java文件,並將java文件編譯成class文件,加載到內存,生成文件在Tomcat的work目錄的對應項目文件夾。如果jsp頁面發生了修改,JSP引擎會對其進行重新編譯並加載到內存,以方便用戶請求。注意,用戶第一次請求一個jsp頁面時,首先被執行的方法是構造方法。
_jspService()方法被調用來處理客戶端的請求。對每一個請求,JSP引擎創建一個線程來處理該請求。如果有多個客戶端同時請求該JSP文件,則JSP引擎會創建多個線程。每個客戶端請求對應一個線程。以多線程方式執行可以大大降低對系統資源的需求,提高系統的並發量及響應時間。但也要主要多線程的編程帶來的額同步問題,由於該Servlet始終駐於內存,所以響應是非常快的。
JSP引擎在調用JSP對應的_jspServlet時,會傳遞或創建9個與web開發相關的對象共_jspServlet使用。JSP技術的設計者為便於開發人員在編寫JSP頁面是獲得這些web對象的引用,特意定義了9個相應的變量,開發人員在JSP頁面中通過這些變量就可以快速獲得這9大對象的引用。
二、JSP 9個內置對象
1、內置對象簡介
JSP內置對象是Web容器創建的一組對象,不使用new關鍵字就可以使用的內置對象。

2、四種作用域范圍
application_SCOPE:作用於整個Web應用,多個用戶之間共享。
session_SCOPE:作用於整個Web應用,單個用戶之間共享。
request_SCOPE:作用於請求,轉發間請求共享。
page_SCOPE:作用於當前頁面,當前頁面可見。
3、out對象
緩沖區:Buffer,即內存中的一塊區域用來保存臨時數據
out對象:是JspWriter類的實例,是向客戶端發送文本常用的對象,是通過調用pageContext對象的getOut方法返回的,其作用和用法與ServletResponse.getWriter方法返回的PrintWriter對象非常相似。JSP頁面中的out隱式對象的類型為JspWriter,JspWriter相當於一種帶緩存功能的PrintWriter,設置JSP頁面的page指令的buffer屬性可以調整它的緩存大小,甚至關閉它的緩存。當緩沖區滿足如下條件是才會寫入Servlet引擎提供的緩沖區:設置page指令的buffer屬性關閉了out對象的緩存功能;out對象的緩沖區已滿。整個JSP頁面結束。
<% out.println("aaa"); response.getWriter().write("bbb"); %> "bbb會比aaa先輸出"
常用方法:1、void println(); 向客戶端打印字符串 2、void clear(); 清除緩沖區的內容,如果在flush之后調用會拋出異常。 3、void clearBuffer(); 清除緩沖區的內容,如果在flush之后調用不會拋出異常 4、void flush(); 將緩沖區內容輸出到客戶端 5、int getBufferSize(); 返回緩沖區以字節數的大小,如不設緩沖區則為0 6、int getRemaining(); 返回緩沖區還剩余多少可用 7、boolean isAutoFlush(); 返回緩沖區滿是,是自動清空還是拋出異常 8、void close(); 關閉輸出流
4、request/response對象
請求響應機制請參考本博客的Servlet部分。
4.1、表單的兩種提交方式:get與post
get:以明文的方式通過URL提交數據,數據在URL中可以看到。提交的數據最多不超過2KB。安全性較低,但效率比post方式搞。適合提交數據量不大,安全性高的數據。比如:搜索、查詢等功能
post:將用戶提交的信息封裝在HTML header內。適合提交數據量大,安全性高的用戶信息。比如:注冊、修改、上傳等功能。
4.2、request對象
客戶端的請求信息被封裝在request對象中,通過它才能了解到客戶端的需求,然后做出響應。它是HttpServletRequest類的實例。request對象具有請求域,即完成客戶端的請求之前,該對象一直有效。
//request常用方法:
String getParameter();返回指定參數的參數值
String[] getParameterValues();返回指定參數的所有值
void setAttribute();存儲此請求中的屬性
object getAttribute();返回指定屬性的屬性值
String getContentType();得到請求體的MIME類型
String getProtocol();返回請求用的協議及版本號
String getServerName();返回接受請求的服務器主機名
int getServerPort();返回服務器接受此請求的端口號
String getCharacterEncoding();返回字符編碼方式
void setContentEncoding();設置請求的字符編碼方式
int getContentLength();返回請求體的長度(以字節數)
String getRemoteAddr();返回發送此請求的客戶端IP地址
String getRealPath(String path);返回一個虛擬路徑的真實路徑或相對路徑的絕對路徑
StringgetContextPath();返回上下文路徑,即項目的根目錄
中文亂碼問題:request.setCharacterEncoding("UTF-8");
URL中中文亂碼問題:Tomcat的/conf/server.xml 的添加<Connector URIEncoding="UTF-8">屬性。
4.3、response對象
response對象包含了響應客戶請求的有關信息,但在JSP中很少直接用到它。它是HttpServletResponse類的實例。response對象具有頁面作用域,及訪問一個頁面時,該頁面內的response對象只能對這次訪問有效,其他的response對象對當前頁面無效。
//response對象的常用方法
String getCharacterEncoding();返回響應用的是何種字符編碼
void setContentType();設置響應的MIME類型,一般為"text/html, charset=UTF-8"
PrintWriter getWriter();返回可以向客戶端輸出字符的一個對象("注意比較:PrintWriter與內置out對象的區別,PrintWriter對象的其輸出總是提前於內置out對象,或者在out中手動flush")
sendRedirect(loaction);重新定向客戶端的請求
請求重定向:是客戶端行為,response。sendRedirect(),從本質上將等同於兩次請求,前一次的請求對象不會保存,地址欄的URL地址會改變。
請求轉發:是服務器行為,request.getRequestDispatcher().forward();是一次請求,轉發后請求對象會保存,地址欄的URL地址不會改變。
5、session對象
session表示客戶端與服務器的一次會話,Web中的session指的是用戶在瀏覽某個網站時,從進入網站到瀏覽器關閉所經過的這段時間,也就是用戶瀏覽這個網站所花費的時間。
在服務器的內存中,為不同的用戶保存着不同的session。
session對象是一個JSP內置的對象。
session對象在第一個JSP頁面被裝載是自動創建,完成會話期管理。
從一個客戶打開瀏覽器並連接到服務器開始,到客戶關閉瀏覽器離開這個服務器結束,被稱為一個會話。
當一個客戶訪問一個服務器時,肯能會在服務器的幾個頁面之間切換,服務器應當通過某種辦法知道這是一個客戶,就需要session對象。
session對象是HttpSession類的實例。
//session的常用方法
long getCreationTime();返回session創建時間
String getId();返回session創建時JSP引擎為它設的唯一ID號
Object setAttribute();使用指定名稱將對象綁定到此會話
Object getAttribute();返回與此會話中的指定名稱綁定在一起的對象,沒有返回null
String[] getValueNames();返回一個包含此session中所有可用屬性的數組
int getMaxInactiveInterval();返回兩次強求間隔多長時間此session被取消(單位秒)
void setMaxInactiveInterval();設置session存活時間。
session的生命周期:
創建:當客戶端第一次訪問某個jsp或者Servlet時候,服務器會為當前會話創建一個SessionId,每次客戶端向服務端發送請求時,都會將此SessionId攜帶過去,服務端會對此SessionId進行校驗。
活動: 某次會話當中通過超鏈接打開的新頁面屬於同一次會話。只要當前會話頁面沒有全部關閉,重新打開新的瀏覽器窗口訪問同一項目資源時屬於同一次會話。除非本次會話的所有頁面都關閉后,在重新訪問某個jsp或者Servlet將會創建新的會話(但此時原有會話還存在,這個就的SessionId仍然存在於服務端,只不過再也沒有客戶端會攜帶它然后教育服務端校驗,直到該會話超時。)。
銷毀:有三種銷毀方式
1、調用了session.invalidate()方法
2、Session過期(超時)可以在web.xml中配置Session超時時間<session-config><session-timeout>1單位是分鍾
3、服務器重新啟動
6、application對象
application對象實現了用戶間數據的共享,可存放全局變量。
application開始於服務器的啟動,終止於服務器的關閉。
在用戶的前后連接或不同用戶之間的連接中,可以對application對象的同一屬性進行操作。
在任何地方對application對象屬性的操作,都將影響到其他用戶對此的訪問。
服務器的啟動和關閉決定了application對象的生命。
application對象是ServletContext類的實例。
//application的常用方法
void setAttribute();使用指定名稱將對象綁定到此會話
Object getAttribute();返回與此會話中的指定名稱綁定在一起的對象,如果沒有,返回null
Enumeration getAttributeNames();返回所有可用屬性名的枚舉
String getServerInfo();返回JSP(Servlet)引擎名及版本號
7、page對象
page對象就是指向當前jsp頁面本身,有點像類中的this指針,它是java.lang.Object類的實例。常用方法就是Object類的成員方法。
8、pageContext對象
pageContext對象是JSP技術中最重要的一個對象,它代表JSP頁面的運行環境,這個對象不僅封裝了對其他8個隱式對象的引用,其自身還是一個域對象,可以用來保存數據。並且,這個對象還封裝了web開發中經常涉及到的一些常用操作,例如include,forward其他資源、檢索其他域對象中的屬性等。
pageContext對象提供了對jsp頁面內所有的對象及名字空間的訪問。該對象可以訪問到本頁所在的Session,也可以取本頁所在的application中的屬性值。該對象相當也頁面中所有功能的集大成者。該對象的本類名也叫pageContext。
//pageContext的常用方法
JspWriter getOut();返回當前客戶端響應被使用的out對象
HttpSession getSession();返回當前頁中的Session對象
Object getPage();返回當前頁的page對象
ServletRequest getRequest();返回當前頁的Request對象
ServletResponse getResponse();返回當前頁的Response對象
void setAttribute();設置屬性及屬性值
Object getAttribute();在指定范圍內取屬性的值
int getAttributeScope();返回某屬性的作用范圍
void forward();跳轉到另一個頁面,地址欄不變
void include();包含另一個頁面,
PageContext.APPLICATION_SCOPE 代表各個域的常量
PageContext.SESSION_SCOPE
PageContext.REQUEST_SCOPE
PageContext.PAGE_SCOPE
findAttribute(); 查找各個域中的屬性
9、Config對象
Config對象是在一個Servlet初始化時,JSP引擎向它傳遞信息用的,此信息包括Servlet初始化時所要用到的參數(通過屬性名和屬性值構成)以及服務器的有關信息(通過傳遞一個ServletContext對象)
//Config的常用方法
ServletContext getServletContext();返回含有服務器信息的ServletContext對象
String getInitParameter();返回初始化參數的值
Enumeration getInitparameterNames();返回Servlet初始化所需所有參數的枚舉
10、Exception對象
Exception對象是一個異常對象,當一個頁面在運行過程中發生了異常,就產生這個對象。如果一個jsp頁面要應用此對象,就必須報isErrorPage設為true,否則無法編譯。它實際上是java.lang.Throwable的對象。頁面要想在有異常時得到處理,就要指定異常處理頁面<% page errorPage="exception.jsp"%>
//exception的常用方法
String getMessage();返回描述異常的消息
String toString();返回關於異常的簡短描述消息
void printStackTrace();顯示異常及其棧軌跡
ThrowableFillInStackTrace();重寫異常的執行棧軌跡
三、JSP動作標簽
1、JSP動作元素(action elements):動作元素為請求處理階段提供信息。動作元素遵循XML元素的語法,有一個包含元素名的開始標簽,可以有屬性、可選的內容、與開始標簽匹配的結束標簽。
JSP標簽也稱之為JSP Action(JSP動作)元素,它用於在JSP頁面中提供業務邏輯功能,避免在JSP頁面中直接編寫java代碼,造成jsp頁面難以維護。
2、JSP動作元素包含五大類:
1、與存儲JavaBean有關的:<jsp:useBean>, <jsp:setProperty>, <jsp:getProperty>
2、JSP1.2規范就有的6個基本動作元素:<jsp:include>, <jsp:forward>, <jsp:param>, <jsp:plugn>, <jsp:params>, <jsp:fallback>
3、JSP2.0新增加,主要與JSP Document有關的6個動作元素:<jsp:root>, <jsp:declaration>, <jsp:scriptlet>, <jsp:exception>, <jsp:text>, <jsp:output>
4、JSP2.0新增加,主要永磊動態生成XML元素標簽的值,包括3個動作:<jsp:attribute>, <jsp:body>, <jsp:element>
5、JSP2.0新增加,只要用在Tag File中:<jsp:invoke>, <jsp:dobody>
3、JSP常用標簽
<jsp:include>:該標簽用於把另外一個資源的輸出內容插入進當前JSP頁面的輸出內容中,這種在JSP頁面執行時的引入方式稱之為動態引入。<jsp:include page="relativeURL | <%=expression%>" flush="true|false" />。flush屬性指定在插入其他資源的輸出內容時,是否先將當前JSP頁面的已輸出的內容刷新到客戶端。
<jsp:include>與include指令比較:<jsp:include>標簽是動態引入, <jsp:include>標簽涉及到的2個JSP頁面會被翻譯成2個servlet,這2個servlet的內容在執行時進行合並。而include指令是靜態引入,涉及到的2個JSP頁面會被翻譯成一個servlet,其內容是在源文件級別進行合並。不管是<jsp:include>標簽,還是include指令,它們都會把兩個JSP頁面內容合並輸出,所以這兩個頁面不要出現重復的HTML全局架構標簽,否則輸出給客戶端的內容將會是一個格式混亂的HTML文檔。但include指令要比<jsp:include>標簽效率高些。可以從下面的表格中更直觀的看到兩者的區別

<jsp:forward>:該標簽用於把請求轉發給另外一個資源。<jsp:forward page="relativeURL | <%=expression%>" />。等同於request.getRequestDispatcher("/url").forward(request,response)。即執行服務器內部跳轉操作。
<jsp:param>:當使用<jsp:include>和<jsp:forward>標簽引入或將請求轉發給其它資源時,可以使用<jsp:param>標簽向這個資源傳遞參數。
-
<jsp:include page="relativeURL | <%=expression%>"> <jsp:param name="parameterName" value="parameterValue|<%= expression %>"/> </jsp:include> 可以使用多個<jsp:param>標簽來傳遞多個參數。
4、映射JSP
映射JSP就是將一個JSP映射成其他任意形式的URL,在web.xml文件張配置
-
<servlet> <servlet-name>SimpleJspServlet</servlet-name> <jsp-file>/jsp/simple.jsp</jsp-file> <load-on-startup>1</load-on-startup > </servlet> …… <servlet-mapping> <servlet-name>SimpleJspServlet</servlet-name> <url-pattern>/xxx/yyy.html</url-pattern> </servlet-mapping>
-
四、JavaBeans
1、Javabean簡介
Javabeans就是符合某種特定規范的Java類。使用Javabeans的好處是解決代碼重復編寫,減少代碼冗余,功能區分明確,提高了代碼的可維護性。
2、Javabean設計原則
必須是共有類;
必須包含一個無參的共有構造方法;
所有屬性必須私有;
使用getter和setter訪問器對屬性訪問封裝。
public class Book { //一個符合要求的Javabean類 private String bookName; private String author; private double price; public Book() { } public String getBookName() { return bookName; } public void setBookName(String bookName) { this.bookName = bookName; } public String getAuthor() { return author; } public void setAuthor(String author) { this.author = author; } public double getPrice() { return price; } public void setPrice(double price) { this.price = price; }
3、JSP中如何使用Javabean
第一種方式:像使用普通java類一樣,創建javabean實例。
第二種方式:在JSP頁面中通常使用jsp動作標簽使用javabean。
4、<jsp:useBeans>
作用:在jsp頁面中實例化或者在指定范圍內使用javabean
<jsp:useBean id="標識符" class="java類名" scope="作用范圍"/> scope默認為page。
5、<jsp:setProperty>
作用:給已經實例化的Javabean對象的屬性賦值,一共有四種形式:
<jsp:setProterty name="Javabean實例名" property="*"/>(跟表單關聯)
<jsp:setProterty name="Javabean實例名" property="Javabean屬性名"/>(跟表單關聯)
<jsp:setProterty name="Javabean實例名" property="Javabean屬性名" value="BeanValue"/>(手工設置)
<jsp:setProterty name="Javabean實例名" property="propertyName" param="request對象中的參數名"/>(跟request參數關聯)通過url地址傳遞的參數
6、<jsp:getProperty>
作用:獲取指定Javabean對象的屬性值。
<jsp:getProperty name="Javabean實例名" property="屬性名"/>
7、Javabean的四個作用域范圍
使用useBeans的scope屬性可以用來指定javabean的作用范圍。
page:僅在當前頁面有效
request:可以通過request.getAttribute方法取得JavaBean對象
session:可以通過session.getAttribute方法取得JavaBean對象
application:可以通過application.getAttribute方法取得JavaBean對象
8、Model1簡介
Model1出現前,整個Web應用幾乎全部由JSP頁面組成,JSP頁面接收處理客戶端請求,對請求處理后直接作出響應。
弊端:在界面層充斥着大量的業務邏輯的代碼和數據訪問層的代碼,Web程序的可擴展性和可維護性非常差。
Javabean的出現可以使jsp頁面中使用Javabean封裝的數據或者調用Javabean的業務邏輯代碼,這樣大大提升了程序的可維護性。

9、Model2簡介

五、JSP狀態管理
1、http協議的無狀態性
無狀態是指,當瀏覽器發送請求該服務器的時候,服務器響應客戶端請求。但是當同一個瀏覽器再次發送請求給服務器的時候,服務器並不知道它就是剛才那個瀏覽器。簡單的說就是服務器不會記得你,所以就是無狀態協議。
2、保存用戶狀態的兩大機制
Cookie技術、Session技術
3、Cookie簡介
Cookie:是Web服務器保存在客戶端的一系列文本信息。
典型應用:判定注冊用戶是否已經登錄網站。
購物車的處理。
Cookie的作用:
對特定對象的追蹤
保存用戶網頁瀏覽記錄與習慣
簡化登錄
安全風險:容易泄露用戶隱私信息。
4、Cookie的創建與使用
Cookie與Session的詳細講解請參考本博客文章《Servlet---JavaWeb技術的核心基礎,JavaWeb框架的基石(二)》中的Cookie與Session部分。
5、Session與Cookie的對比
共同點:都是用來保存用戶狀態的;都會過期
Session:在服務器端保存用戶信息,保存的是Object類型,隨會話的結束而將其存儲的數據銷毀,保存重要的信息。
Cookie:在客戶端保存用戶信息,保存的是String類型,可以長期保存在客戶端,保存不重要的用戶信息。