三、Servlet
(一) 概念相关
Servlet: Servlet Applet,官方定义是运行在服务器端的小程序。Servlet是用Java编写的服务器端程序,主要运行在服务器端,由服务器端调用,专门用于开发动态WEB资源的技术。本质也是java类,和http协议紧密相连,可以处理http协议相关的所有内容。该java类和普通的java类不同,它必须依赖于服务器才能运行,没有main方法,由Tomcat识别,而普通的java类需要实现特定的规范规则才能被Tomcat识别调用执行,这种规范规则就是Servlet.
java类实现servlet就需要实现servlet规范,实现servlet规范就需要继承httpservlet类,该类需要jar包支持,jar包是放在tomcat里的,将tomcat集成到环境中即可使用。
满足servlet规范,只是让我们的java类能够满足接收请求的要求,接收到请求后需要对请求进行分析,进行业务逻辑处理,计算结果。通过自己定义一个类,实现servlet接口,复写方法,就可以被tomcat识别.
【总结】 Servlet就是一个接口,定义了java类被浏览器访问到tomcat的规则。
【作用/功能】 其主要功能在于交互式地浏览和修改数据,生成动态Web内容。
动态页面:
不是指含有动画的页面,而是指通过执行Servlet,Jsp等程序即时生成客户端网页代码的网页。
动态页面的特点:
- 并不是存储在服务器上独立的网页文件。只有当用户发出请求时,服务器才动态组装并返回一个完整的网页
- 动态网页的内容往往存放在数据库中,根据用户的不同请求来提供个性化的网页内容
(二) 执行原理
执行原理:
1.当服务器接收到客户端的浏览器请求后,会解析请求的url路径,获取访问的Servlet的资源路径
2.查找web.xml文件,是否有对应的 < url-pattern > 标签体内容
3.如果有,则在找到对应的< servlet-class >全类名
4.Tomcat会将字节码文件加载进内存,并创建对象
5.调用其方法
以上基于xml来说明,如果是通过WebServlet("")注解的方式来实现服务器根据解析得到的路径逐层调用。
(三) 生命周期
Servlet没有mian()方法,不能独立运行,它的运行完全由Servlet引擎来控制和调度。所谓的生命周期,指的是Servlet容器何时创建servlet实例,何时调用其方法进行请求的处理,何时销毁整个实例的过程。
package cn.itcast.web.servlet;
import javax.servlet.*;
import java.io.IOException;
//servlet方法
public class ServletDemo02 implements Servlet {
// 初始化方法,在servlet被创建时执行且只会执行一次,说明内存中只有一个servlet对象,是单例的
@Override
public void init(ServletConfig servletConfig) throws ServletException {
}
@Override
public ServletConfig getServletConfig() {
return null;
}
//提供服务的方法,每一次servlet被访问都会被执行,可以执行多次
@Override
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
}
@Override
public String getServletInfo() {
return null;
}
//销毁方法,在服务器正常关闭时执行,仅执行一次
@Override
public void destroy() {
}
}
1. 被创建(加载资源):执行init方法,只执行一次
Q: 什么时候被创建?
A: 默认情况下,第一次被访问的时候被创建或者配置执行servlet的创建时机
Servlet的init方法,只执行一次,说明一个Servlet在内存中只存在一个对象,Servlet是单例的。当存在多个用户访问同一个Servlet时,可能存在线程安全问题。解决方法尽量不要在Servlet中定义成员变量,即使定义了也不要赋值。
2. 被调用执行(处理请求):执行service方法,执行多次
3. 被销毁(释放资源):执行destroy方法,只执行一次,只有服务器正常关闭时才会执行。
(四)Servlet的体系结构
Servlet – 接口
|
GenericServlet – 抽象类
|
HttpServlet – 抽象类
GenericServlet:将Servlet接口中其它方法做了默认实现,只将Service()方法作为抽象,将来定义Servlet类时,可以继承GenericServlet实现service()方法即可
HttpServlet:对http协议的一种封装,简化操作,定义类继承HttpServlet,复写doGet()和doPost()方法。
(五)Request对象
1.Request对象原理
Request对象有由服务器创建,我们仅仅是使用。Request对象作用就是用来获取请求消息。
2.Request继承体系结构
ServletRequest -接口
|继承
HttpServletRequest-接口
|继承
org.apache.catalina.connector.RequestFacade@7fcd419f-类(Tomcat)
3.Request功能
(1)获取请求消息功能
1. 获取请求行数据
- 获取请求方式:GET
- (重要)获取虚拟目录:String getContextPath()
- 获取Servlet路径
- 获取get方法请求参数
- (重要)获取请求URL:String getRequestURI()和StringBuffer getRequestURL()
- 获取协议及版本
- 获取客户机的IP地址
2. 获取请求头数据
- (重要)String getHeader(String name):通过请求头的名称获取请求头的值
package cn.itcast.web.request;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet("/RequestDemo03")
public class RequestDemo03 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 演示获取请求头数据:user-agent
String agent = request.getHeader("user-agent");
if (agent.contains("Chrome")){
System.out.println("谷歌浏览器");
}else if(agent.contains("Firefox")){
System.out.println("火狐来了");
}
}
}
- Enumeration< String > getHeaderNames()
package cn.itcast.web.request;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.Enumeration;
@WebServlet("/RequestDemo02")
public class RequestDemo02 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//演示获取请求头数据
// 获取所有请求头名称
Enumeration<String> headerNames = request.getHeaderNames();
// 遍历
while(headerNames.hasMoreElements()){
String name = headerNames.nextElement();
// 根据名称获取请求头的值
String value = request.getHeader(name);
System.out.println(name+":"+value);
}
}
}
3. 获取请求体数据
请求体:
只有POST请求方式,才有请求体,在请求体中封装了POST请求的请求参数
步骤:
1.获取流对象(包括字符流和字节流,普通表单都是字符流数据,文件视频等都是字节流数据)
1.BufferReader getReader():获取字符输入流,只能操作字符数据
2.ServletStream getInputStream():获取字节输入流,可以操作所有类型数据
2.对获取的流对象进行处理
(2)其它功能(重要)
1. 获取请求参数的通用方式
package com.xxx.resquest;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet("/ser01")
/*请求乱码问题
* tomcat8以上的版本
* get请求不会乱码
* post会乱码,原因:默认编码格式为iso-8859-1,解决方法:设置服务器解析编码格式,要在获取之前设置
* */
public class Encode extends HttpServlet {
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 设置编码格式,该方法只能解决post乱码
// request.setCharacterEncoding("UTF-8");
// String name = request.getParameter("name");
// String sspwd = request.getParameter("pwd");
// System.out.println("姓名:"+name);
// System.out.println("密码:"+sspwd);
// 该方法针对任意请求处理乱码,和tomcat版本有关,如果是7,get没有问题,tomcat8的get因为原始是不乱码使用该方法反而会乱码
String getname = new String(request.getParameter("name").getBytes("ISO-8859-1"),"UTF-8");
System.out.println("name:"+getname);
}
}
2. 请求转发:一种在服务器内部进行资源跳转的方式
步骤:
1.通过request对象获取请求转发器对象:RequestDispatcher getRequestDispatcher(String path)
2.使用RequestDispatcher对象来进行转发:forward(ServletRequest request ,ServletResponse response)
package cn.itcast.web.request;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet("/RequestDemo05")
public class RequestDemo05 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("RequestDemo05");
request.getRequestDispatcher("/RequestDemo06").forward(request,response);
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
}
package cn.itcast.web.request;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet("/RequestDemo06")
public class RequestDemo06 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("RequestDemo06");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
}
【特点】:
- 浏览器地址栏路径不发生改变
- 只能转发到当前服务器内部资源中
- 转发是一次请求
3. 共享数据
在请求转发时不同的Servlet之间需要进行数据通信,也就是共享数据
- 域对象:一个有作用范围的对象,可以在范围内共享数据
- request对象:代表一次请求的范围,一般用于请求转发的多个资源中共享数据
方法:
1.void setAttribute(String name,Object obj):存储数据
1.void setAttribute(String name,Object obj):通过键获取值
1.void setAttribute(String name,Object obj):通过键移除键值对
package cn.itcast.web.request;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet("/RequestDemo05")
public class RequestDemo05 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("RequestDemo05");
request.setAttribute("msg","hello");
request.getRequestDispatcher("/RequestDemo06").forward(request,response);
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
}
package cn.itcast.web.request;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet("/RequestDemo06")
public class RequestDemo06 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("RequestDemo06");
Object msg = request.getAttribute("msg");
System.out.println(msg);
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
}
4. 获取ServletContext
- 概念:代表整个WEB应用,可以和程序的容器通信
- 获取方式:
①通过request对象获取,request.getServletContext()
②通过HttpServlet获取,this.getServletContext() - 功能:
①获取MIME类型
MIME类型:在互联网通信过程中定义一种文件数据类型,格式:大类型/小类型
②域对象:共享数据
方法:
setAttribute(String name, Object value)
getAttribute(String name)
removeAttribute(String name)
Servletcontext对象的范围:所有用户的所有请求数据
③获取文件的真实(服务器)路径
方法:String getRealPath(String path)
(五)Response对象
1.Respnse功能:设置响应消息
1. 设置响应行
1.格式:HTTP/1.1 200 OK
2.设置状态码:setStatus(int sc)
2. 设置响应头: setHeader(String name,String value)
3. 设置响应体:
步骤:
1.获取输出流
字符输出流:PrintWriter getWriter()
字节输出流:ServletOutputStream getOutputStream()
2.使用输出流,将客户端输出到客户端浏览器
2.重定向
package cn.itcast.web.response;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet("/responsedemo1")
public class responsedemo1 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("demo1");
// 方法一
// response.setStatus(302);
// response.setHeader("location","/servlet/responsedemo2");
// 方法二
response.sendRedirect("/servlet/responsedemo2");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
}
package cn.itcast.web.response;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet( "/responsedemo2")
public class responsedemo2 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("demo2ssssss");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
}
【特点】:
- 转发地址栏发生变化
- 重定向可以访问其它站点(服务器)的资源
- 重定向是俩次请求,不能使用request对象来共享数据
3.服务器输出字符和字节流
package cn.itcast.web.response;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
@WebServlet( "/responsedemo3")
public class responsedemo3 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//获取流对象之前,设置流的默认编码:iso-8859-1 设置为gbk
// response.setCharacterEncoding("GBK");
// 或者
// response.setHeader("content-type","text/html;charset=utf-8");
// 简单形式
response.setContentType("text/html;charset=utf-8");
// 1. 获取字符输出流
PrintWriter pw = response.getWriter();
// 2. 输出数据
pw.write("你好");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
}
package cn.itcast.web.response;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@WebServlet(name = "responsedemo4")
public class responsedemo4 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 1.获取字节输出流
ServletOutputStream sos = response.getOutputStream();
// 2.输出数据,hello是字符串,要用getBytes()来转换为字节(数组),可以在getBytes()设置解码格式
// 通过setContentType("text/html;charset=utf-8")解决即可
sos.write("hello".getBytes());
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
}
相关:
Servlet笔记整理(一)