在動(dòng)態(tài)web項(xiàng)目的開發(fā)中,經(jīng)常需要?jiǎng)討B(tài)生成html內(nèi)容(如系統(tǒng)中的當(dāng)前在線人數(shù)需要?jiǎng)討B(tài)生成)。如果使用servlet實(shí)現(xiàn)html頁(yè)面數(shù)據(jù)的統(tǒng)計(jì),則需要使用大量的輸出語句。同時(shí),如果靜態(tài)內(nèi)容和動(dòng)態(tài)內(nèi)容混合在一起,那么也將導(dǎo)致程序非常臃腫。為了客服servlet的這些缺點(diǎn),oracle(sun)公司推出了jsp技術(shù)。
1.jsp概述
jsp(java server pages)是建立在servlet規(guī)范之上的動(dòng)態(tài)網(wǎng)頁(yè)開發(fā)技術(shù),其實(shí)質(zhì)是一個(gè)簡(jiǎn)化的servlet。在jsp文件中,html和java代碼共同存在,其中,html代碼用于實(shí)現(xiàn)網(wǎng)頁(yè)中靜態(tài)內(nèi)容的顯示,java代碼用于實(shí)現(xiàn)網(wǎng)頁(yè)中動(dòng)態(tài)內(nèi)容的實(shí)現(xiàn)。為了和傳統(tǒng)的html有所區(qū)別,jsp文件擴(kuò)展名為jap。
jsp技術(shù)所開發(fā)的web應(yīng)用程序是基于java的,其具有以下特征:
?。?)預(yù)編譯
預(yù)編譯指在用戶第一次通過瀏覽器訪問jsp頁(yè)面時(shí),服務(wù)器將對(duì)jsp頁(yè)面代碼進(jìn)行編譯,并且僅指向一次編譯。編譯好的代碼將被保存,在用戶下一次訪問時(shí)會(huì)直接執(zhí)行編譯好的代碼。這樣不僅節(jié)約了服務(wù)器的cpu資源,還大幅度提升了客戶端的訪問速度。
?。?)業(yè)務(wù)代碼相分離
在使用jsp技術(shù)開發(fā)web應(yīng)用時(shí),可以將界面的開發(fā)和應(yīng)用程序的開發(fā)分離。
?。?)組件重用
jsp可以使用javabean編寫業(yè)務(wù)組件,也就是使用一個(gè)javabean類封裝業(yè)務(wù)處理代碼或者將其作為一個(gè)數(shù)據(jù)存儲(chǔ)模型,在jsp頁(yè)面甚至整個(gè)項(xiàng)目中,都可以重復(fù)使用這個(gè)javabean,同時(shí),javabean也可以應(yīng)用帶其他java應(yīng)用程序中。
?。?)跨平臺(tái)
由于jsp是基于java語言的,它可以使用java api,所有它也是跨平臺(tái)的,可以應(yīng)用與不同的系統(tǒng),如windows和linux。
jsp 運(yùn)行原理
jsp的工作模式是請(qǐng)求/響應(yīng)模式,客戶端首先發(fā)出http請(qǐng)求,jsp程序收到請(qǐng)求后將進(jìn)行處理并返回處理結(jié)果。在一個(gè)jsp文件第一次請(qǐng)求時(shí),jsp引擎(容器)把該jsp文件轉(zhuǎn)化成一個(gè)servlet,而這個(gè)引擎本身也是一個(gè)servlet。
jsp運(yùn)行過程:
(1)客戶端發(fā)出請(qǐng)求,請(qǐng)求訪問jsp文件。
?。?)jsp容器先將jsp文件轉(zhuǎn)化成一個(gè)java源文件(java servlet源程序),在轉(zhuǎn)換過程中,如果發(fā)現(xiàn)jsp文件存在任何語法錯(cuò)誤,則中斷轉(zhuǎn)換過程,并向服務(wù)器和客戶端返回出錯(cuò)信息。
?。?)如果轉(zhuǎn)換成功,則jsp容器會(huì)將生成的java源文件編譯成相應(yīng)的字節(jié)碼文件*.class。該class文件就是一個(gè)servlet,servlet容器會(huì)像處理其他servlet一樣處理它。
?。?)有servlet容器加載轉(zhuǎn)換后的servlet類(class文件)創(chuàng)建該servlet(jsp頁(yè)面的轉(zhuǎn)換結(jié)果)的實(shí)例,并執(zhí)行servlet的jspinit()方法。jspinit()方法在servlet的整個(gè)生命周期只會(huì)執(zhí)行一次。
?。?)執(zhí)行jspservice()方法處理客戶端的請(qǐng)求。對(duì)于每一個(gè)請(qǐng)求,jsp容器都會(huì)創(chuàng)建一個(gè)新的線程處理它。如果多個(gè)客戶端同時(shí)請(qǐng)求該jsp文件,則jsp容器會(huì)創(chuàng)建多個(gè)線程,使每一個(gè)客戶端請(qǐng)求都對(duì)應(yīng)一個(gè)線程。
?。?)如果jsp文件被修改了,則服務(wù)器將根據(jù)設(shè)置決定是否對(duì)該文件重新進(jìn)行編譯,如果需要重新編譯,則使用重新編譯后的結(jié)果取代內(nèi)存中的servlet,并繼續(xù)上述處理過程。需要注意的是,雖然jsp效率很高,但在第一次調(diào)用時(shí)往往需要轉(zhuǎn)換和編譯,所以會(huì)產(chǎn)生一些輕微的延遲。
?。?)如果系統(tǒng)出現(xiàn)資源不足等問題,jsp容器可能會(huì)以某種不確定的方式將servlet從內(nèi)存中移除,發(fā)生這種情況時(shí),首先會(huì)調(diào)用jspdestroy()方法,然后servlet實(shí)例會(huì)被作為垃圾進(jìn)行處理。
(8)當(dāng)請(qǐng)求處理完成后,響應(yīng)對(duì)象由jsp容器接收,并將html格式的響應(yīng)信息送回客戶端。
因此:瀏覽器向服務(wù)器發(fā)送請(qǐng)求,不管訪問的是什么資源啊,其實(shí)都是在訪問servlet,所有當(dāng)訪問一個(gè)jsp頁(yè)面時(shí),其實(shí)也是在訪問一個(gè)servlet,服務(wù)器在執(zhí)行jsp的時(shí)候,首先把jsp翻譯成一個(gè)servlet,所有訪問jsp時(shí),其實(shí)不是在訪問jsp,而是在訪問jsp翻譯過后的那個(gè)servlet。例如:
c1.jsp
<%@ page language="java" contenttype="text/html; charset=utf-8" pageencoding="utf-8"%> <!doctype html public "-//w3c//dtd html 4.01 transitional//en" "http://www.w3.org/tr/html4/loose.dtd"> <html> <head> <meta http-equiv="content-type" content="text/html; charset=iso-8859-1"> <title>insert title here</title> </head> <body> this is my first jsp <% out.print("你好啊c1"); %> </body> </html>
當(dāng)我們通過瀏覽器(http://localhost:8080/day11_01_jsp(項(xiàng)目名稱)/c1.jsp
)訪問c1.jsp時(shí),服務(wù)器首先將c1.jsp翻譯成一個(gè)c1_jsp.class,在tomcat服務(wù)器的work\catalina\localhost\項(xiàng)目名\org\apache\jsp目錄下可以看到c1_jsp.class的源代碼。(1.jsp翻譯成_1_jsp.class)
c1_jap.java的代碼:
/* * generated by the jasper component of apache tomcat * version: apache tomcat/7.0.52 * generated at: 2018-10-05 08:32:50 utc * note: the last modified time of this file was set to * the last modified time of the source file after * generation to assist with modification tracking. */ package org.apache.jsp; import javax.servlet.*; import javax.servlet.http.*; import javax.servlet.jsp.*; public final class c1_jsp extends org.apache.jasper.runtime.httpjspbase implements org.apache.jasper.runtime.jspsourcedependent { private static final javax.servlet.jsp.jspfactory _jspxfactory = javax.servlet.jsp.jspfactory.getdefaultfactory(); private static java.util.map<java.lang.string,java.lang.long> _jspx_dependants; private javax.el.expressionfactory _el_expressionfactory; private org.apache.tomcat.instancemanager _jsp_instancemanager; public java.util.map<java.lang.string,java.lang.long> getdependants() { return _jspx_dependants; } public void _jspinit() { _el_expressionfactory = _jspxfactory.getjspapplicationcontext(getservletconfig().getservletcontext()).getexpressionfactory(); _jsp_instancemanager = org.apache.jasper.runtime.instancemanagerfactory.getinstancemanager(getservletconfig()); } public void _jspdestroy() { } public void _jspservice(final javax.servlet.http.httpservletrequest request, final javax.servlet.http.httpservletresponse response) throws java.io.ioexception, javax.servlet.servletexception { final javax.servlet.jsp.pagecontext pagecontext; javax.servlet.http.httpsession session = null; final javax.servlet.servletcontext application; final javax.servlet.servletconfig config; javax.servlet.jsp.jspwriter out = null; final java.lang.object page = this; javax.servlet.jsp.jspwriter _jspx_out = null; javax.servlet.jsp.pagecontext _jspx_page_context = null; try { response.setcontenttype("text/html; charset=utf-8"); pagecontext = _jspxfactory.getpagecontext(this, request, response, null, true, 8192, true); _jspx_page_context = pagecontext; application = pagecontext.getservletcontext(); config = pagecontext.getservletconfig(); session = pagecontext.getsession(); out = pagecontext.getout(); _jspx_out = out; out.write("\r\n"); out.write("<!doctype html public \"-//w3c//dtd html 4.01 transitional//en\" \"http://www.w3.org/tr/html4/loose.dtd\">\r\n"); out.write("<html>\r\n"); out.write("<head>\r\n"); out.write("<meta http-equiv=\"content-type\" content=\"text/html; charset=iso-8859-1\">\r\n"); out.write("<title>insert title here</title>\r\n"); out.write("</head>\r\n"); out.write("<body>\r\n"); out.write("\tthis is my first jsp \r\n"); out.write("\t"); out.print("你好啊c1"); out.write("\r\n"); out.write("</body>\r\n"); out.write("</html>"); } catch (java.lang.throwable t) { if (!(t instanceof javax.servlet.jsp.skippageexception)){ out = _jspx_out; if (out != null && out.getbuffersize() != 0) try { out.clearbuffer(); } catch (java.io.ioexception e) {} if (_jspx_page_context != null) _jspx_page_context.handlepageexception(t); else throw new servletexception(t); } } finally { _jspxfactory.releasepagecontext(_jspx_page_context); } } }
我們可以看到,c1_jsp這個(gè)類是繼承 org.apache.jasper.runtime.httpjspbase這個(gè)類的,通過查看tomcat服務(wù)器的源代碼,可以知道在apache-tomcat-6.0.20-src\java\org\apache\jasper\runtime目錄下存httpjspbase這個(gè)類的源代碼文件,如下圖所示:
httpjsbase這個(gè)類的源代碼:
/* * licensed to the apache software foundation (asf) under one or more * contributor license agreements. see the notice file distributed with * this work for additional information regarding copyright ownership. * the asf licenses this file to you under the apache license, version 2.0 * (the "license"); you may not use this file except in compliance with * the license. you may obtain a copy of the license at * * http://www.apache.org/licenses/license-2.0 * * unless required by applicable law or agreed to in writing, software * distributed under the license is distributed on an "as is" basis, * without warranties or conditions of any kind, either express or implied. * see the license for the specific language governing permissions and * limitations under the license. */ package org.apache.jasper.runtime; import java.io.ioexception; import javax.servlet.servletconfig; import javax.servlet.servletexception; import javax.servlet.http.httpservlet; import javax.servlet.http.httpservletrequest; import javax.servlet.http.httpservletresponse; import javax.servlet.jsp.httpjsppage; import org.apache.jasper.compiler.localizer; /** * this is the super class of all jsp-generated servlets. * * @author anil k. vijendran */ public abstract class httpjspbase extends httpservlet implements httpjsppage { private static final long serialversionuid = 1l; protected httpjspbase() { } @override public final void init(servletconfig config) throws servletexception { super.init(config); jspinit(); _jspinit(); } @override public string getservletinfo() { return localizer.getmessage("jsp.engine.info"); } @override public final void destroy() { jspdestroy(); _jspdestroy(); } /** * entry point into service. */ @override public final void service(httpservletrequest request, httpservletresponse response) throws servletexception, ioexception { _jspservice(request, response); } @override public void jspinit() { } public void _jspinit() { } @override public void jspdestroy() { } protected void _jspdestroy() { } @override public abstract void _jspservice(httpservletrequest request, httpservletresponse response) throws servletexception, ioexception; }
httpjspbase類是繼承httpservlet的,所以httpjspbase類是一個(gè)servlet,而c1_jsp又是繼承httpjspbase類的,所以c1_jsp類也是一個(gè)servlet,所以當(dāng)瀏覽器訪問服務(wù)器上的c1.jsp頁(yè)面時(shí),其實(shí)就是在訪問c1_jsp這個(gè)servlet,c1_jsp這個(gè)servlet使用_jspservice這個(gè)方法處理請(qǐng)求。
2.jsp的基本語法
2.1 jsp模板元素
網(wǎng)頁(yè)的靜態(tài)內(nèi)容。如:html標(biāo)簽和文本。
2.2 jsp腳本元素
(1)jsp scriptlets(腳本片斷)用于在jsp頁(yè)面中編寫多行java代碼。語法:
<% java代碼(變量、方法、表達(dá)式等 ) %>
<% int sum=0;//聲明變量 /*編寫語句*/ for (int i=1;i<=100;i++){ sum+=i; } out.println("<h1>sum="+sum+"</h1>"); %>
jsp腳本片斷中只能出現(xiàn)java代碼,不能出現(xiàn)其它模板元素, jsp引擎在翻譯jsp頁(yè)面中,會(huì)將jsp腳本片斷中的java代碼將被原封不動(dòng)地放到servlet的_jspservice方法中。jsp腳本片斷中的java代碼必須嚴(yán)格遵循java語法,例如,每執(zhí)行語句后面必須用分號(hào)(;)結(jié)束。在一個(gè)jsp頁(yè)面中可以有多個(gè)腳本片斷,在兩個(gè)或多個(gè)腳本片斷之間可以嵌入文本、html標(biāo)記和其他jsp元素。多個(gè)腳本片斷中的代碼可以相互訪問,猶如將所有的代碼放在一對(duì)<%%>之中的情況。如:out.println(x);單個(gè)腳本片斷中的java語句可以是不完整的,但是,多個(gè)腳本片斷組合后的結(jié)果必須是完整的java語句。
<% for (int i=1; i<5; i++) { %> <h1>http://localhost:8080/javaweb_jsp_study_20140603/</h1> <% } %>
(2)jsp聲明
jsp頁(yè)面中編寫的所有代碼,默認(rèn)會(huì)翻譯到servlet的service方法中,而jsp聲明中的java代碼會(huì)被翻譯到_jspservice方法外面。
<%! java代碼:定義變量或者方法 %>
多個(gè)靜態(tài)代碼塊、變量和方法可以定義在一個(gè)jsp文件中,也可以分別單獨(dú)定義在多個(gè)jsp聲明中。
jsp隱式對(duì)象的作用范圍僅限于servlet的_japservice方法。所以在jsp聲明中不能使用這些隱式對(duì)象。
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)jsp 表達(dá)式
jsp腳本表達(dá)式(expression)用于將程序數(shù)據(jù)輸出到客戶端,他將要輸出的變量或者表達(dá)式直接封裝在以<%= %>標(biāo)記中,語法為:
<%=expression%>
舉例:輸出當(dāng)前系統(tǒng)時(shí)間:
<%= new java.util.date() %>
jsp引擎在翻譯腳本表達(dá)式時(shí),會(huì)將程序數(shù)據(jù)轉(zhuǎn)成字符串,然后在相應(yīng)位置用out.print(...)將數(shù)據(jù)輸給客戶端。
jsp腳本表達(dá)式的變量或者表達(dá)式后不能有分號(hào)(;)。
3.jsp注釋
?。?)顯式注釋:直接使用html風(fēng)格的注釋:<!- - 注釋內(nèi)容- -> 特點(diǎn):不安全,費(fèi)流量;html的注釋在瀏覽器中查看源文件的時(shí)候是可以看得到的
?。?)隱式注釋:直接使用java的注釋://、/*……*/
jsp自己的注釋:<%- - 注釋內(nèi)容- -%> 特點(diǎn):安全,省流量
java注釋和jsp注釋在瀏覽器中查看源文件時(shí)是看不到注釋的內(nèi)容的
<!--這個(gè)注釋可以看見--> <% //java中的單行注釋 /* java中的多行注釋 */ %> <% --jsp自己的注釋--%>
參考:
到此這篇關(guān)于jsp動(dòng)態(tài)網(wǎng)頁(yè)開發(fā)技術(shù)概述的文章就介紹到這了,更多相關(guān)jsp動(dòng)態(tài)網(wǎng)頁(yè)內(nèi)容請(qǐng)搜索碩編程以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持碩編程!
- jsp+servlet實(shí)現(xiàn)文件上傳與下載功能
- EJB3.0部署消息驅(qū)動(dòng)Bean拋javax.naming.NameNotFoundException異常
- 在JSP中使用formatNumber控制要顯示的小數(shù)位數(shù)方法
- 秒殺系統(tǒng)Web層設(shè)計(jì)的實(shí)現(xiàn)方法
- 將properties文件的配置設(shè)置為整個(gè)Web應(yīng)用的全局變量實(shí)現(xiàn)方法
- JSP使用過濾器防止Xss漏洞
- 在JSP頁(yè)面中動(dòng)態(tài)生成圖片驗(yàn)證碼的方法實(shí)例
- 詳解JSP 內(nèi)置對(duì)象request常見用法
- 使用IDEA編寫jsp時(shí)EL表達(dá)式不起作用的問題及解決方法
- jsp實(shí)現(xiàn)局部刷新頁(yè)面、異步加載頁(yè)面的方法
- Jsp中request的3個(gè)基礎(chǔ)實(shí)踐
- JavaServlet的文件上傳和下載實(shí)現(xiàn)方法
- JSP頁(yè)面的靜態(tài)包含和動(dòng)態(tài)包含使用方法