Web框架是開(kāi)發(fā)者在使用某種語(yǔ)言編寫(xiě)Web應(yīng)用服務(wù)端項(xiàng)目時(shí)關(guān)于架構(gòu)的最佳實(shí)踐。很多Web框架是從實(shí)際的Web項(xiàng)目抽取出來(lái)的,僅和Web的請(qǐng)求和響應(yīng)處理有關(guān),形成一個(gè)基礎(chǔ),在開(kāi)發(fā)別的應(yīng)用項(xiàng)目的時(shí)候則可以從這個(gè)剝離出來(lái)的基礎(chǔ)做起,讓開(kāi)發(fā)者更關(guān)注更具體的業(yè)務(wù)問(wèn)題,而不是Web的請(qǐng)求和響應(yīng)的控制。
框架很多,但套路基本類(lèi)似,幫你隱藏很多關(guān)于 HTTP 協(xié)議細(xì)節(jié)內(nèi)容,專注功能開(kāi)發(fā)。
但對(duì)一個(gè)初學(xué)者來(lái)說(shuō),過(guò)早的接觸框架往往是事倍功半!同樣一個(gè)問(wèn)題,換一種框架你可能需要從頭開(kāi)始研究。
下面是針對(duì)初學(xué) Java 開(kāi)發(fā) Web 過(guò)程一些個(gè)人見(jiàn)解和思路,高手可略過(guò)。
1. 基本要求:Java 編程基礎(chǔ)
有良好的 Java 語(yǔ)言編程基礎(chǔ),這是必須的,在討論 Web 開(kāi)發(fā)技術(shù)時(shí)提了一個(gè) Java 編程基礎(chǔ)的問(wèn)題會(huì)被鄙視的。
2. 環(huán)境準(zhǔn)備 (Eclipse + Tomcat)
選擇一個(gè)你喜愛(ài)的Servlet容器,或者說(shuō)大一點(diǎn)就是應(yīng)用服務(wù)器,推薦 Tomcat 、Resin 或者 Jetty 這些輕量級(jí)的產(chǎn)品。這三個(gè)產(chǎn)品下載 zip 包解壓后就可以用了。如果你不熟悉 Tomcat 的話請(qǐng)不要使用 exe 版本的 Tomcat,那會(huì)徒增很多煩惱。
把應(yīng)用服務(wù)器啟動(dòng)起來(lái)并能訪問(wèn)到其默認(rèn)的頁(yè)面為準(zhǔn)。
關(guān)于開(kāi)發(fā)工具:不推薦使用 MyEclipse 和 Eclipse 的 JEE 版本,徒增煩惱、運(yùn)行緩慢而且還讓你無(wú)法了解 Web 項(xiàng)目的結(jié)構(gòu)。普通的 Eclipse 或者你喜歡的開(kāi)發(fā)工具就足夠了,能支持普通 Java 項(xiàng)目開(kāi)發(fā)即可。
為了方便,我做了一個(gè)最基本的Java 項(xiàng)目 ——ServletDemo.zip(評(píng)論區(qū)見(jiàn)鏈接),你可將它導(dǎo)入到 Eclipse 里就是一個(gè)完整的、最簡(jiǎn)單的 Web 項(xiàng)目。
然后將下面 XML 內(nèi)容替換 Tomcat 下的 conf/server.xml 文件:
其中 D:\WORKDIR\ServletDemo 替換為你導(dǎo)入的項(xiàng)目路徑,再次啟動(dòng) Tomcat 后在瀏覽器打開(kāi) http://localhost:8080/hello 便可看到 Hello World 的輸出信息。
3. 了解 Servlet 和 Filter
好了,我已經(jīng)把環(huán)境搭起來(lái)了,接下來(lái)該干嘛呢?
前面的步驟為的是搭建一個(gè)測(cè)試的環(huán)境,然后讓你了解一個(gè)最基本的 Java Web 項(xiàng)目的結(jié)構(gòu)。
一個(gè)最基本的 Java Web 項(xiàng)目所需的 jar 包只需要一個(gè) servlet-api.jar ,這個(gè) jar 包中的類(lèi)大部分都是接口,還有一些工具類(lèi),共有 2 個(gè)包,分別是 javax.servlet 和 javax.servlet.http。我把這個(gè)jar包放到了 webapp 目錄外的一個(gè)獨(dú)立 packages 文件夾里,這是因?yàn)樗械?Servlet 容器都帶有這個(gè)包,你無(wú)需再放到Web項(xiàng)目里,我們放到這里只不過(guò)是編譯的需要,運(yùn)行是不需要的。如果你硬是把 servlet-api.jar 放到 webapp/WEB-INF/lib 目錄下,那么 Tomcat 啟動(dòng)時(shí)還會(huì)報(bào)一個(gè)警告信息。
Java Web 項(xiàng)目還需要一個(gè)非常重要的配置文件 web.xml ,在這個(gè)項(xiàng)目中已經(jīng)被我最小化了,只保留有用的信息:
每個(gè) servlet 都必須在 web.xml 中定義并進(jìn)行 URL 映射配置,早期 Java 開(kāi)發(fā) Web 在沒(méi)有框架滿天飛的時(shí)候,這個(gè)文件會(huì)定義了大量的 servlet,或者有人為了省事干脆來(lái)一個(gè) /servlet/* 來(lái)通過(guò)類(lèi)名直接調(diào)用 Servlet。
Servlet 規(guī)范里還有另外一個(gè)非常重要而且非常有用的接口那就是 Filter 過(guò)濾器。
下面是一個(gè)最簡(jiǎn)單的 Filter 類(lèi)以及相應(yīng)的定義方法:
package demo;import java.io.IOException;import javax.servlet.Filter;import javax.servlet.FilterChain;import javax.servlet.FilterConfig;import javax.servlet.ServletException;import javax.servlet.ServletRequest;import javax.servlet.ServletResponse;import javax.servlet.http.HttpServletRequest;public class HelloFilter implements Filter {@Overridepublic void init(FilterConfig arg0) throws ServletException {System.out.println("Filter 初始化");}@Overridepublic void doFilter(ServletRequest req, ServletResponse res,FilterChain chain) throws IOException, ServletException {HttpServletRequest request = (HttpServletRequest)req;System.out.println("攔截 URI="+request.getRequestURI());chain.doFilter(req, res);}@Overridepublic void destroy() {System.out.println("Filter 結(jié)束");}}
在 web.xml 中的配置必須放在 Servlet 的前面:
訪問(wèn) http://localhost:8080/hello 時(shí)看看 Tomcat 控制臺(tái)有何輸出信息。
4. Servlet 和 HTTP 的對(duì)應(yīng)關(guān)系
Servlet 是 J2EE 最重要的一部分,有了 Servlet 你就是 J2EE 了,J2EE 的其他方面的內(nèi)容擇需采用。而 Servlet 規(guī)范你需要掌握的就是 servlet 和 filter 這兩項(xiàng)技術(shù)。絕大多數(shù)框架不是基于 servlet 就是基于 filter,如果它要在 Servlet 容器上運(yùn)行,就永遠(yuǎn)也脫離不開(kāi)這個(gè)模型。
為什么 Servlet 規(guī)范會(huì)有兩個(gè)包,javax.servlet 和 javax.servlet.http ,早先設(shè)計(jì)該規(guī)范的人認(rèn)為 Servlet 是一種服務(wù)模型,不一定是依賴某種網(wǎng)絡(luò)協(xié)議之上,因此就抽象出了一個(gè) javax.servlet ,同時(shí)在提供一個(gè)基于 HTTP 協(xié)議上的接口擴(kuò)展。但是從實(shí)際運(yùn)行這么多年來(lái)看,似乎沒(méi)有發(fā)現(xiàn)有在其他協(xié)議上實(shí)現(xiàn)的 Servlet 技術(shù)。
javax.servlet 和 javax.servlet.http 這兩個(gè)包總共加起來(lái)也不過(guò)是三十四個(gè)接口和類(lèi)。你需要通過(guò) J2EE 的 JavaDoc 文檔 熟知每個(gè)類(lèi)和接口的具體意思。特別是下面幾個(gè)接口必須熟知每個(gè)方法的意思和用途:
HttpServlet
ServetConfig
ServletContext
Filter
FilterConfig
FilterChain
RequestDispatcher
HttpServletRequest
HttpServletResponse
HttpSession
一些 Listenser 類(lèi)
再次強(qiáng)調(diào) HttpServletRequest 和 HttpServletResponse 這兩個(gè)接口更應(yīng)該是爛熟于心。
如果你從字面上無(wú)法理解某個(gè)方法的意思,你可以在前面那個(gè)項(xiàng)目的基礎(chǔ)上做實(shí)驗(yàn)看看其輸出,再不行你可以到討論區(qū)提問(wèn),這樣的提問(wèn)非常明確,很多人都可以幫到你。
為什么我這么強(qiáng)調(diào) HttpServletRequest 和 HttpServletResponse 這兩個(gè)接口,因?yàn)?Web 開(kāi)發(fā)是離不開(kāi) HTTP 協(xié)議的,而 Servlet 規(guī)范其實(shí)就是對(duì) HTTP 協(xié)議做面向?qū)ο蟮姆庋b,HTTP協(xié)議中的請(qǐng)求和響應(yīng)就是對(duì)應(yīng)了 HttpServletRequest 和 HttpServletResponse 這兩個(gè)接口。
你可以通過(guò) HttpServletRequest 來(lái)獲取所有請(qǐng)求相關(guān)的信息,包括 URI、Cookie、Header、請(qǐng)求參數(shù)等等,別無(wú)它路。因此當(dāng)你使用某個(gè)框架時(shí),你想獲取HTTP請(qǐng)求的相關(guān)信息,只要拿到 HttpServletRequest 實(shí)例即可。
而 HttpServletResponse接口是用來(lái)生產(chǎn) HTTP 回應(yīng),包含 Cookie、Header 以及回應(yīng)的內(nèi)容等等。
5. 再談?wù)?Session
HTTP 協(xié)議里是沒(méi)有關(guān)于 Session 會(huì)話的定義,Session 是各種編程語(yǔ)言根據(jù) HTTP 協(xié)議的無(wú)狀態(tài)這種特點(diǎn)而產(chǎn)生的。其實(shí)現(xiàn)無(wú)非就是服務(wù)器端的一個(gè)哈希表,哈希表的Key就是傳遞給瀏覽器的名為 jsessionid 的 Cookie 值。
當(dāng)需要將某個(gè)值保存到 session 時(shí),容器會(huì)執(zhí)行如下幾步:
a. 獲取 jsessionid 值,沒(méi)有的話就生成一個(gè),也就是 request.getSession() 這個(gè)方法b. 拿到的 HttpSession 對(duì)象實(shí)例就相當(dāng)于一個(gè)哈希表,你可以往哈希表里存放數(shù)據(jù)(setAttribute)c. 你也可以通過(guò) getAttribute 來(lái)獲取某個(gè)值
而這個(gè)名為 jsessionid 的 Cookie 在瀏覽器關(guān)閉時(shí)會(huì)自動(dòng)刪除。把 Cookie 的 MaxAge 值設(shè)為 -1 就能達(dá)到瀏覽器關(guān)閉自動(dòng)刪除的效果。
6. 關(guān)于 JSP
首先我已經(jīng)不用 JSP 很多年了,現(xiàn)在一直是使用 Velocity 模板引擎。
任何一個(gè) JSP 頁(yè)面在執(zhí)行的時(shí)候都會(huì)編譯成一個(gè) Servlet 類(lèi)文件,如果是 Tomcat 的話,這些生成的 java 文件會(huì)放置在 {TOMCAT}/work 目錄下對(duì)應(yīng)項(xiàng)目的子目錄中,例如 Tomcat 生成的類(lèi)文件如下:
package org.apache.jsp;import javax.servlet.*;import javax.servlet.http.*;import javax.servlet.jsp.*;import java.util.*;public final class test_jsp extends org.apache.jasper.runtime.HttpJspBase implements org.apache.jasper.runtime.JspSourceDependent { private static final JspFactory _jspxFactory = JspFactory.getDefaultFactory(); private static java.util.ListTest Demo (oschina)
\r\n"); out.write("\r\n");Enumeration Names=request.getHeaderNames();while(Names.hasMoreElements()){String name=(String)Names.nextElement();String value=request.getHeader(name); out.write("\r\n"); out.write("
\r\n"); out.write(" \r\n"); out.write(""); } catch (Throwable t) { if (!(t instanceof 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); } } finally { _jspxFactory.releasePageContext(_jspx_page_context); } }}\r\n"); out.write(" \r\n"); out.write(" "); } out.write("\r\n"); out.write(""); out.print(name); out.write(" \r\n"); out.write(" "); out.print(value); out.write(" \r\n"); out.write(" \r\n"); out.write("
在 servlet 中有一個(gè)包 javax.servlet.jsp 是跟 JSP 相關(guān)的一些接口規(guī)范定義。JSP 比 Servlet 方便的地方在于可直接修改立即生效,不像 Servlet 修改后必須重啟容器才能生效。
因此 JSP 適合用來(lái)做視圖,而 Servlet 則適合做控制層。
7. 總結(jié)
羅哩羅嗦一大堆,歸納一下就是下面幾點(diǎn):
熟知 Servlet 規(guī)范之前,請(qǐng)不要學(xué)習(xí)任何框架
使用最簡(jiǎn)單的工具,不要任何向?qū)Ш涂梢暬?/p>
熟知 HTTP 協(xié)議
等你真的掌握了 Servlet 規(guī)范再去看框架,便會(huì)覺(jué)得一些都小菜。總之一點(diǎn):不要被框架牽著鼻子走,框架是你的工具,它應(yīng)該聽(tīng)你的!
-
Web
+關(guān)注
關(guān)注
2文章
1252瀏覽量
69052 -
JAVA
+關(guān)注
關(guān)注
19文章
2943瀏覽量
104084 -
XML
+關(guān)注
關(guān)注
0文章
186瀏覽量
32993
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論