JavaWeb(一)Servlet详解

一、Servlet是什么?
   Servlet是用Java语言编写的程序,它运行在Web服务器上,用以处理来自客户端的请求,并返回处理的结果,在客户端形成动态网页
优点:
  1、具有Java的所有优点,能够跨平台;
  2、运行在Web服务器上,可扩展服务器的功能;
  3、可以和其他资源交互,从而返回客户端相应的内容,形成动态网页;
  4、Servlet很安全,访问Servlet的唯一途径是通过服务器;
  5、只要符合Servlet规范的Servlet容器都可以运行Servlet
在这里插入图片描述

二、Tomcat和Servlet的关系
  Tomcat既是Web服务器软件,同时也是符合规范的Servlet/JSP容器。Tomcat作为Servlet容器,负责处理客户端的请求,把请求传给Servlet,并将Servlet的响应返回给客户端。而Servlet是一种运行在支持Java语言服务器上的组件。Servlet最常见的用途是扩展Java Web服务器功能,提供非常安全的,可移植的,易于使用的CGI替代品.
  从http协议中的请求和响应可以得知,浏览器发出的请求是一个请求文本,而浏览器接收到的也应该是一个响应文本。但是在上面这个图中,并不知道是如何转变的,只知道浏览器发送过来的请求也就是request,我们响应回去的就用response。忽略了其中的细节,现在就来探究一下。
在这里插入图片描述
1、 Tomcat将http请求文本接收并解析,然后封装成HttpServletRequest类型的request对象,所有的HTTP头数据读可以通过request对象调用对应的方法查询到。
2、 Tomcat同时会要响应的信息封装为HttpServletResponse类型的response对象,通过设置response属性就可以控制要输出到浏览器的内容,然后将response交给tomcat,tomcat就会将其变成响应文本的格式发送给浏览器

  • Java Servlet API 是Servlet容器(tomcat)和servlet之间的接口,它定义了serlvet的各种方法,还定义了Servlet容器传送给Servlet的对象类,其中最重要的就是ServletRequest和ServletResponse。所以说我们在编写servlet时,需要实现Servlet接口,按照其规范进行操作。

三、编写Servlet

3.1 新建Web Project
在这里插入图片描述
1、 添加项目名称,Next新建一个与src同级的source folder为resources,用来放项目的配置文件;
2、 Next,项目名可以与Web Content root不同;勾选Generate web.xml deployment descriptor,将在WEB-INF下生成web.xml,web.xml就是web project的入口
在这里插入图片描述
注:该项目需要绑定服务器程序才能在网站上访问,jar包需要添加到Web项目的lib包下

3.2 创建Servlet

  • 手动创建:创建一个MyServlet继承HttpServlet,重写doGet和doPost方法,也就是看请求的方式是get还是post,然后用不同的处理方式来处理请求,在web.xml中配置MyServlet,为什么需要配置?让浏览器发出的请求知道到达哪个servlet,也就是让tomcat将封装好的request找到对应的servlet让其使用。
    在这里插入图片描述
     按照步骤,首先浏览器通过http://localhost:8080/test01/MyServlet来找到web.xml中的url-pattern,然后,匹配到了url-pattern后,就会知道servlet-name的MyServlet,再通过servlet-name寻找就能够知道servlet的位置了,最后根据servlet的对应方式进行处理。
  • 利用向导新建Servlet
    1、添加Servlet
    在这里插入图片描述
    2、右击项目,在new选项中有直接新建servlet的选项
    在这里插入图片描述
    3、配置xml文件(已通过注解方法替代)
    在这里插入图片描述
    4、配置MyServlet类中的信息,重写doPost()和doGet()方法
    在这里插入图片描述
    5、测试是否能进行接受请求和响应
    在这里插入图片描述
     在浏览器中输入对应的网站跟路径和servlet url
    在这里插入图片描述
    在这里插入图片描述

