JSP
JSP程序
我的第一個JSP程序:
- 在WEB-INF目錄之外創(chuàng)建一個index.jsp文件凿渊,然后這個文件中沒有任何內容。
- 將上面的項目部署之后骚勘,啟動服務器虹茶,打開瀏覽器,訪問以下地址:
- http://localhost:8080/jsp/index.jsp 展現(xiàn)在大家面前的是一個空白踱启。
過程:
- 實際上訪問以上的這個:index.jsp报账,底層執(zhí)行的是:index_jsp.class 這個java程序。
- 這個index.jsp會被tomcat翻譯生成index_jsp.java文件埠偿,然后tomcat服務器又會將index_jsp.java編譯生成index_jsp.class文件
- 訪問index.jsp透罢,實際上執(zhí)行的是index_jsp.class中的方法。
JSP生命周期
JSP實際上就是一個Servlet
- index.jsp訪問的時候冠蒋,會自動翻譯生成index_jsp.java羽圃,會自動編譯生成index_jsp.class,那么index_jsp 這就是一個類浊服。
- index_jsp 類繼承 HttpJspBase统屈,而HttpJspBase類繼承的是HttpServlet。所以index_jsp類就是一個Servlet類牙躺。
- jsp的生命周期和Servlet的生命周期完全相同愁憔。完全就是一個東西。沒有任何區(qū)別孽拷。
- jsp和servlet一樣吨掌,都是單例的。(假單例)
思考
jsp文件第一次訪問的時候是比較慢的脓恕,為什么膜宋?
為什么大部分的運維人員在給客戶演示項目的時候,為什么提前先把所有的jsp文件先訪問一遍炼幔。
第一次比較麻煩:
- 要把jsp文件翻譯生成java源文件
- java源文件要編譯生成class字節(jié)碼文件
- 然后通過class去創(chuàng)建servlet對象
- 然后調用servlet對象的init方法
- 最后調用servlet對象的service方法秋茫。
第二次就比較快了,為什么乃秀?
因為第二次直接調用單例servlet對象的service方法即可肛著。
JSP定義
JSP是什么?
JSP是java程序跺讯。(JSP本質還是一個Servlet)
- JSP是:JavaServer Pages的縮寫枢贿。(基于Java語言實現(xiàn)的服務器端的頁面。)
- Servlet是JavaEE的13個子規(guī)范之一刀脏,那么JSP也是JavaEE的13個子規(guī)范之一局荚。
- JSP是一套規(guī)范。所有的web容器/web服務器都是遵循這套規(guī)范的,都是按照這套規(guī)范進行的“翻譯”
- 每一個web容器/web服務器都會內置一個JSP翻譯引擎耀态。
- 對JSP進行錯誤調試的時候轮傍,還是要直接打開JSP文件對應的java文件,檢查java代碼茫陆。
開發(fā)JSP的最高境界
眼前是JSP代碼金麸,但是腦袋中呈現(xiàn)的是java代碼。
JSP語法
在jsp文件中直接編寫文字會被翻譯到servlet類的service方法的out.write("翻譯到這里")簿盅,直接翻譯到雙引號里挥下,被java程序當做普通字符串打印輸出到瀏覽器。
(在JSP中編寫的HTML CSS JS代碼桨醋,這些代碼對于JSP來說只是一個普通的字符串棚瘟。但是JSP把這個普通的字符串一旦輸出到瀏覽器,瀏覽器就會對HTML CSS JS進行解釋執(zhí)行喜最。展現(xiàn)一個效果.)-
在JSP中編寫java程序
<% java語句; %> //向瀏覽器上輸出一個java變量偎蘸。 <% String name = “jack”; out.write("name = " + name); %>
- 在這個符號當中編寫的被視為java程序,被翻譯到Servlet類的service方法內部瞬内。
- 在service方法當中編寫的代碼是有順序的迷雪,方法體當中的代碼要遵循自上而下的順序依次逐行執(zhí)行。
- service方法當中不能寫靜態(tài)代碼塊虫蝶,不能寫方法章咧,不能定義成員變量
- 在同一個JSP當中 <%%> 這個符號可以出現(xiàn)多個。
- 注意:以上代碼中的out是JSP的九大內置對象之一能真×扪希可以直接拿來用。當然粉铐,必須只能在service方法內部使用疼约。
如果輸出的內容中含有“java代碼”,這個時候可以使用以下語法格式:
<%= %>
<%= %> 這個符號會被翻譯到哪里蝙泼?最終翻譯成什么程剥?
- 翻譯成了這個java代碼:
out.print();
- 翻譯到service方法當中了
-
在JSP中如何編寫JSP的專業(yè)注釋
- <%--JSP的專業(yè)注釋,不會被翻譯到java源代碼當中汤踏。--%>
-
JSP語法總結
- JSP中直接編寫普通字符串
- 翻譯到service方法的out.write("這里")
- <%%>
- 翻譯到service方法體內部倡缠,里面是一條一條的java語句。
- <%! %>
- 翻譯到service方法之外茎活。
- <%= %>
- 翻譯到service方法體內部,翻譯為:out.print();
- <%@page contentType="text/html;charset=UTF-8"%>
- page指令琢唾,通過contentType屬性用來設置響應的內容類型载荔。
- JSP中直接編寫普通字符串
JSP指令
指令的作用:指導JSP的翻譯引擎如何工作(指導當前的JSP翻譯引擎如何翻譯JSP文件。)
指令包括哪些呢采桃?
- include指令:包含指令懒熙,在JSP中完成靜態(tài)包含丘损,很少用了。(這里不講)
- taglib指令:引入標簽庫的指令工扎。這個到JJSTL標簽庫的時候再學習∨窃浚現(xiàn)在先不管。
- page指令:目前重點學習一個page指令肢娘。
指令的使用語法是什么呈础?
<%@指令名 屬性名=屬性值 屬性名=屬性值 屬性名=屬性值....%>
關于page指令當中都有哪些常用的屬性呢?
<%@page session="true|false" %>
true表示啟用JSP的內置對象session橱健,表示一定啟動session對象而钞。沒有session對象會創(chuàng)建。
如果沒有設置拘荡,默認值就是session="true"
session="false" 表示不啟用內置對象session臼节。當前JSP頁面中無法使用內置對象session。
<%@page contentType="text/json" %>
contentType屬性用來設置響應的內容類型
但同時也可以設置字符集珊皿。
<%@page contentType="text/json;charset=UTF-8" %>
<%@page import="java.util.List, java.util.Date, java.util.ArrayList" %>
<%@page import="java.util.*" %>
import語句网缝,導包
<%@page errorPage="/error.jsp" %>
當前頁面出現(xiàn)異常之后,跳轉到error.jsp頁面蟋定。
errorPage屬性用來指定出錯之后的跳轉位置粉臊。
<%@page isErrorPage="true" %>
表示啟用JSP九大內置對象之一:exception
默認值是false
JSP九大內置對象
- jakarta.servlet.jsp.PageContext pageContext 頁面作用域
- jakarta.servlet.http.HttpServletRequest request 請求作用域
- jakarta.servlet.http.HttpSession session 會話作用域
- jakarta.servlet.ServletContext application 應用作用域
- pageContext < request < session < application
- 以上四個作用域都有:setAttribute、getAttribute溢吻、removeAttribute方法维费。
- 以上作用域的使用原則:盡可能使用小的域。
- java.lang.Throwable exception
- jakarta.servlet.ServletConfig config
- java.lang.Object page (其實是this促王,當前的servlet對象)
- jakarta.servlet.jsp.JspWriter out (負責輸出)
- jakarta.servlet.http.HttpServletResponse response (負責響應)
EL表達式
什么是EL表達式
- Expression Language(表達式語言)
- EL表達式可以代替JSP中的java代碼犀盟,讓JSP文件中的程序看起來更加整潔,美觀蝇狼。
- JSP中夾雜著各種java代碼阅畴,例如<% java代碼 %>、<%=%>等迅耘,導致JSP文件很混亂贱枣,不好看,不好維護颤专。所以才有了后期的EL表達式纽哥。
- EL表達式可以算是JSP語法的一部分。EL表達式歸屬于JSP栖秕。
- EL表達式出現(xiàn)在JSP中主要是:
- 從某個作用域中取數(shù)據(jù)春塌,然后將其轉換成字符串,然后將其輸出到瀏覽器。這就是EL表達式的功效只壳。三大功效:
- 第一功效:從某個域中取數(shù)據(jù)俏拱。
- 四個域:
- pageContext
- request
- session
- application
- 第二功效:將取出的數(shù)據(jù)轉成字符串。
- 如果是一個java對象吼句,也會自動調用java對象的toString方法將其轉換成字符串锅必。
- 第三功效:將字符串輸出到瀏覽器。
- 和這個一樣:<%= %>惕艳,將其輸出到瀏覽器搞隐。
- EL表達式很好用,基本的語法格式:
- ${表達式}
EL表達式的使用
<%
// 創(chuàng)建User對象
User user = new User();
user.setUsername("jackson");
user.setPassword("1234");
user.setAge(50);
// 將User對象存儲到某個域當中尔艇。一定要存尔许,因為EL表達式只能從某個范圍中取數(shù)據(jù)。
// 數(shù)據(jù)是必須存儲到四大范圍之一的终娃。
request.setAttribute("userObj", user);
%>
<%--使用EL表達式取--%>
${這個位置寫什么味廊??棠耕?余佛?這里寫的一定是存儲到域對象當中時的name}
要這樣寫:
${userObj}
等同于java代碼:<%=request.getAttribute("userObj")%>
你不要這樣寫:${"userObj"}
面試題:
${abc} 和 ${"abc"}的區(qū)別是什么?
${abc}表示從某個域中取出數(shù)據(jù)窍荧,并且被取的這個數(shù)據(jù)的name是"abc"辉巡,之前一定有這樣的代碼: 域.setAttribute("abc", 對象);
${"abc"} 表示直接將"abc"當做普通字符串輸出到瀏覽器。不會從某個域中取數(shù)據(jù)了蕊退。
${userObj} 底層是怎么做的郊楣?從域中取數(shù)據(jù),取出user對象瓤荔,然后調用user對象的toString方法净蚤,轉換成字符串,輸出到瀏覽器输硝。
<%--如果想輸出對象的屬性值今瀑,怎么辦?--%>
${userObj.username} 使用這個語法的前提是:User對象有getUsername()方法点把。
${userObj.password} 使用這個語法的前提是:User對象有getPassword()方法橘荠。
${userObj.age} 使用這個語法的前提是:User對象有getAge()方法。
${userObj.email} 使用這個語法的前提是:User對象有getEmail()方法郎逃。
EL表達式中的. 這個語法哥童,實際上調用了底層的getXxx()方法。
注意:如果沒有對應的get方法褒翰,則出現(xiàn)異常如蚜。報500錯誤压恒。
${userObj.addr222.zipcode}
以上EL表達式對應的java代碼:
user.getAddr222().getZipcode()
EL表達式優(yōu)先從小范圍中讀取數(shù)據(jù)。
pageContext < request < session < application
EL表達式中有四個隱含的隱式的范圍:
- pageScope 對應的是 pageContext范圍错邦。
- requestScope 對應的是 request范圍。
- sessionScope 對應的是 session范圍型宙。
- applicationScope 對應的是 application范圍撬呢。
EL表達式對null進行了預處理。如果是null妆兑,則向瀏覽器輸出一個空字符串魂拦。
EL表達式取數(shù)據(jù)的時候有兩種形式:
- 第一種:. (大部分使用這種方式)
- 第二種:[ ] (如果存儲到域的時候,這個name中含有特殊字符搁嗓,可以使用 [ ])
- request.setAttribute("abc.def", "zhangsan");
- ${requestScope.abc.def} 這樣是無法取值的芯勘。
- 應該這樣:${requestScope["abc.def"]}
從Map集合中取數(shù)據(jù)
掌握使用EL表達式,怎么從Map集合中取數(shù)據(jù):
${map.key}
從數(shù)組和list集合中取數(shù)據(jù)
掌握使用EL表達式腺逛,怎么從數(shù)組和List集合中取數(shù)據(jù)
- ${數(shù)組[0]
- ${數(shù)組[1]}
- ${list[0]}
其它
page指令當中荷愕,有一個屬性,可以忽略EL表達式
<%@page contentType="text/html;charset=UTF-8" isELIgnored="true" %>
isELIgnored="true" 表示忽略EL表達式
isELIgnored="false" 表示不忽略EL表達式棍矛。(這是默認值)
isELIgnored="true" 這個是全局的控制安疗。
可以使用反斜杠進行局部控制:\${username} 這樣也可以忽略EL表達式。
EL表達式中內置對象
- pageContext
- param
- paramValues
- initParam
- 其他(不是重點)
四個重要隱式對象
pageContext
應用的根路徑:${pageContext.request.contextPath}
param
用戶在瀏覽器上提交數(shù)據(jù):http://localhost:8080/EL/1.jsp?username=jack
用戶名:<%=request.getParameter("username")%>
//使用param
用戶名:${param.username}
paramValues
用戶在瀏覽器上提交數(shù)據(jù):http://localhost:8080/EL/1.jsp?aihao=smoke&aihao=drike&aihao=tangtou
愛好:${paramValues.aihao[0]}够委、${paramValues.aihao[1]}荐类、${paramValues.aihao[2]}
initParam
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="https://jakarta.ee/xml/ns/jakartaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="https://jakarta.ee/xml/ns/jakartaee https://jakarta.ee/xml/ns/jakartaee/web-app_5_0.xsd"
version="5.0">
<!--Servlet上下文初始化參數(shù)-->
<!--上下文初始化參數(shù)被封裝到:ServletContext對象當中了-->
<context-param>
<param-name>pageSize</param-name>
<param-value>20</param-value>
</context-param>
</web-app>
<%
String a =application.getInitParameter("pageSize");
%>
每頁顯示的記錄條數(shù):<%=a%> //20
//使用initParam
每頁顯示的記錄條數(shù)${initParam.pageSize} //20
EL表達式中的運算符
算數(shù)運算符
+只能做求和運算,不會進行字符串的拼接
兩邊不是數(shù)字的時候茁帽,一定會轉成數(shù)字玉罐,轉不成就報錯:NumberFormatException
關系運算符
在EL表達式中 ==
調用了equals
方法
<%
Student stu1 = new Student("110",警察);
Student stu2 = new Student("110",警察);
request.setAttribute("stu1",stu1);
request.setAttribute("stu2",stu2);
%>
${stu1==stu2} //true
<%
Object o1 = new Object();
Object o2 = new Object();
request.setAttribute("o1",o1);
request.setAttribute("o2",o2);
%>
${o1==o2}// fasle
在EL表達式中 eq
調用了equals
方法
<%
Student stu1 = new Student("110",警察);
Student stu2 = new Student("110",警察);
request.setAttribute("stu1",stu1);
request.setAttribute("stu2",stu2);
%>
${stu1 eq stu2} //true
在EL表達式中 !=
調用了equals
方法
<%
Student stu1 = new Student("110",警察);
Student stu2 = new Student("110",警察);
request.setAttribute("stu1",stu1);
request.setAttribute("stu2",stu2);
%>
${stu1 != stu2} //false
條件運算符
? :
${empty param.username ? "用戶名不能為空" : "歡迎訪問!"}
empty運算符
判斷是否為空 如果為空結果是true,如果不為空結果是false
${empty param.username}
JSTL標簽庫
什么是JSTL標簽庫
- Java Standard Tag Lib (Java標準的標簽庫)
- JSTL標簽庫通常結合EL表達式一起使用。目的是讓JSP中的java代碼消失潘拨。
- 標簽是寫在JSP當中的吊输,但實際上最終還是要執(zhí)行對應的java程序。(java程序在jar包當中战秋。)
使用JSTL的步驟
-
引入JSTL對應的jar包
- 在WEB-INF下面新建lib目錄 然后把對應的jar包 復制過去
-
在JSP中引入要使用的標簽庫璧亚。
- 使用taglib指令引入標簽庫
在JSTL中有很多標簽 ,我們只需要重點掌握核心標簽庫
<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
prefix="隨便起名字"
- 在需要使用標簽的位置使用即可
表面使用的是標簽脂信,底層實際上還是java
程序
JSTL標簽原理
<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
以上uri后面的路徑實際上指向了一個×x×.tld文件癣蟋。
tld文件實際上是一個xml配置文件。
在tld文件中描述了"標簽"和"java類"之間的關系狰闪。
以上核心標簽庫對應的tld文件是:c.tld文件疯搅。它在哪里。
在jakarta.serv1et.jsp.jst1-2.0.0.jar里面META-INF目錄下埋泵,有一個c.t1d文件幔欧。
配置文件:tld解析
<tag>
<description>對標簽的描述</description>
<name>標簽的名字</name>
<tag-class>標簽對應java類</tag-class>
<body-content>JSP</body-content>標簽體當中可以出現(xiàn)的內容罪治,如果是JSp,
就表示標簽體中可以出現(xiàn)符合JSp所有語法的代碼礁蔗。例如EL表達式觉义。
<attribute>
<description>
對這個屬性的描述
</description>
<name>var</name>屬性名
<required>false</required>false表示該屬性不是必須的
<rtexprvalue>false</rtexprvalue>false 不支持EL表達式
</attribute>
</tag>
常用的標簽
JSTL中的核心標簽庫core當中有哪些常用的標簽呢?
<c:if text="boolean類型,支持EL表達式"></c:if>
<c:forEach items="集合,支持EL表達式" var = "集合中的元素" varStatus="元素的狀態(tài)對象">${元素狀態(tài)對象:count}</c:forEach>
<c:forEach var="i" begin="1" end="10" step="2"> ${i}</c:forEach>
<c:choose>
<c:when test="${param.age < 18}">
青少年
</c:when>
<c:when test="${param.age < 35}">
青年
</c:when>
<c:when test="${param.age < 55}">
中年
</c:when>
<c:otherwise>
老年
</c:otherwise>
</c:choose>