一.Servlet規(guī)范
什么是Servlet:
JavaEE中的一種規(guī)范,也是一種組件。在Java中最小的程序單元就是類,Servlet其實(shí)就是一種特殊的類.特殊在必須遵循Servlet規(guī)范(接口).
二.Servlet的第一個程序
1.搭建JavaWeb項(xiàng)目:
1.創(chuàng)建一個Java項(xiàng)目:HelloServletWeb;
2.在HelloServletWeb中創(chuàng)建一個文件夾webapp,表示W(wǎng)eb項(xiàng)目的根;
3.在webapp中創(chuàng)建WEB-INF文件夾,
4.在WEB-IN中創(chuàng)建文件夾:lib,classes
5.在WEB-IN中去Tomcat根/conf拷貝web.xml文件,只需要保留根元素.
6.把當(dāng)前項(xiàng)目的classpath路徑改成webapp/WEB-IN下的classes中.
2.編寫Servlet:
1.為該項(xiàng)目增加Servlet的支持.
- 1).把Tomcat根/lib中servlet-api.jar文件拷貝到項(xiàng)目下WEB-INF下的lib中
- 2).在項(xiàng)目中選擇servlet-api.jar,鼠標(biāo)右鍵,build path-->add to build path
2.開發(fā)Servlet程序:
- 1)定義一個類HelloKittyServlet,并讓該類去實(shí)現(xiàn)javax.servlet.Servlet接口;
- 2)實(shí)現(xiàn)Servlet接口中的init,service,destory等方法.
注意:若生成方法中的參數(shù)是arg0或則arg1等格式的,原因是還沒有關(guān)聯(lián)源代碼的問題:關(guān)聯(lián)上:apache-tomcat-7.0.42-src.zip,并重新實(shí)現(xiàn)/復(fù)寫方法就OK;
3.配置Servlet:
HelloKittyServlet,僅僅是一個普通的實(shí)現(xiàn)類而已,而我最終要運(yùn)行在Tomcat服務(wù)器中,所以得告訴Tomcat,來幫我管理HelloKittyServlet類;
1.找到項(xiàng)目根下的WEB-INF下的web.xml文件:
2.在根元素web-app中創(chuàng)建一個新的元素節(jié)點(diǎn):servlet
2.在根元素web-app中創(chuàng)建一個新的元素節(jié)點(diǎn):servlet-mapping
Tomcat服務(wù)器是Servlet容器:
容器:負(fù)責(zé)了容器內(nèi)對象的生命周期維護(hù)工作.
創(chuàng)建對象-->初始化操作--->使用對象調(diào)用方法-->銷毀操作--->GC
4.部署項(xiàng)目/訪問項(xiàng)目:
訪問方式:http://ip:port/contextPath/資源名
http://localhost:80/day3/hello .
三.Servlet的生命周期
1.javax.servlet.Servlet接口中的方法:
- 1):ServletConfig getServletConfig:獲取當(dāng)前Servlet的配置信息對象.
- 2):String getServletInfo:用于返回當(dāng)前Servlet的作者,版本,版權(quán)信息.
- 3):void init: 初始化方法
- 4):void service: 服務(wù)方法
- 5):void destory:銷毀方法
2.Servlet的生命周期方法:
- 1)Servlet的構(gòu)造器: 只在第一次請求Servlet的時候,先執(zhí)行構(gòu)造器(創(chuàng)建Servlet對象).
- 2)void init(ServletConfig config):只在第一次請求時,創(chuàng)建好Servlet對象之后,做初始化操作.
- 3)void service(ServletRequest req, ServletResponse res): 每次請求都會執(zhí)行,處理請求的.
- 4)void destroy():在Tomcat正常關(guān)閉的時候執(zhí)行,不要期望該方法一定會執(zhí)行,不需要把掃尾的操作存放在這里.
結(jié)論:
1.Servlet是單例的.Servlet只要第一次請求之后,創(chuàng)建好一個對象,該對象就一直存在,直到Tomcat關(guān)閉.
一個Servlet類,最多之后一個實(shí)例.(單實(shí)例,性能有保證,但是多個線程并發(fā)可能造成安全問題).2.Servlet構(gòu)造器必須是公共的無參數(shù)的.
注意:
Servlet的第一次請求:
執(zhí)行構(gòu)造器,init方法,service方法.
Servlet的非第一次請求:
只會執(zhí)行service方法.
四.Servlet的請求流程
1.瀏覽器先向Tomcat發(fā)出請求:
http://localhost:80/day3/hello.
2.解析請求:
http://:表示當(dāng)前請求遵循h(huán)ttp協(xié)議.
localhost:通過DNS解析成IP,從互聯(lián)網(wǎng)上去尋找該IP的主機(jī).
80: 從該主機(jī)中找到80端口對應(yīng)的程序.
day3: 上下文路徑
/hello:資源名稱
3.讀取Tomcat根/conf中的serverx.xml文件
并獲取所有的<Context/>元素(document.getElementsByTagName(..)):
遍歷多個<Context/>找到path屬性值為day3的<Context>元素.
進(jìn)而得到當(dāng)前<Context/>元素的docBase屬性值(當(dāng)前訪問的Web項(xiàng)目的根路徑):D:\JavaApps\Servlet\webapp.
4.從當(dāng)前訪問Web項(xiàng)目的根路徑/WEB-INF中去獲取web.xml文件.
獲取web.xml中所有的<url-pattern>元素,并判斷哪一個<url-pattern>元素的文本內(nèi)容為/hello.
找不到:響應(yīng)404.
找 到:根據(jù)文本內(nèi)容為/hello的<url-pattern>找到當(dāng)前被訪問Servlet的全限定名:
String className = "com._520it._01_hello.HelloServlet";
5.判斷Servlet的實(shí)例緩存池(Map)中是否存在key為com._520it._01_hello.HelloServlet的Servlet對象.
Map<String,Servlet> cacheMap = ......;
Servlet obj = cacheMap.get("com._520it._01_hello.HelloServlet");
if(obj == null){
//第一次
GOTO 6:
}else{
//非第一次
GOTO 8:
}
6.使用反射來創(chuàng)建Servlet對象(調(diào)用構(gòu)造器),并存放于Servlet實(shí)例緩存池中,供下次使用.
Servet obj = (Servlet)Class.forName("com._520it._01_hello.HelloServlet").newInstance();
//存放于Servlet實(shí)例緩存池中,供下次使用
cacheMap.put("com._520it._01_hello.HelloServlet",obj);
7.Tomcat創(chuàng)建ServletConfig對象,并調(diào)用init方法,完成初始化操作.
obj.init(config);
8.Tomcat創(chuàng)建ServletRequest和ServletResponse對象,并調(diào)用service方法,完成處理請求的方法.
obj.service(req,resp);
9.應(yīng)該在service方法中,處理完請求給瀏覽器做相應(yīng)操作.
五.Servlet初始化參數(shù)
ServletConfig接口:表示Servlet的配置對象,提供了四個方法用于獲取當(dāng)前Servlet的配置信息.
ServletConfig對象是有容器(Tomcat)創(chuàng)建的,我們就只管用.
四個方法:
String getServletName(): 獲取<servlet-name>的文本內(nèi)容.
ServletContext getServletContext():獲取當(dāng)前Servlet的上下文對象,當(dāng)前應(yīng)用對象.
String getInitParameter(String paranName):根據(jù)指定的初始化參數(shù)名稱獲取對應(yīng)的初始化參數(shù)值.
Enumeration<String> getInitParameterNames():獲取所有的初始化參數(shù)的名字.
上述代碼中出現(xiàn)硬編碼,不利于維護(hù).
解決方案:把寫死的數(shù)據(jù)存放到配置文件中.
Servlet的初始化參數(shù)機(jī)制.
六.Servlet的繼承體系
自定義Servlet繼承HttpServlet之后,又如何提高處理請求的方法:
public class NormalServlet extends HttpServlet{
protected void service(HttpServletRequest arg0, HttpServletResponse arg1) throws ServletException, IOException {
// TODO Auto-generated method stub
super.service(arg0, arg1);
}
//專門用于處理GET請求
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// TODO Auto-generated method stub
this.doGet(req, resp);
}
//專門用于處理POST請求
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// TODO Auto-generated method stub
this.doPost(req, resp);
}
public void init(ServletConfig config) throws ServletException {
// TODO Auto-generated method stub
super.init(config);
System.out.println("自身的初始化操作...");
}
public void init() throws ServletException {
// TODO Auto-generated method stub
System.out.println("自身的初始化操作...");
}
}
Servlet編寫總結(jié):
- 1):自定義Servlet類直接繼承于HttpServlet即可.
- 2):Servlet需要處理請求,只需要覆蓋父類的service方法(參數(shù)是Http類型的);
public void service(ServletRequest arg0, ServletResponse arg1) throws ServletException, IOException {
// TODO Auto-generated method stub
//super.service(arg0, arg1);
}
- 3):如果Servlet需要做初始化操作,只需要提供無參數(shù)的init方法即可.
七.HttpServletRequest常用方法
1.Request的類型
ServletRequest接口:表示Servlet的請求對象,包含了處理請求的方法.
HttpServletRequest接口:是ServletRequest接口的子接口,支持Http的請求處理.
2.HTTP請求信息
HTTP請求:包含三部分(請求行,請求頭,請求實(shí)體).
HttpServletRequest中就提供了獲取HTTP請求信息的所有的方法.
3.常用方法:
- 1):String getMethod():返回請求方式:如GET/POST
- 2):String getContextPath():返回請求URL所屬Web應(yīng)用的路徑怀偷。路徑以"/"開頭
- 3):String getRequestURI():返回請求行中的資源名字部分:如/test/index.html
- 4):StringBuffer getRequestURL():返回瀏覽器地址欄信息
- 5):String getRemoteAddr():返回發(fā)出請求的客戶機(jī)的IP地址
- 6):String getHeader(String name):根據(jù)指定的請求頭名稱獲取對應(yīng)的請求頭的值.
4.獲取請求參數(shù)(重要)
1.String String getParameter(String name):返回指定名字參數(shù)的值。
注意: 請求參數(shù)一般說來,是用戶在表單中填寫的數(shù)據(jù).2.String[] getParameterValues(String name):返回指定名字參數(shù)的多個參數(shù)值减噪。
3.Enumeration<String> getParameterNames():返回所有參數(shù)名的Enumeration對象。
4.Map<String,String[]> getParameterMap():返回所有的參數(shù)和值所組成的Map對象车吹。
5.ServletConfig:
String getInitParameter(String paranName):獲取初始化參數(shù)的,
初始化參數(shù)是我們自己為了解除硬編碼,自己在web.xml中<servlet>元素中配置的.
6.HttpServletRequest:
String getParameter(String paranName):獲取請求參數(shù).
請求參數(shù)一般說來,是用戶在表單中填寫的數(shù)據(jù).
八.注冊案例
九.請求的中文亂碼處理
1.請求參數(shù)中帶有中文會出現(xiàn)亂碼問題:
為什么會出現(xiàn)中文亂碼問題:
Tomcat的默認(rèn)編碼使用的ISO-8859-1,不支持中文.
原因是:服務(wù)端把二進(jìn)制數(shù)據(jù)轉(zhuǎn)換為String的時候,使用了ISO-8859-1來編碼.
2.解決方案一:
針對于POST和GET方式都起作用.
- 1):先按照ISO-8859-1把亂碼的數(shù)據(jù)恢復(fù)成二進(jìn)制格式(byte數(shù)組).
byte [] data= username.getBytes("ISO-8859-1"); - 2):使用UTF-8對二進(jìn)制數(shù)據(jù)重新編碼.
username = new String(data,"UTF-8");
上述代碼沒問題,但是如果表單中的數(shù)據(jù)比較多,那么就的處理N次解碼,再編碼:
3.解決方案二:
1)針對于POST請求,不包括GET請求:
在獲取任意參數(shù)之前,設(shè)置請求的編碼:
req.setCharacterEncoding("UTF-8");
2)針對GET請求:
修改Tomcat中默認(rèn)的處理GET請求的編碼(了解),僅僅只對GET方式有效..
Tomcat根/conf/server.xml,在修改端口的那一行(71行);
建議:以后表單統(tǒng)統(tǒng)使用POST方式來傳遞.
十.HttpServletResponse常用方法
1.Response的類型
ServletResponse接口:處理一般的響應(yīng)操作,包含處理響應(yīng)操作的方法.
HttpServletResponse接口:處理HTPP的響應(yīng)操作,包含了處理HTTP響應(yīng)的方法.
HttpServletResponse是ServletResponse的子接口.
2.常用方法:
1)獲取輸出流對象:
獲取字節(jié)輸出流: OutputStream out = resp.getOutputStream();(文件下載使用)
獲取字符輸出流: PrintWriter out = resp.getWriter();
** 注意: **resp.getWriter()和resp.getOutputStream(),只能調(diào)用一個方法,否則報錯.
2) 設(shè)置響應(yīng)時的編碼:必須在獲取輸出流之前設(shè)置.
response.setCharacterEncoding("UTF-8");
3)設(shè)置響應(yīng)輸出的MIME類型:
response.setContentType("text/html");
注意:上述兩行代碼可以合體,同時設(shè)置響應(yīng)的MIME類型和編碼:
response.setContentType("text/html;charset=utf-8");