Java的Web框架雖然各不相同,但基本也都是遵循特定的路數(shù)的:使用Servlet或者Filter攔截請(qǐng)求,使用MVC的思想設(shè)計(jì)架構(gòu),使用約定,XML或 Annotation實(shí)現(xiàn)配置,運(yùn)用Java面向?qū)ο蟮奶攸c(diǎn),面向抽象實(shí)現(xiàn)請(qǐng)求和響應(yīng)的流程,支持Jsp,F(xiàn)reemarker,Velocity等視圖。
成都網(wǎng)站建設(shè)、網(wǎng)站制作的開(kāi)發(fā),更需要了解用戶(hù),從用戶(hù)角度來(lái)建設(shè)網(wǎng)站,獲得較好的用戶(hù)體驗(yàn)。創(chuàng)新互聯(lián)多年互聯(lián)網(wǎng)經(jīng)驗(yàn),見(jiàn)的多,溝通容易、能幫助客戶(hù)提出的運(yùn)營(yíng)建議。作為成都一家網(wǎng)絡(luò)公司,打造的就是網(wǎng)站建設(shè)產(chǎn)品直銷(xiāo)的概念。選擇創(chuàng)新互聯(lián),不只是建站,我們把建站作為產(chǎn)品,不斷的更新、完善,讓每位來(lái)訪(fǎng)用戶(hù)感受到浩方產(chǎn)品的價(jià)值服務(wù)。
JSP優(yōu)點(diǎn):
Java EE標(biāo)準(zhǔn),這意味著有很大的市場(chǎng)需求和更多的工作機(jī)會(huì)
上手快速并且相對(duì)容易
有大量可用的組件庫(kù)
JSP缺點(diǎn):
大量的JSP標(biāo)簽
對(duì)REST和安全支持不好
沒(méi)有一個(gè)統(tǒng)一的實(shí)現(xiàn)。既有SUN的實(shí)現(xiàn),又有Apache的實(shí)現(xiàn)--MyFaces。
Spring MVC優(yōu)點(diǎn):
對(duì)Spring MVC覆蓋綁定(overriding binding)、驗(yàn)證(validation)等提供生命周期管理
與許多表示層技術(shù)/框架無(wú)縫集成:JSP/JSTL、Tiles、Velocity、FreeMarker、Excel、XSL、PDF 等
便于測(cè)試--歸功于IoC
Spring MVC缺點(diǎn):
大量的XML配置文件
太過(guò)靈活--沒(méi)有公共的父控制器
沒(méi)有內(nèi)置的Ajax支持
Stripes優(yōu)點(diǎn):
不需要書(shū)寫(xiě)XML配置文件
良好的學(xué)習(xí)文檔
社區(qū)成員很熱心
Stripes缺點(diǎn):
社區(qū)比較小
不如其他的項(xiàng)目活躍
ActionBean里面的URL是硬編碼的
Struts 2優(yōu)點(diǎn):
架構(gòu)簡(jiǎn)單--易于擴(kuò)展
標(biāo)記庫(kù)很容易利用FreeMarker或者Velocity來(lái)定制
基于控制器或
者基于頁(yè)面的導(dǎo)航
Struts 2缺點(diǎn):
文檔組織得很差
對(duì)新特征過(guò)分關(guān)注
Tapestry優(yōu)點(diǎn):
一旦學(xué)會(huì)它,將極大地提高生產(chǎn)率
HTML模板--對(duì)頁(yè)面設(shè)計(jì)師非常有利
每出一個(gè)新版本,都會(huì)有大量的創(chuàng)新
Tapestry缺點(diǎn):
文檔過(guò)于概念性,不夠?qū)嵱?/p>
學(xué)習(xí)曲線(xiàn)陡峭
發(fā)行周期長(zhǎng)--每年都有較大的升級(jí)
Wicket優(yōu)點(diǎn):
對(duì)Java開(kāi)發(fā)者有利(不是Web開(kāi)發(fā)者)
頁(yè)面和顯示綁定緊密
社區(qū)活躍--有來(lái)自創(chuàng)建者的支持
Wicket缺點(diǎn):
HTML模板和Java代碼緊挨著
需要對(duì)OO有較好的理解
Wicket邏輯--什么都用Java搞定
想要實(shí)現(xiàn)一個(gè)簡(jiǎn)單的登錄功能的話(huà),可以使用Servlet+jsp來(lái)實(shí)現(xiàn),jsp編寫(xiě)登錄界面和登錄后的要出現(xiàn)信息界面和登錄失敗的信息界面,Servlet類(lèi)用來(lái)對(duì)表單提交的用戶(hù)名和密碼進(jìn)行判斷和處理。
具體代碼如下:
Servlet類(lèi):
public class DemoServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
String loginname = request.getParameter("loginname");
String password = request.getParameter("password");
if(loginname.equals("a") password.equals("a")){
request.setAttribute("msg", "登錄成功");
request.getRequestDispatcher("/loginsuccess.jsp").forward(request, response);
}else{
request.setAttribute("msg", "登錄失敗");
request.getRequestDispatcher("/loginsuccess.jsp").forward(request, response);
}
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doGet(request, response);
}
}
jsp頁(yè)面:
%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%
!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
html
head
titleDemo/title
meta http-equiv="pragma" content="no-cache"
meta http-equiv="cache-control" content="no-cache"
meta http-equiv="expires" content="0"
meta http-equiv="keywords" content="keyword1,keyword2,keyword3"
meta http-equiv="description" content="This is my page"
/head
body
form action="demoServlet" method="post"
input type="text" name="loginname"/br/
input type="password" name="password"/br/
input type="submit" value="登錄"/
/form
/body
/html
登錄信息頁(yè)面:
%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%
%@ taglib prefix="c" uri="標(biāo)簽庫(kù)地址"%
!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
html
head
titleMy JSP 'loginsuccess.jsp' starting page/title
meta http-equiv="pragma" content="no-cache"
meta http-equiv="cache-control" content="no-cache"
meta http-equiv="expires" content="0"
meta http-equiv="keywords" content="keyword1,keyword2,keyword3"
meta http-equiv="description" content="This is my page"
/head
body
${msg }
/body
/html
需要介紹一下:登錄信息的這個(gè)頁(yè)面中的${msg }是使用jstl標(biāo)簽,需要在jsp頁(yè)面中導(dǎo)入jstl標(biāo)簽庫(kù),使用這個(gè)標(biāo)簽庫(kù)可以節(jié)省很多代碼量。
Java代碼編寫(xiě)的30條實(shí)踐建議,很多人認(rèn)為學(xué)習(xí)java需要較好的計(jì)算機(jī)語(yǔ)言基礎(chǔ),而事實(shí)上高中學(xué)歷的學(xué)習(xí)java,月薪過(guò)萬(wàn)的比比皆是,Java代碼編寫(xiě)的30條實(shí)踐建議,java知識(shí)點(diǎn)學(xué)習(xí)貴在積累。
Java代碼編寫(xiě)的30條實(shí)踐建議:
(1) 類(lèi)名首字母應(yīng)該大寫(xiě)。字段、方法以及對(duì)象(句柄)的首字母應(yīng)小寫(xiě)。對(duì)于所有標(biāo)識(shí)符,其中包含的所有單詞都應(yīng)緊靠在一起,而且大寫(xiě)中間單詞的首字母。
例如:
ThisIsAClassName
thisIsMethodOrFieldName
若在定義中出現(xiàn)了常數(shù)初始化字符,則大寫(xiě)static final基本類(lèi)型標(biāo)識(shí)符中的所有字母。這樣便可標(biāo) 志出它們屬于編譯期的常數(shù)。
Java包(Package)屬于一種特殊情況:它們?nèi)际切?xiě)字母,即便中間的單詞亦是如此。對(duì)于域名擴(kuò)展名稱(chēng),如com,org,net或者edu等,全部都應(yīng)小寫(xiě)(這也是Java 1.1和Java 1.2的區(qū)別之一)。
(2) 為了常規(guī)用途而創(chuàng)建一個(gè)類(lèi)時(shí),請(qǐng)采取"經(jīng)典形式",并包含對(duì)下述元素的定義:
equals()
hashCode()
toString()
clone()(implement Cloneable)
implement Serializable
(3) 對(duì)于自己創(chuàng)建的每一個(gè)類(lèi),都考慮置入一個(gè)main(),其中包含了用于測(cè)試那個(gè)類(lèi)的代碼。為使用一個(gè)項(xiàng)目中的類(lèi),我們沒(méi)必要?jiǎng)h除測(cè)試代碼。若進(jìn)行了任何形式的改動(dòng),可方便地返回測(cè)試。這些代碼也可作為如何使用類(lèi)的一個(gè)示例使用。
(4) 應(yīng)將方法設(shè)計(jì)成簡(jiǎn)要的、功能性單元,用它描述和實(shí)現(xiàn)一個(gè)不連續(xù)的類(lèi)接口部分。理想情況下,方法應(yīng)簡(jiǎn)明扼要。若長(zhǎng)度很大,可考慮通過(guò)某種方式將其分割成較短的幾個(gè)方法。這樣做也便于類(lèi)內(nèi)代碼的重復(fù)使用(有些時(shí)候,方法必須非常大,但它們?nèi)詰?yīng)只做同樣的一件事情)。
(5) 設(shè)計(jì)一個(gè)類(lèi)時(shí),請(qǐng)?jiān)O(shè)身處地為客戶(hù)程序員考慮一下(類(lèi)的使用方法應(yīng)該是非常明確的)。然后,再設(shè)身處地為管理代碼的人考慮一下(預(yù)計(jì)有可能進(jìn)行哪些形式的修改,想想用什么方法可把它們變得更簡(jiǎn)單)。
(6) 使類(lèi)盡可能短小精悍,而且只解決一個(gè)特定的問(wèn)題。下面是對(duì)類(lèi)設(shè)計(jì)的一些建議:
■一個(gè)復(fù)雜的開(kāi)關(guān)語(yǔ)句:考慮采用"多形"機(jī)制
■數(shù)量眾多的方法涉及到類(lèi)型差別極大的操作:考慮用幾個(gè)類(lèi)來(lái)分別實(shí)現(xiàn)
■許多成員變量在特征上有很大的差別:考慮使用幾個(gè)類(lèi)
(7) 讓一切東西都盡可能地"私有"--private??墒箮?kù)的某一部分"公共化"(一個(gè)方法、類(lèi)或者一個(gè)字段等等),就永遠(yuǎn)不能把它拿出。若強(qiáng)行拿出,就可能破壞其他人現(xiàn)有的代碼,使他們不得不重新編寫(xiě)和設(shè)計(jì)。若只公布自己必須公布的,就可放心大膽地改變其他任何東西。
在多線(xiàn)程環(huán)境中,隱私是特別重要的一個(gè)因素--只有private字段才能在非同步使用的情況下受到保護(hù)。
(8) 謹(jǐn)惕"巨大對(duì)象綜合癥"。對(duì)一些習(xí)慣于順序編程思維、且初涉OOP領(lǐng)域的新手,往往喜歡先寫(xiě)一個(gè)順序執(zhí)行的程序,再把它嵌入一個(gè)或兩個(gè)巨大的對(duì)象里。根據(jù)編程原理,對(duì)象表達(dá)的應(yīng)該是應(yīng)用程序的概念,而非應(yīng)用程序本身。
(9) 若不得已進(jìn)行一些不太雅觀(guān)的編程,至少應(yīng)該把那些代碼置于一個(gè)類(lèi)的內(nèi)部。
(10) 任何時(shí)候只要發(fā)現(xiàn)類(lèi)與類(lèi)之間結(jié)合得非常緊密,就需要考慮是否采用內(nèi)部類(lèi),從而改善編碼及維護(hù)工作(參見(jiàn)第14章14.1.2小節(jié)的"用內(nèi)部類(lèi)改進(jìn)代碼")。
(11) 盡可能細(xì)致地加上注釋?zhuān)⒂胘avadoc注釋文檔語(yǔ)法生成自己的程序文檔。
(12) 避免使用"魔術(shù)數(shù)字",這些數(shù)字很難與代碼很好地配合。如以后需要修改它,無(wú)疑會(huì)成為一場(chǎng)噩夢(mèng),因?yàn)楦静恢?100"到底是指"數(shù)組大小"還是"其他全然不同的東西"。所以,我們應(yīng)創(chuàng)建一個(gè)常數(shù),并為其使用具有說(shuō)服力的描述性名稱(chēng),并在整個(gè)程序中都采用常數(shù)標(biāo)識(shí)符。這樣可使程序更易理解以及更易維護(hù)。
(13) 涉及構(gòu)建器和異常的時(shí)候,通常希望重新丟棄在構(gòu)建器中捕獲的任何異常--如果它造成了那個(gè)對(duì)象的創(chuàng)建失敗。這樣一來(lái),調(diào)用者就不會(huì)以為那個(gè)對(duì)象已正確地創(chuàng)建,從而盲目地繼續(xù)。
(14) 當(dāng)客戶(hù)程序員用完對(duì)象以后,若你的類(lèi)要求進(jìn)行任何清除工作,可考慮將清除代碼置于一個(gè)良好定義的方法里,采用類(lèi)似于cleanup()這樣的名字,明確表明自己的用途。除此以外,可在類(lèi)內(nèi)放置一個(gè)boolean(布爾)標(biāo)記,指出對(duì)象是否已被清除。在類(lèi)的finalize()方法里,請(qǐng)確定對(duì)象已被清除,并已丟棄了從RuntimeException繼承的一個(gè)類(lèi)(如果還沒(méi)有的話(huà)),從而指出一個(gè)編程錯(cuò)誤。在采取象這樣的方案之前,請(qǐng)確定finalize()能夠在自己的系統(tǒng)中工作(可能需要調(diào)用System.runFinalizersOnExit(true),從而確保這一行為)。
(15) 在一個(gè)特定的作用域內(nèi),若一個(gè)對(duì)象必須清除(非由垃圾收集機(jī)制處理),請(qǐng)采用下述方法:初始化對(duì)象;若成功,則立即進(jìn)入一個(gè)含有finally從句的try塊,開(kāi)始清除工作。
(16) 若在初始化過(guò)程中需要覆蓋(取消)finalize(),請(qǐng)記住調(diào)用super.finalize()(若Object屬于我們的直接超類(lèi),則無(wú)此必要)。在對(duì)finalize()進(jìn)行覆蓋的過(guò)程中,對(duì)super.finalize()的調(diào)用應(yīng)屬于最后一個(gè)行動(dòng),而不應(yīng)是第一個(gè)行動(dòng),這樣可確保在需要基礎(chǔ)類(lèi)組件的時(shí)候它們依然有效。
(17) 創(chuàng)建大小固定的對(duì)象集合時(shí),請(qǐng)將它們傳輸至一個(gè)數(shù)組(若準(zhǔn)備從一個(gè)方法里返回這個(gè)集合,更應(yīng)如此操作)。這樣一來(lái),我們就可享受到數(shù)組在編譯期進(jìn)行類(lèi)型檢查的好處。此外,為使用它們,數(shù)組的接收者也許并不需要將對(duì)象"造型"到數(shù)組里。
(18) 盡量使用interfaces,不要使用abstract類(lèi)。若已知某樣?xùn)|西準(zhǔn)備成為一個(gè)基礎(chǔ)類(lèi),那么第一個(gè)選擇應(yīng)是將其變成一個(gè)interface(接口)。只有在不得不使用方法定義或者成員變量的時(shí)候,才需要將其變成一個(gè)abstract(抽象)類(lèi)。接口主要描述了客戶(hù)希望做什么事情,而一個(gè)類(lèi)則致力于(或允許)具體的實(shí)施細(xì)節(jié)。
(19) 在構(gòu)建器內(nèi)部,只進(jìn)行那些將對(duì)象設(shè)為正確狀態(tài)所需的工作。盡可能地避免調(diào)用其他方法,因?yàn)槟切┓椒赡鼙黄渌烁采w或取消,從而在構(gòu)建過(guò)程中產(chǎn)生不可預(yù)知的結(jié)果(參見(jiàn)第7章的詳細(xì)說(shuō)明)。
(20) 對(duì)象不應(yīng)只是簡(jiǎn)單地容納一些數(shù)據(jù);它們的行為也應(yīng)得到良好的定義。
(21) 在現(xiàn)成類(lèi)的基礎(chǔ)上創(chuàng)建新類(lèi)時(shí),請(qǐng)首先選擇"新建"或"創(chuàng)作"。只有自己的設(shè)計(jì)要求必須繼承時(shí),才應(yīng)考慮這方面的問(wèn)題。若在本來(lái)允許新建的場(chǎng)合使用了繼承,則整個(gè)設(shè)計(jì)會(huì)變得沒(méi)有必要地復(fù)雜。
(22) 用繼承及方法覆蓋來(lái)表示行為間的差異,而用字段表示狀態(tài)間的區(qū)別。一個(gè)非常極端的例子是通過(guò)對(duì)不同類(lèi)的繼承來(lái)表示顏色,這是絕對(duì)應(yīng)該避免的:應(yīng)直接使用一個(gè)"顏色"字段。
(23) 為避免編程時(shí)遇到麻煩,請(qǐng)保證在自己類(lèi)路徑指到的任何地方,每個(gè)名字都僅對(duì)應(yīng)一個(gè)類(lèi)。否則,編譯器可能先找到同名的另一個(gè)類(lèi),并報(bào)告出錯(cuò)消息。若懷疑自己碰到了類(lèi)路徑問(wèn)題,請(qǐng)?jiān)囋囋陬?lèi)路徑的每一個(gè)起點(diǎn),搜索一下同名的.class文件。
(24) 在Java 1.1 AWT中使用事件"適配器"時(shí),特別容易碰到一個(gè)陷阱。若覆蓋了某個(gè)適配器方法,同時(shí)拼寫(xiě)方法沒(méi)有特別講究,最后的結(jié)果就是新添加一個(gè)方法,而不是覆蓋現(xiàn)成方法。然而,由于這樣做是完全合法的,所以不會(huì)從編譯器或運(yùn)行期系統(tǒng)獲得任何出錯(cuò)提示--只不過(guò)代碼的工作就變得不正常了。
(25) 用合理的設(shè)計(jì)方案消除"偽功能"。也就是說(shuō),假若只需要?jiǎng)?chuàng)建類(lèi)的一個(gè)對(duì)象,就不要提前限制自己使用應(yīng)用程序,并加上一條"只生成其中一個(gè)"注釋。請(qǐng)考慮將其封裝成一個(gè)"獨(dú)生子"的形式。若在主程序里有大量散亂的代碼,用于創(chuàng)建自己的對(duì)象,請(qǐng)考慮采納一種創(chuàng)造性的方案,將些代碼封裝起來(lái)。
(26) 警惕"分析癱瘓"。請(qǐng)記住,無(wú)論如何都要提前了解整個(gè)項(xiàng)目的狀況,再去考察其中的細(xì)節(jié)。由于把握了全局,可快速認(rèn)識(shí)自己未知的一些因素,防止在考察細(xì)節(jié)的時(shí)候陷入"死邏輯"中。
(27) 警惕"過(guò)早優(yōu)化"。首先讓它運(yùn)行起來(lái),再考慮變得更快--但只有在自己必須這樣做、而且經(jīng)證實(shí)在某部分代碼中的確存在一個(gè)性能瓶頸的時(shí)候,才應(yīng)進(jìn)行優(yōu)化。除非用專(zhuān)門(mén)的工具分析瓶頸,否則很有可能是在浪費(fèi)自己的時(shí)間。
性能提升的隱含代價(jià)是自己的代碼變得難于理解,而且難于維護(hù)。
(28) 請(qǐng)記住,閱讀代碼的時(shí)間比寫(xiě)代碼的時(shí)間多得多。思路清晰的設(shè)計(jì)可獲得易于理解的程序,但注釋、細(xì)致的解釋以及一些示例往往具有不可估量的價(jià)值。無(wú)論對(duì)你自己,還是對(duì)后來(lái)的人,它們都是相當(dāng)重要的。如對(duì)此仍有懷疑,那么請(qǐng)?jiān)囅胱约涸噲D從聯(lián)機(jī)Java文檔里找出有用信息時(shí)碰到的挫折,這樣或許能將你說(shuō)服。
(29) 如認(rèn)為自己已進(jìn)行了良好的分析、設(shè)計(jì)或者實(shí)施,那么請(qǐng)稍微更換一下思維角度。試試邀請(qǐng)一些外來(lái)人士--并不一定是專(zhuān)家,但可以是來(lái)自本公司其他部門(mén)的人。請(qǐng)他們用完全新鮮的眼光考察你的工作,看看是否能找出你一度熟視無(wú)睹的問(wèn)題。采取這種方式,往往能在最適合修改的階段找出一些關(guān)鍵性的問(wèn)題,避免產(chǎn)品發(fā)行后再解決問(wèn)題而造成的金錢(qián)及精力方面的損失。
(30) 良好的設(shè)計(jì)能帶來(lái)最大的回報(bào)。簡(jiǎn)言之,對(duì)于一個(gè)特定的問(wèn)題,通常會(huì)花較長(zhǎng)的時(shí)間才能找到一種最恰當(dāng)?shù)慕鉀Q方案。但一旦找到了正確的方法,以后的工作就輕松多了,再也不用經(jīng)歷數(shù)小時(shí)、數(shù)天或者數(shù)月的痛苦掙扎。我們的努力工作會(huì)帶來(lái)最大的回報(bào)(甚至無(wú)可估量)。而且由于自己傾注了大量心血,最終獲得一個(gè)出色的設(shè)計(jì)方案,成功的快感也是令人心動(dòng)的。堅(jiān)持抵制草草完工的誘惑--那樣做往往得不償失。
一:學(xué)會(huì)如何讀一個(gè)JavaWeb項(xiàng)目源代碼 步驟:表結(jié)構(gòu)-web.xml-mvc-db-spring
ioc-log- 代碼
1、先了解項(xiàng)目數(shù)據(jù)庫(kù)的表結(jié)構(gòu),這個(gè)方面是最容易忘記 的,有時(shí)候我們只顧著看每一個(gè)方法是怎么進(jìn)行的,卻沒(méi)
有去了解數(shù)據(jù)庫(kù)之間的主外鍵關(guān)聯(lián)。其實(shí)如果先了解數(shù)據(jù) 庫(kù)表結(jié)構(gòu),再去看一個(gè)方法的實(shí)現(xiàn)會(huì)更加容易。
2、然后需要過(guò)一遍web.xml,知道項(xiàng)目中用到了什么攔
截器,監(jiān)聽(tīng)器,過(guò)濾器,擁有哪些配置文件。如果是攔截 器,一般負(fù)責(zé)過(guò)濾請(qǐng)求,進(jìn)行AOP 等;如果是監(jiān) 可能是定時(shí)任務(wù),初始化任務(wù);配置文件有如使用了 spring
后的讀取mvc 相關(guān),db 相關(guān),service 相關(guān),aop 相關(guān)的文件。
3、查看攔截器,監(jiān)聽(tīng)器代碼,知道攔截了什么請(qǐng)求,這
個(gè)類(lèi)完成了怎樣的工作。有的人就是因?yàn)槿鄙倭诉@一步, 自己寫(xiě)了一個(gè)action,配置文件也沒(méi)有寫(xiě)錯(cuò),但是卻怎么
調(diào)試也無(wú)法進(jìn)入這個(gè)action,直到別人告訴他,請(qǐng)求被攔
4、接下來(lái),看配置文件,首先一定是mvc相關(guān)的,如 springmvc
中,要請(qǐng)求哪些請(qǐng)求是靜態(tài)資源,使用了哪些 view 策略,controller 注解放在哪個(gè)包下等。 然后是db 相關(guān)配置文件,看使用了什么數(shù)據(jù)庫(kù),使用了
什么orm框架,是否開(kāi)啟了二級(jí)緩存,使用哪種產(chǎn)品作 為二級(jí)緩存,事務(wù)管理的處理,需要掃描的實(shí)體類(lèi)放在什 么位置。最后是spring 核心的ioc
功能相關(guān)的配置文件, 知道接口與具體類(lèi)的注入大致是怎樣的。當(dāng)然還有一些如 apectj 置文件,也是在這個(gè)步驟中完成
5、log
相關(guān)文件,日志的各個(gè)級(jí)別是如何處理的,在哪些 地方使用了log 記錄日志
6、從上面幾點(diǎn)后知道了整個(gè)開(kāi)源項(xiàng)目的整體框架,閱讀 每個(gè)方法就不再那么難了。
7、當(dāng)然如果有項(xiàng)目配套的開(kāi)發(fā)文檔也是要閱讀的。
1.從代碼結(jié)構(gòu)入手,搞清楚Java源碼所在目錄以及jsp代碼所在目錄。\x0d\x0a2.從Web.xml入手,搞清楚首頁(yè),上下文及配置情況。\x0d\x0a3.從lib目錄里初步查看系統(tǒng)使用的框架和工具包。\x0d\x0a4.從頁(yè)面大體瀏覽,看頁(yè)面的scriptlet使用的主要語(yǔ)言。\x0d\x0a5.從代碼包結(jié)構(gòu)初步知曉代碼的層次。\x0d\x0a6.從配置文件入手,弄清楚頁(yè)面到Action/Servlet的配置關(guān)系。\x0d\x0a7.搞清楚代碼提交的方式,前臺(tái)請(qǐng)求如何傳遞到后臺(tái),又是如何返回到前臺(tái)頁(yè)面的,這里以L(fǎng)ogin頁(yè)面為最佳切入點(diǎn)。\x0d\x0a8.從處理Login的Action/Servlet中觀(guān)察,看在session中記錄了多少重要的數(shù)據(jù),后面程序又是怎樣用到這些數(shù)據(jù)的。\x0d\x0a9.如果存在樹(shù)形菜單,要搞清楚樹(shù)形菜單的形成及傳遞請(qǐng)求的方式。\x0d\x0a10.歸納幾種典型頁(yè)面,搞清楚其處理和響應(yīng)形式。\x0d\x0a11.歸納通用的頁(yè)面組件如分頁(yè),上傳下載,異步請(qǐng)求等原系統(tǒng)是如何處理的。\x0d\x0a12.搞清楚后臺(tái)數(shù)據(jù)來(lái)源及配置,主要是數(shù)據(jù)庫(kù)及通過(guò)WebService方式。\x0d\x0a13.歸納后臺(tái)處理的主要業(yè)務(wù)。\x0d\x0a14.可能的話(huà),把后臺(tái)數(shù)據(jù)表的ER圖畫(huà)出來(lái)。\x0d\x0a15.典型業(yè)務(wù)的流程圖可以繪制出來(lái)。
FileOutputStream很明顯你是用的文件流返回的
// 以byte流的方式打開(kāi)文件 d:\1.gif? ?
FileInputStream hFile = new FileInputStream(url); ? ? ?//得到文件大小
int i=hFile.available();
byte data[]=new byte[i]; ? ? ? ?//讀數(shù)據(jù)
hFile.read(data); ?? ? ? ?//得到向客戶(hù)端輸出二進(jìn)制數(shù)據(jù)的對(duì)象
OutputStream toClient=response.getOutputStream(); ?? ? ? ?//輸出數(shù)據(jù)
toClient.write(data);
toClient.flush();
toClient.close();
hFile.close();
擴(kuò)展資料:
如果是純文本使用字符流,如果二進(jìn)制文件,使用字節(jié)流。
如果只是得到信息,原樣不動(dòng),不進(jìn)行修改操作,例如文件上傳和下載,這時(shí)就使用字節(jié)流。文件上傳:在服務(wù)器端把瀏覽器端信息提取出來(lái)。文件下載:把服務(wù)器端內(nèi)容寫(xiě)給瀏覽器端。
如果要操作的是自定義信息,這時(shí)使用字符流。
通過(guò)response獲取的輸出流它的真實(shí)類(lèi)型是什么?
ServletOutputStream? response.getOutputStream();
PrintWriter? response.getWriter();
ServletOutputStream由于使用字節(jié)流多數(shù)是原樣復(fù)制,所以使用write方法,而不是print方法。
PrintWriter:打印流,兩個(gè)特點(diǎn):1.可以設(shè)置自動(dòng)刷新。2.可以將信息原樣輸出。