一.Servlet规范
什么是Servlet:
JavaEE中的一种规范,也是一种组件。在Java中最小的程序单元就是类,Servlet其实就是一种特殊的类.特殊在必须遵循Servlet规范(接口).
二.Servlet的第一个程序
1.搭建JavaWeb项目:
1.创建一个Java项目:HelloServletWeb;
2.在HelloServletWeb中创建一个文件夹webapp,表示Web项目的根;
3.在webapp中创建WEB-INF文件夹,
4.在WEB-IN中创建文件夹:lib,classes
5.在WEB-IN中去Tomcat根/conf拷贝web.xml文件,只需要保留根元素.
6.把当前项目的classpath路径改成webapp/WEB-IN下的classes中.
2.编写Servlet:
1.为该项目增加Servlet的支持.
- 1).把Tomcat根/lib中servlet-api.jar文件拷贝到项目下WEB-INF下的lib中
- 2).在项目中选择servlet-api.jar,鼠标右键,build path-->add to build path
2.开发Servlet程序:
- 1)定义一个类HelloKittyServlet,并让该类去实现javax.servlet.Servlet接口;
- 2)实现Servlet接口中的init,service,destory等方法.
注意:若生成方法中的参数是arg0或则arg1等格式的,原因是还没有关联源代码的问题:关联上:apache-tomcat-7.0.42-src.zip,并重新实现/复写方法就OK;
3.配置Servlet:
HelloKittyServlet,仅仅是一个普通的实现类而已,而我最终要运行在Tomcat服务器中,所以得告诉Tomcat,来帮我管理HelloKittyServlet类;
1.找到项目根下的WEB-INF下的web.xml文件:
2.在根元素web-app中创建一个新的元素节点:servlet
2.在根元素web-app中创建一个新的元素节点:servlet-mapping
Tomcat服务器是Servlet容器:
容器:负责了容器内对象的生命周期维护工作.
创建对象-->初始化操作--->使用对象调用方法-->销毁操作--->GC
4.部署项目/访问项目:
访问方式:http://ip:port/contextPath/资源名
http://localhost:80/day3/hello .
三.Servlet的生命周期
1.javax.servlet.Servlet接口中的方法:
- 1):ServletConfig getServletConfig:获取当前Servlet的配置信息对象.
- 2):String getServletInfo:用于返回当前Servlet的作者,版本,版权信息.
- 3):void init: 初始化方法
- 4):void service: 服务方法
- 5):void destory:销毁方法
2.Servlet的生命周期方法:
- 1)Servlet的构造器: 只在第一次请求Servlet的时候,先执行构造器(创建Servlet对象).
- 2)void init(ServletConfig config):只在第一次请求时,创建好Servlet对象之后,做初始化操作.
- 3)void service(ServletRequest req, ServletResponse res): 每次请求都会执行,处理请求的.
- 4)void destroy():在Tomcat正常关闭的时候执行,不要期望该方法一定会执行,不需要把扫尾的操作存放在这里.
结论:
1.Servlet是单例的.Servlet只要第一次请求之后,创建好一个对象,该对象就一直存在,直到Tomcat关闭.
一个Servlet类,最多之后一个实例.(单实例,性能有保证,但是多个线程并发可能造成安全问题).2.Servlet构造器必须是公共的无参数的.
注意:
Servlet的第一次请求:
执行构造器,init方法,service方法.
Servlet的非第一次请求:
只会执行service方法.
四.Servlet的请求流程
1.浏览器先向Tomcat发出请求:
http://localhost:80/day3/hello.
2.解析请求:
http://:表示当前请求遵循http协议.
localhost:通过DNS解析成IP,从互联网上去寻找该IP的主机.
80: 从该主机中找到80端口对应的程序.
day3: 上下文路径
/hello:资源名称
3.读取Tomcat根/conf中的serverx.xml文件
并获取所有的<Context/>元素(document.getElementsByTagName(..)):
遍历多个<Context/>找到path属性值为day3的<Context>元素.
进而得到当前<Context/>元素的docBase属性值(当前访问的Web项目的根路径):D:\JavaApps\Servlet\webapp.
4.从当前访问Web项目的根路径/WEB-INF中去获取web.xml文件.
获取web.xml中所有的<url-pattern>元素,并判断哪一个<url-pattern>元素的文本内容为/hello.
找不到:响应404.
找 到:根据文本内容为/hello的<url-pattern>找到当前被访问Servlet的全限定名:
String className = "com._520it._01_hello.HelloServlet";
5.判断Servlet的实例缓存池(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.使用反射来创建Servlet对象(调用构造器),并存放于Servlet实例缓存池中,供下次使用.
Servet obj = (Servlet)Class.forName("com._520it._01_hello.HelloServlet").newInstance();
//存放于Servlet实例缓存池中,供下次使用
cacheMap.put("com._520it._01_hello.HelloServlet",obj);
7.Tomcat创建ServletConfig对象,并调用init方法,完成初始化操作.
obj.init(config);
8.Tomcat创建ServletRequest和ServletResponse对象,并调用service方法,完成处理请求的方法.
obj.service(req,resp);
9.应该在service方法中,处理完请求给浏览器做相应操作.
五.Servlet初始化参数
ServletConfig接口:表示Servlet的配置对象,提供了四个方法用于获取当前Servlet的配置信息.
ServletConfig对象是有容器(Tomcat)创建的,我们就只管用.
四个方法:
String getServletName(): 获取<servlet-name>的文本内容.
ServletContext getServletContext():获取当前Servlet的上下文对象,当前应用对象.
String getInitParameter(String paranName):根据指定的初始化参数名称获取对应的初始化参数值.
Enumeration<String> getInitParameterNames():获取所有的初始化参数的名字.
上述代码中出现硬编码,不利于维护.
解决方案:把写死的数据存放到配置文件中.
Servlet的初始化参数机制.
六.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编写总结:
- 1):自定义Servlet类直接继承于HttpServlet即可.
- 2):Servlet需要处理请求,只需要覆盖父类的service方法(参数是Http类型的);
public void service(ServletRequest arg0, ServletResponse arg1) throws ServletException, IOException {
// TODO Auto-generated method stub
//super.service(arg0, arg1);
}
- 3):如果Servlet需要做初始化操作,只需要提供无参数的init方法即可.
七.HttpServletRequest常用方法
1.Request的类型
ServletRequest接口:表示Servlet的请求对象,包含了处理请求的方法.
HttpServletRequest接口:是ServletRequest接口的子接口,支持Http的请求处理.
2.HTTP请求信息
HTTP请求:包含三部分(请求行,请求头,请求实体).
HttpServletRequest中就提供了获取HTTP请求信息的所有的方法.
3.常用方法:
- 1):String getMethod():返回请求方式:如GET/POST
- 2):String getContextPath():返回请求URL所属Web应用的路径。路径以"/"开头
- 3):String getRequestURI():返回请求行中的资源名字部分:如/test/index.html
- 4):StringBuffer getRequestURL():返回浏览器地址栏信息
- 5):String getRemoteAddr():返回发出请求的客户机的IP地址
- 6):String getHeader(String name):根据指定的请求头名称获取对应的请求头的值.
4.获取请求参数(重要)
1.String String getParameter(String name):返回指定名字参数的值。
注意: 请求参数一般说来,是用户在表单中填写的数据.2.String[] getParameterValues(String name):返回指定名字参数的多个参数值。
3.Enumeration<String> getParameterNames():返回所有参数名的Enumeration对象。
4.Map<String,String[]> getParameterMap():返回所有的参数和值所组成的Map对象。
5.ServletConfig:
String getInitParameter(String paranName):获取初始化参数的,
初始化参数是我们自己为了解除硬编码,自己在web.xml中<servlet>元素中配置的.
6.HttpServletRequest:
String getParameter(String paranName):获取请求参数.
请求参数一般说来,是用户在表单中填写的数据.
八.注册案例
九.请求的中文乱码处理
1.请求参数中带有中文会出现乱码问题:
为什么会出现中文乱码问题:
Tomcat的默认编码使用的ISO-8859-1,不支持中文.
原因是:服务端把二进制数据转换为String的时候,使用了ISO-8859-1来编码.
2.解决方案一:
针对于POST和GET方式都起作用.
- 1):先按照ISO-8859-1把乱码的数据恢复成二进制格式(byte数组).
byte [] data= username.getBytes("ISO-8859-1"); - 2):使用UTF-8对二进制数据重新编码.
username = new String(data,"UTF-8");
上述代码没问题,但是如果表单中的数据比较多,那么就的处理N次解码,再编码:
3.解决方案二:
1)针对于POST请求,不包括GET请求:
在获取任意参数之前,设置请求的编码:
req.setCharacterEncoding("UTF-8");
2)针对GET请求:
修改Tomcat中默认的处理GET请求的编码(了解),仅仅只对GET方式有效..
Tomcat根/conf/server.xml,在修改端口的那一行(71行);
建议:以后表单统统使用POST方式来传递.
十.HttpServletResponse常用方法
1.Response的类型
ServletResponse接口:处理一般的响应操作,包含处理响应操作的方法.
HttpServletResponse接口:处理HTPP的响应操作,包含了处理HTTP响应的方法.
HttpServletResponse是ServletResponse的子接口.
2.常用方法:
1)获取输出流对象:
获取字节输出流: OutputStream out = resp.getOutputStream();(文件下载使用)
获取字符输出流: PrintWriter out = resp.getWriter();
** 注意: **resp.getWriter()和resp.getOutputStream(),只能调用一个方法,否则报错.
2) 设置响应时的编码:必须在获取输出流之前设置.
response.setCharacterEncoding("UTF-8");
3)设置响应输出的MIME类型:
response.setContentType("text/html");
注意:上述两行代码可以合体,同时设置响应的MIME类型和编码:
response.setContentType("text/html;charset=utf-8");