其他更多java基礎文章:
java基礎學習(目錄)
本章內容較多,全部認真看完可能需要一小時以上,建議邊看邊做筆記,否則容易混亂
一、JSP概述
1.1、JSP結構
網絡服務器需要一個JSP引擎,也就是一個容器來處理JSP頁面。容器負責截獲對JSP頁面的請求。內嵌JSP容器的Apache支持JSP開發。
JSP容器與Web服務器協同合作,為JSP的正常運行提供必要的運行環境和其他服務,并且能夠正確識別專屬于JSP網頁的特殊元素。
下圖顯示了JSP容器和JSP文件在Web應用中所處的位置。
1.2、JSP運行流程
以下步驟表明了Web服務器是如何使用JSP來創建網頁的:
1)就像其他普通的網頁一樣,您的瀏覽器發送一個HTTP請求給服務器。Web服務器識別出這是一個對JSP網頁的請求,并且將該請求傳遞給JSP引擎。通過使用URL或者.jsp文件來完成。
2)JSP引擎從磁盤中載入JSP文件,然后將它們轉化為servlet。這種轉化只是簡單地將所有模板文本改用println()語句,并且將所有的JSP元素轉化成Java代碼。
3)JSP引擎將servlet編譯成可執行類,并且將原始請求傳遞給servlet引擎。
4)Web服務器的某組件將會調用servlet引擎,然后載入并執行servlet類。在執行過程中,servlet產生HTML格式的輸出并將其內嵌于HTTP response中上交給Web服務器。Web服務器以靜態HTML網頁的形式將HTTP response返回到您的瀏覽器中。
最終,Web瀏覽器處理HTTP response中動態產生的HTML網頁,就好像在處理靜態網頁一樣。
以上提及到的步驟可以用下圖來表示:
一般情況下,JSP引擎會檢查JSP文件對應的servlet是否已經存在,并且檢查JSP文件的修改日期是否早于servlet。如果JSP文件的修改日期早于對應的servlet,那么容器就可以確定JSP文件沒有被修改過并且servlet有效。這使得整個流程與其他腳本語言(比如PHP)相比要高效快捷一些。
Jsp 生成java源碼,默認第一次生成,之后直接執行,除非內容修改,具體點說,由于JSP只會在客戶端第一次請求的時候被編譯,因此第一次請求JSP時會感覺比較慢,而之后的請求因為不會編譯JSP,所以速度就快多了。
如果將Tomcat保存的JSP編譯后的class文件刪除,Tomcat也會重新編譯JSP。在開發Web程序的時候經常需要修改JSP, Tomcat能夠自動檢測到JSP程序的改動,如果檢測到JSP源代碼發生了改動,Tomcat會在下次客戶端請求JSP時重新編譯JSP,而不需要重啟Tomcat,這種自動檢測功能默認是開啟的,檢測改動會消耗少量的時間,在部署web應用程序的時候可以在web.xml中將它關掉。
這也就是為什么我們能夠在jsp頁面直接修改內容,而不用重新啟動服務器的原因。
總的來說,JSP網頁就是用另一種方式來編寫servlet而不用成為Java編程高手。除了解釋階段外,JSP網頁幾乎可以被當成一個普通的servlet來對待。
1.3、JSP生命周期
理解JSP底層功能的關鍵就是去理解它們所遵守的生命周期。JSP生命周期就是從創建到銷毀的整個過程,類似于servlet生命周期,區別在于JSP生命周期還包括將JSP文件編譯成servlet。
以下是JSP生命周期中所走過的幾個階段:
編譯階段:
servlet容器編譯servlet源文件,生成servlet類
初始化階段:
加載與JSP對應的servlet類,創建其實例,并調用它的初始化方法
執行階段:
調用與JSP對應的servlet實例的服務方法
銷毀階段:
調用與JSP對應的servlet實例的銷毀方法,然后銷毀servlet實例
很明顯,JSP生命周期的四個主要階段和servlet生命周期非常相似,下面給出圖示:
二、JSP語法
2.1、jsp腳本
1)使用<% 編寫java代碼 %>,中間java代碼必須遵循Java語法
來看看,jsp變為servlet時的代碼是如何編寫的
2)使用<%=xxx %>來輸出結果
使用<%=result %>來輸出結果,servlet中就會將其轉換為out.print(result)進行輸出。輸出各種類型數據:int、double、boolean、String、Object等。
3)注釋
<%-- --%>:jsp注釋
<!-- -->:這個注釋,會發送到瀏覽器端的源碼中顯示
注釋分別在servlet中如何顯示:
在servlet中
總結:JSP注釋不會在servlet文件中顯示,而java注釋則會,但其所有的注釋到了瀏覽器端,都不會出現在源碼中,只有這個注釋會到瀏覽器的網頁源碼中去。
4)JSP中申明方法與屬性(全局變量) 使用<%! 方法、屬性%>
2.2、3個指令
JSP指令(directive)是為JSP引擎而設計的,它們并不直接產生任何可見輸出,而只是告訴引擎如何處理JSP頁面中的其余部分。指令用來申明JSP頁面的一些屬性,比如編碼方式,文檔類型。我們在servlet中也會申明我們使用的編碼方式和響應的文檔類型的,而JSP就是用指令來申明。
JSP指令格式:<%@ directive {attribute=value}* %>(<%@ 指令名稱 屬性1=“屬性值1” 屬性2=“屬性值2”。。。%>)
分析:
directive:指令名稱,例如page指令
attribute=value:緊跟指令名稱后面的就是各種屬性,以鍵值對的形式書寫
*:代表后面能跟0個或多個屬性。
2.2.1、page指令(用來聲明JSP頁面的屬性等)
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> page指令,后面跟著三個屬性,分別是language、contentType、pageEncoding。
這只是其中的幾個屬性,并沒有寫全,page指令允許的屬性如下表所示:
屬性名稱 | 取值范圍 | 描述 |
---|---|---|
language | java | 解釋該JSP文件時采用的語言,一般為java語言,默認為java |
extends | 任何類的全名 | 編譯該JSP文件時繼承哪個類,JSP為Servlet,因此當指明繼承普通類時需要實現Servlet的init、destroy等方法 |
import | 任何包名、類名 | 引入該JSP中用到的類、包等,import是唯一可以聲明多次的page指令屬性,一個import可以引用uogelei,中間用英文逗號隔開,如<%@ page import="java.util.List,java.util.ArrayList"%> |
session | true、false | 該JSP內是否內置Session對象,如果為true,則內置Session對象,可直接使用,否則反之,默認為true |
autoFlush | true,false | 是否運行緩存,如果為true,則使用out.println()等方法輸出的字符串并不是立刻到達客戶端服務器的,而是暫時存到緩存里,緩存滿了或者程序執行完畢或者執行out.flush()操作時才到客戶端,默認為true。 |
buffer | none或者數字KB | 指定緩存大小,當autoFlush設為true時有效,例如<%@ page buffer=10kb%> |
isThreadSafe | true,false | 是否線程安全,如果為true,則運行多個線程同時運行該jsp程序,否則只運行一個線程,其余線程等待,默認為false |
isErrorPage | true,false | 指定該頁面是否為錯誤顯示頁面,如果為true,則該JSP內置有一個Exception對象exception,可直接使用,否則沒有,默認為false |
errorPage | 某個JSP頁面的相對路徑 | 指明一個錯誤頁面,如果該JSP程序拋出一個未捕捉的異常,則轉到errorPage指定的頁面,errorPage指定的頁面通常isErrorPage屬性為true,且內置的exception對象為未捕捉的異常 |
contentType | 有效的文檔類型 | 客戶端瀏覽器根據該屬性判斷文檔類型,例如 HTML格式為text/html、純文本格式為text/plain、JPG圖像為image/jpeg、GIF圖像為image/gif、WORD文檔為application/msword,該屬性常跟著charset設置編碼一起,作用是通知服務器和瀏覽器都使用同一個碼表 |
info | 任意字符串 | 指明JSP的信息,該信息可以通過Servlet.getServletInfo()方法獲取到 |
trimDirective Whitespaces | true、false | 是否去掉指令前后的空白字符,默認為false |
pageEncoding | UTF-8,ISO-8859-1等 | 指定一張碼表來對該JSP頁面進行編碼 |
2.2.2、include指令
比較簡單,只有一種形式 <%@ include file="relativeURL"%> relativeURL:本應用程序內另一個JSP文件或者HTML文件的路徑,例如,網址內所有頁面均有一個統一風格的導航欄和頁腳版權,那么就可以使用該指令將其包含進來。
特點:include指令會將包含頁面的源代碼添加到使用include指令的頁面中來,然后編譯成class文件,而等下會講到的一個JSP行為,<jsp:include page="relativeURL">作用跟include指令一樣,但是不同的是,include行為是運行時單獨執行包含頁面,然后把執行的結果包含到本頁面來,屬于先運行后包含。
注意
靜態包含:把其它資源包含到當前頁面中。
<%@ include file="/include/header.jsp" %>
動態包含:
<jsp:include page="/include/header.jsp"></jsp:include>
兩者的區別:翻譯的時間段不同
前者:在翻譯時就把兩個文件合并
后者:不會合并文件,當代碼執行到include時,才包含另一個文件的內容。
原則:能用靜的就不用動的。
2.2.3、taglib指令
JSP支持標簽技術,后面會講到標簽的用法,jstl標簽庫的使用等
作用:用來指明JSP頁面內使用的JSP標簽庫,taglib指令有兩個屬性,uri為類庫的地址,prefix為標簽的前綴
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
2.3、6個動作
前面講了JSP語法,介紹了JSP頁面中的內容有哪些,分別有什么作用,就兩個東西,模塊數據和元素。其中元素有包括腳本,指令,標簽,腳本就是JSP中嵌入java代碼,指令作用就是申明頁面的屬性,那標簽是干嘛的,標簽分為JSP自帶內置的標簽,和通過taglib指令來使用JSP標簽庫,或者自定義標簽。現在我們先來講一些JSP內置的標簽。
JSP內置的標簽就被稱為JSP行為(JSP Actions)。只要書寫很少的標記代碼就能使用JSP提供的豐富功能,JSP行為其實是對常用的JSP功能的抽象與封裝,可以取代jsp腳本,讓JSP中就少一些嵌入java代碼的地方。
簡單的說就是使用標簽的形式來表示一段java代碼,格式:
<jsp:elements {attribute="value"}* />
分析:
jsp:標簽的前綴,說明是jsp內置的標簽,
elements:行為的名稱,
attribute=value:使用鍵值對來編寫屬性
*:能指定0個或多個屬性對
2.3.1、<jsp:include />行為(動態包含)
<jsp:include page="/include/header.jsp"></jsp:include>
include行為用于運行時包含某個文件,如果被包含的文件為JSP程序,則先會執行JSP程序,然后在把執行的結果包含進來。
作用是跟include指令一樣的,唯一的區別就在于,include指令是將被包含的文件的源碼加入到了本JSP程序中,然后在進行編譯,屬于靜態包含,而include行為只是將被包含的文件的運行結果包含進自己。屬于動態包含。
2.3.2、Java bean行為
是一組與Java Bean 相關的行為,包括useBean行為、setProperty行為、getProperty行為等。Java Bean就是普通的Java類,也被稱為POJO,只有私有的屬性與對應的getter方法和setter方法,注意其中當私有的屬性為boolean類型時,習慣上一般把getter方法寫成isXxx();而不是getXxx();
1)userBean行為
<jsp:useBean id="beanObject" class="className" scope="Value"> 作用:在jsp中定義一個java bean對象。
分析:
id:指明Java Bean對象的名稱,JSP中可以使用該名稱引用該Java Bean對象,相當于給new出來的對象取一個變量名,
class:Java Bean類的全名
scope:該java bean對象的作用范圍,可以寫的就四個,也就是JSP的四大作用域,page、request、session、application
---page:只能在當前JSP頁面使用,如果不在JSP頁面,那么就會失效
---request:這個前面學過,A頁面請求轉發到B頁面,那么使用的是同一個request,那么A,B頁面都算是request的作用域,也就是通過請求轉發的頁面都是其作用域
----session:該作用域在一個web項目下任何位置應該讀訪問的到,只要cookie不關閉,并且cookie設置的訪問路徑為"/",
---application:其實就是Servlet中的servletContext,服務器下的所有項目都能訪問到。
2)setProperty行為
<jsp:setProperty name="beanName" property="propertyName" value="">
分析:
對Java Bean對象進行屬性的設置
name:java bean對象的名稱,也就是在useBean行為中的id
property:對象中的屬性名,
value:要對其屬性進行賦值的值
3)getProperty行為
<jsp:getProperty name="beanName" property="propertyName" />
分析:
獲取JavaBean對象的某個屬性值
name:java bean 對象的名稱,也就是在useBean行為中的id
property:對象的屬性名
2.3、<jsp:forward />行為
實現請求轉發功能,Servlet中通過request.getRequestDispatcher("someServlet").forward(request,response);而在JSP中也能夠實現相同的功能,只不過用的是<jsp:forward />行為,實際上forward行為就是對其進行了封裝。
格式:
<jsp:forward page="someServlet">
<jsp:param name="param1" value="value1"/>
<jsp:param name="param2" value="value2"/>
</jsp:forward>
分析:page:需要跳轉到的頁面或者servlet、<jsp:param/>參數行為,帶一些參數過去,name、value是以鍵值對的形式帶過去的
2.4、9個內置對象
我們知道JSP中的內容就只有兩種,模版數據和元素,元素就包括了指令,腳本,標簽(行為),腳本會慢慢被標簽全部代替,也就是說JSP中基本上不會嵌入Java代碼,但是我們也知道JSP會轉換為servlet,在Servlet中,輸出數據時,都需要通過response.getWrite();但是在JSP中,直接使用out對象進行輸出,為什么呢?這就是因為out為JSP的一個隱藏對象,JSP中內置了9個隱藏對象,使得JSP比Servlet使用起來更簡單,更方便。
2.4.1、九大內置對象概述
分析:
request:請求對象,類型:httpServletRequest
response:響應對象,類型:httpServletResponse
session:表示一次會話,在服務器端記錄用戶狀信息的技術
application:標識web應用上下文,類型:ServletContext,詳情就看Servlet中的ServletContext的使用
exception:表示發生異常對象,類型 Throwable,在上面我們介紹page指令中的一個errorPage屬性時就有說到他
page:page對象代表當前JSP頁面,是當前JSP編譯后的Servlet類的對象。相當于this。
config:標識Servlet配置,類型:ServletConfig,api跟Servlet中的ServletConfig對象是一樣的,能獲取該servlet的一些配置信息,能夠獲取ServletContext
out:輸出響應體 類型:JspWriter
pageContext:表示 jsp頁面上下文(jsp管理者) 類型:PageContext
注意:標記了紅色的對象就是JSP獨有的,其他的都是Servlet中的老東西。
在這個由jsp轉換為servlet的文件中,只能看到8個內置對象,少了exception對象,因為我們在將page指令時,說過一個isErrorPage屬性,默認是false,被關閉了,所以其中并沒有exception對象。
2.4.2、pageContext(重要)
這個功能就比較強大了,基本上什么他都有,因為是它是JSP頁面的管理者(上下文),所以JSP中的內置對象呀,它統統能夠獲得,下面介紹它的api:
1)獲得其它八大內置對象 getXxx()
在普通類中可以通過PageContext獲取其他JSP隱式對象。自定義標簽時就使用。
pageContext.getOut(); //獲得out對象
pageContext.getApplication(); //獲得application對象
等等....
2)對作用域的屬性進行操作(四大作用域)
對默認作用域的屬性進行操作。page
Object getAttribute(String name); //獲得page作用域數據
void setAttribute(String name,Object o); //給page作用域設置內容
void removeAttribute(String name); //給page作用域移除內容
3)對指定作用域的屬性進行操作
Object getAttribute(String name,int Scope); //獲得 指定作用域中的數據
void setAttribute(String name,Object o,int Scope); //給指定作用域設置內容
void removeAttribute(String name,int Scope); // 移除指定作用域的內容(page/request/session/application)
4)提供作用域常量
PageContext.PAGE_SCOPE page
PageContext.REQUEST_SCOPE request
PageContext.SESSION_SCOPE response
PageContext.APPLICATION_SCOPE application
5)一次獲得指定名稱內容
pageContext中最厲害的方法是:
findAttribute(String name); //自動從page request session application依次查找,找到了就取值,結束查找。
6)提供了的簡易方法
pageContext.forward("2.jsp");
pageContext.include("2.jsp");
2.4.3、out對象
類型:JspWriter
jsp 輸出底層使用 response.getWriter();什么意思呢?這里就要講解一下JSP緩存和Servlet緩存了,輸出的過程是這樣的
JSP頁面轉換為Servlet后,使用的out對象是JspWriter類型的,所以是會先將要發送的數據存入JSP輸出緩存中,然后,等JSP輸出緩存滿了在自動刷新到servlet輸出緩存等serlvet輸出緩存滿了,或者程序結束了,就會將其輸出到瀏覽器上。除非手動out.flush()。
驗證servlet輸出緩存和JSP輸出緩存和我們上面所說的是正確:
結果:
分析:
如果按沒有jsp緩存和servlet緩存的話,輸出的結果應該是aaaabbbbcccc,但是輸出的卻是bbbbaaaacccc,為什么呢?按照我們上面所說的原理進行分析,out對象是先將其輸出到JSP緩存中,所以aaaa加入了jsp緩存,而response.getWriter().print("bbbb")是直接將bbbb輸出到servlet緩存中,然后又使用out對象將cccc輸出到jsp緩存,到程序結束,servlet緩存中有bbbb,然后jsp會將緩存中的內容就刷新到servlet緩存中,serlvet就是bbbbaaaacccc了,然后到瀏覽器也就得到我們的輸出結果了。如果在12行將注釋去掉,那么輸出的結果又會是什么呢?答案就是aaaabbbbcccc,過程自行分析。
2.4.4、config對象
類型:ServletConfig
能夠獲取servlet的初始化參數,獲取servletContext對象,獲取servletName。
2.4.5、exception異常對象
包含了異常的信息
使用它,必須結合page指令中的isErrorPage屬性和errorPage屬性。
如下例子,exception.jsp拋異常的一個NullPointException,并且跳轉到error.jsp錯誤顯示頁面,其中errorPage屬性的意思是如果發生未捕捉到的異常,將會跳轉到error.jsp頁面
error.jsp isErrorPage屬性說明該頁面是一個錯誤顯示頁面,則可以使用exception對象
訪問:訪問http://localhost:8080/Web_Jsp/exception.jsp
2.4.6、總結:九大內置對象和servlet中對象的關系
page就是jsp轉換為servlet對象本身,也就是this
config -- Servlet中的servletConfig
application -- Servlet中的ServletContext
request -- Servlet中的request
response -- Servlet中的response
session -- Servlet中的session
out -- JspWriter
exception -- 異常對象
pageContext -- 表示 jsp頁面上下文(jsp管理者) 類型:PageContext
其中pageContext是最厲害的,因為它可以得到其他8個內置對象
2.5、4大作用域
這四大作用域,其實就是其九大內置對象中的四個,為什么說他們也是JSP的四大作用域呢?
因為這四個對象都能存儲數據,比如request.setAttribute()注意和request.setParameter()區分開來,一個是存儲在域中的、一個是請求參數,session.setAttribute()、application其實就是SerlvetContext,自然也有setAttribute()方法。
而page作用域的操作就需要依靠pageContext對象來進行了。在上面我們也有提到JSP的四大作用域。
1)page作用域
代表變量只能在當前頁面上生效
2)request作用域
代表變量能在一次請求中生效,一次請求可能包含一個頁面,也可能包含多個頁面,比如頁面A請求轉發到頁面B。
3)session作用域
代表變量能在一次會話中生效,基本上就是能在web項目下都有效,session的使用也跟cookie有很大的關系。一般來說,只要瀏覽器不關閉,cookie就會一直生效,cookie生效,session的使用就不會受到影響。
4)application作用域
代表變量能一個應用下(多個會話),在服務器下的多個項目之間都能夠使用。比如baidu、wenku等共享帳號。