四、创建Servlet的原理
 1、Servlet的生命周期:Servlet容器负责调度并控制Servlet的生命周期
 服务器启动时(web.xml中配置load-on-startup=xxx,默认为0),如果该元素的值为负数或者没有设置,则容器会当Servlet被请求时再加载。如果值为正整数或者0时,表示容器在应用启动时就加载并初始化这个servlet,值越小,servlet的优先级越高,就越先被加载。值相同时,容器就会自己选择顺序来加载。

 图中的三个方法为Servlet生命周期的关键方法
在这里插入图片描述

 2、创建Sevlet是继承父类HttpServlet,而不直接实现Servlet接口的原因:
    在这里插入图片描述

实现和继承结构图:HttpServlet继承GenericServlet,GenericServlet(通用Servlet)就是实现Servlet接口的方法,简化编写servlet的步骤。具体下面详解
在这里插入图片描述 GenericServlet的继承结构,实现了Servlet接口和ServletConfig接口
在这里插入图片描述
  Servlet接口内容
    在这里插入图片描述
  从这里可以看到,Servlet生命周期的三个关键方法,init、service、destroy。还有另外两个方法,一个getServletConfig()方法来获取ServletConfig对象,ServletConfig对象可以获取到Servlet的一些信息,ServletName、ServletContext、InitParameter、InitParameterNames、通过查看ServletConfig这个接口就可以知道
  ServletConfig接口内容
    在这里插入图片描述
  其中ServletContext对象是servlet上下文对象,功能有很多,获得了ServletContext对象,就能获取大部分我们需要的信息,比如获取servlet的路径。
  到此,已经知道了Servlet接口中的内容和作用,总结起来就是,三个生命周期运行的方法,获取ServletConfig,而通过ServletConfig又可以获取到ServletContext。而GenericServlet实现了Servlet接口后,也就说明我们可以直接继承GenericServlet,就可以使用上面我们所介绍Servlet接口中的那几个方法了,能拿到ServletConfig,也可以拿到ServletContext,不过那样太麻烦,不能直接获取ServletContext,所以GenericServlet除了实现Servlet接口外,还实现了ServletConfig接口,那样,就可以直接获取ServletContext了。
  GenericServlet类的内容详解
    在这里插入图片描述
  看上图,用红色框框起来的就是实现Servlet和ServletConfig接口所实现的方法,有9个,数目刚好,但是我们可以发现,init方法有两个,一个是带有参数ServletConfig的,一个是无参的方法,为什么这样设计?这里需要知道其中做了什么事情,来看看这两个方法分别做了什么事?

private transient ServletConfig config;

public void init(ServletConfig config) throws ServletExection{
	this.config = config;
	this.init();
}
public void init() throws ServletExection{
	//等待重写
}
public ServletConfig getServletConfig() {
	return config;
}

  首先看init(ServletConfig config)方法,因为只有init(ServletConfig config)中带有ServletConfig对象,为了方便能够在其他地方也能直接使用ServletConfig对象,而不仅仅局限在init(ServletConfig config)方法中,所以创建一个私有的成员变量config,在init(ServletConfig config)方法中就将其赋值给config,然后通过getServletConfig()方法就能够获取ServletConfig对象了,这个可以理解,但是在init(ServletConfig config)中,还调用了一个init()方法
  这是为什么呢?这个原因是为了防止一件事情,当我们需要在init方法中做一点别的事情,我们想到的方法就是继承GenericServlet并且重写了init(ServletConfig config)方法,这样依赖,就破坏了原本在GenericServlet类中init(ServletConfig config)写的代码了,也就是在GenericServlet类中的成员变量config会一直是null,无法得到赋值,因为被重写了,就不会在执行GenericServlet中init(ServletConfig config)方法中的代码。要想赋值,就必须在重写的init(ServletConfig config)方法中调用父类的init(ServletConfig config)方法,也就是super.init(ServletConfig config),这样一来,就很不方便,怕有时候会忘了写这句代码,所以在GenericServlet类中增加一个init()方法,以后需要在init方法中需要初始化别的数据,只需要重写init()这个方法,而不需要去覆盖init(ServletConfig config)这个方法,这样设计,就好很多,不用在管init(ServletConfig config)这个其中的内容了,也不用出现其他的问题。
  GenericServlet并没有实现Servlet接口中的service(servletRequest req,ServletResponse rse)方法
