java基礎學習:JavaWeb之JSP

其他更多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頁面


exception.jsp

error.jsp  isErrorPage屬性說明該頁面是一個錯誤顯示頁面,則可以使用exception對象


error.jsp

訪問:訪問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等共享帳號。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,936評論 6 535
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,744評論 3 421
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,879評論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,181評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,935評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,325評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,384評論 3 443
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,534評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,084評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,892評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,067評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,623評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,322評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,735評論 0 27
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,990評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,800評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,084評論 2 375