在这里插入图片描述
  servletReques和ServletResponse都没有与http挂钩,所以出现了HttpServlet继承GenericServlet
HttpServlet类详解:
  继承了GenericServlet类,通过我们上面的推测,这个类主要的功能肯定是实现service方法的各种细节和设计。并且通过类名可以知道,该类就跟http挂钩了。 
    在这里插入图片描述
  关注service(HttpServletRequest req, HttpServletResponse resp)方法和service(ServletRequest req, ServletResponse res)方法。
  service(ServletRequest req, ServletResponse res)方法
在这里插入图片描述
  该方法中就做一件事情,就是将ServletRequest和ServletResponse这两个对象向上转型为HttpServletRequest和HttpServletResponse对象。为什么能这样转?
  首先要知道req、res是什么类型,通过打印System.out.println(req),可以知道,req实际的类型是org.apache.catalina.connector.RequestFacade
  Tomcat中的源码:
在这里插入图片描述在这里插入图片描述
  通过图可以得知,req的继承结构:RequestFacade、httpServletRequest、ServletRequest,我们知道本身req是ServletRequest,那么从继承结构上看,它也可以看成HttpServletRequest,也可以看成ServletRequest,所以强转为HttpServletRequest是可以的
  重点:这个方法就是判断浏览器过来的请求方式是哪种,并将req和res转为http类型,每种的处理方式不一样,我们常用的就是get,post,并且,我们处理的方式可能有很多的内容,所以,在该方法内会将get,post等其他5种请求方式提取出来,变成单个的方法,然后我们需要编写servlet时,就可以直接重写doGet或者doPost方法就行了,而不是重写service方法,更加有针对性。 所以这里就回到了我们上面编写servlet时的情况,继承httpServlet,而只要重写两个方法,一个doGet,一个doPost,其实就是service方法会调用这两个方法中的一个(看请求方式)

五、重点对象
 ServletConfig、ServletContext,request、response

  1、ServletConfig对象
  获取途径: getServletConfig();
  功能: 获取以下属性
   在这里插入图片描述

getServletName(); 获取servlet的名称,也就是我们在web.xml中配置的servlet-name
getServletContext(); 获取ServletContext对象,该对象的作用看下面讲解
getInitParameter(String); 获取在servlet中初始化参数的值。这里注意与下文的全局初始化参数的区分。这个获取的只是在该servlet下的初始化参数
getInitParameterNames(); 获取在Servlet中所有初始化参数的名字,也就是key值,可以通过key值,来找到各个初始化参数的value值。注意返回的是枚举类型

  通过注解设置局部变量

在这里插入图片描述在这里插入图片描述
在这里插入图片描述

  注意: 在上面我们所分析的源码过程中,我们就知道,其实可以不用先获得ServletConfig,然后在获取其各种参数,可以直接使用其方法,比如上面我们用的ServletConfig().getServletName();可以直接写成getServletName();而不用在先获取ServletConfig();了,原因就是在GenericServlet中,已经帮我们获取了这些数据,我们只需要直接拿就行。
  2、ServletContext对象
  获取途径: getServletContext(); 、getServletConfig().getServletContext();这两种获取方式的区别就跟上面的解释一样,第一种是直接拿,在GenericServlet中已经帮我们getServletConfig().getServletContext();拿到了ServletContext。我们只需要直接获取就行了,第二种就相当于我们自己在获取一遍,两种都是一样的。
  功能: tomcat为每个web项目都创建一个ServletContext实例,tomcat在启动时创建,服务器关闭时销毁,在一个web项目中共享数据,管理web项目资源,为整个web配置公共信息等,通俗点讲,就是一个web项目,就存在一个ServletContext实例,每个Servlet读可以访问到它。
  1、web项目中共享数据,getAttribute(String name)、setAttribute(String name, Object obj)、removeAttribute(String name)

setAttribute(String name, Object obj) 在web项目范围内存放内容,以便让在web项目中所有的servlet读能访问到
getAttribute(String name) 通过指定名称获得内容
removeAttribute(String name) 通过指定名称移除内容
在这里插入图片描述
访问localhost查看console
在这里插入图片描述
  2、整个web项目初始化参数 这个就是全局初始化参数,每个Servlet中都能获取到该初始化值,与上文的getServletConfig.getInitParameter(String)区分
getInitPatameter(String name)  通过指定名称获取初始化值
getInitParameterNames()  获得枚举类型
web.xml 配置 整个web项目的初始化
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
  3、获取web项目资源
    3.1获取web项目下指定资源的路径:getServletContext().getRealPath("/WEB-INF/web.xml")
在这里插入图片描述
在这里插入图片描述
    3.2获取web项目下指定资源的内容,返回的是字节输入流:
InputStream getResourceAsStream(java.lang.String path)
在这里插入图片描述
  部分内容:
在这里插入图片描述
  3、request对象
  功能: request就是将请求文本封装而成的对象,所以通过request能获得请求文本中的所有内容,请求头、请求体、请求行 。
  1、 请求行内容的获取
在这里插入图片描述在这里插入图片描述
  2、 请求头的获取
String getHeader(java.lang.String name) 获得指定头内容String【】
long getDateHeader(java.lang.String name) 获得指定头内容Date
int getIntHeader(java.lang.String name) 获得指定头内容int
Enumeration getHeaders(java.lang.String name) 获得指定名称所有内容
  3、 请求体的获取 – 请求参数的获取
  分两种:get请求,post请求
get请求参数:http://localhost:8080/test/MyServlet?username=jack&password=1234
post请求参数: <form method="post"><input type="text" name="username"> //默认是post请求
String request.getParameter(String) 获得指定名称,一个请求参数值。
String[] request.getParameterValues(String) 获得指定名称,所有请求参数值。例如:checkbox、select等
Map<String , String[]> request.getParameterMap() 获得所有的请求参数 
  4、 请求转发
  转发后跳转的页面,这里不管用不用"/"开头,都是以web项目根开始,因为这是请求转发,请求转发只局限与在同一个web项目下使用,所以这里一直都是从web项目根下开始的
在这里插入图片描述

  特点: 浏览器中url不会改变,也就是浏览器不知道服务器做了什么,是服务器帮我们跳转页面的,并且在转发后的页面,能够继续使用原先的request,因为是原先的request,所以request域中的属性都可以继续获取到。
  4、response对象
在这里插入图片描述
常用的一个方法:response.setHeader(java.lang.String name, java.lang.String value) 设置指定的头,一般常用,例如:设置每隔3秒就自动刷新一次,可观察到网页端时间在变化
在这里插入图片描述
重定向(页面跳转)
方式一:手动方案(也是重定向的原理
 response.setStatus(302);  //状态码302就代表重定向
  response.setHeader(“location”,“http://www.baidu.com”);

方式二:通过response.sendRedirect(“http://www.baidu.com”);
注意:response.sendRedirect(path);
第一种:response.sendRedirect("/test/MyServlet");  使用了"/"开头,说明是从web站点根开始
第二种:response.sendRedirect(“MyServlet”);  从web项目根开始
  重定向没有任何局限,可以重定向web项目内的任何路径,也可以访问别的web项目中的路径,并且这里就用"/“区分开来,如果使用了”/“开头,就说明我要重新开始定位了,不访问刚才的web项目,自己写项目名,如果没有使用”/"开始,那么就知道是访问刚才那个web项目下的servlet,就可以省略项目名了。就是这样来区别。
  特点: 服务器告诉浏览器要跳转的页面,是浏览器主动去跳转的页面,浏览器知道,所以浏览器的地址栏中url会变,是浏览器重新发起一个请求到另外一个页面,所以request是重新发起的,跟请求转发不一样,不能再访问request的参数。

注: 重定向时浏览器的行为,转发则是服务器的行为,所以重定向大大加大了带宽,但是转发也加大了服务器压力,对于可能有些人分不清跳转的url该怎么写,这里说明:绝对路径一般用在跳转其他网站,相对路径尽量避免使用,根路径则大多数情况都适用。

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值