============================================================
一、什么是Servlet?(非常非常重要)
servlet 是运行在 Web 服务器中的小型 Java 程序(即:服务器端的小应用程序)。
servlet 作用:通常通过 HTTP(超文本传输协议) 接收和响应来自 Web 客户端的请求。
Servlet也是一个java类,但比较特殊,不需要new,自动就可以运行。也有创建、垃圾回收和销毁过程。
1.1、编写一个servlet程序
a、写一个java类,实现Servlet接口,实现了该接口,该类就不是普通的类了。
b、修改 web.xml 文件,给servlet提供(映射)一个可访问的URI地。
c、部署应用到tomcat服务器。
d、测试:http://localhost:8080/day08_00_servlet/demo1 在MyEcplise控制台输出 hello servlet,初体验成功!
e、执行流程:demo1 --> servletDemo1 --> com.itheima.servlet.ServletDemo1 -->
Tomcat安装目录/webroot/Demo1/WEB-INF/classes/com.itheima.servlet.ServletDemo1.class --> service() --> hello servlet
============================================================
============================================================
实例化-->初始化-->服务->销毁
出生:(实例化-->初始化)第一次访问Servlet就出生了(默认情况下)。(说明我可以改变它的出生时机,比如让servlet在服务器启动时就创建)
活着:(服务)应用活着,servlet就活着。(Servlet是单实例,其中的service()方法是多线程的。)
死亡:(销毁)应用被卸载了,servlet就销毁了。
如何在网页界面下卸载应用呢?
1.先访问http://localhost:8080/
2.再访问http://localhost:8080/,点击 Manager App 输入账户名和密码,登录管理界面成功后就可以开启、停止、重新加载、卸载你的应用了,如下图所示:
小知识:
如何设置让servlet在服务器启动时就创建,数值越小越先启动 (不建议写0)。
GenericServlet类(抽象类)已经实现了Servlet接口,我们用哪个方法就调用哪方法,不用再去全部实现了,哪怕是全部的空实现。
模板是HttpServlet类中的私有service()方法(开发中常用方式)
HttpServlet类(抽象类)继承了GenericServlet类(抽象类)
// 注意: 不要重写父类Httpservlet中的service()方法,因为该方法会调用Httpservlet中私有的service()方法,
// 而该私有的service()方法会根据不同的提交表单的方式,会调用相应的doGet()、doPost()、doHead()等方法。
// 我们去重写这些不同的表单提交的方法即可。
Servlet(接口) --> GenericServlet(抽象类/适配器) --> HttpServlet(功能抽象类/模板) --> (继承HttpServlet,具体实现类,重写其功能)
爷爷 爸爸 儿子 孙子
-----------------------------------------------------------------------------
ServletDemo4.java中我们新建的文件是Servlet,不是Class了,里面的配置如下:
该方式可以自动生成web.xml文件,不用我们去手动配置了!
这种方式生成的文件包含无用的注释和标签太多了,如何变得小清新呢?即:修改Servlet的默认模板代码。
小技巧:使自动生成的servlet文件更清新一些(太多的标签和注释了):
找到:MyEclipse\Common\plugins目录
把 com.genuitec.eclipse.wizards_9.0.0.me201108091322.jar 复制到上面目录。
详细过程链接:https://cloud.tencent.com/developer/article/1353287
servet的映射细节:
servet的映射细节1:
一个servlet可以配置多个映射路径(或者说:多个映射路径可以指向同一个servlet),只要是name是一样的就行。
servet的映射细节2:
通配符* 代表任意字符串
url-pattern:*.xxx 以.xxx结尾的请求都可以访问 注:不要加/
url-pattern:/* 以任意字符串结尾的请求都可以访问
url-pattern:/xxxx/* 以/xxxx/开头的请求都可以访问
servet的映射细节3:
匹配规则:
优先级:从高到低
绝对匹配 优先 /开头匹配 优先 扩展名方式(后缀名)匹配
如果url-pattern的值是/,表示的是执行默认映射。所有资源都是经过servlet。
整个生命周期是单实例,但每次访问时为多线程。
法一:单实例变成多实例,但过时了,因为耗费资源多,服务器压力大。
法二:加线程锁,但数据会重复出现(没有同步机制),且运行效率低。
法三:解决线程安全问题的最佳办法:不要写全局变量,而写局部变量(即改变变量的作用域)。
ServletConfig对象的使用
作用1:可以获取servlet配置文件的信息,即web.xml的信息。
作用2:可以获取ServletContext对象。
方式1:重写HttpServlet父类GenericServlet类(抽象类)的int()方法,ServletConfig对象由服务器创建。
方式2:通过调用HttpServlet父类GenericServlet类(抽象类)的方法得到ServletConfig对象,因为GenericServlet类(抽象类)实现了Servlet接口和ServletConfig接口。
方式3:原理同方式2。
ServletContext: 代表的是整个应用。一个应用只有一个ServletContext对象。单实例。
作用:
即在一定范围内(当前应用),使多个Servlet通过ServletContext对象实现数据共享。因为当前应用中会有很多java类,很多Servlet。
常用方法:
public void setAttribute(String name, Object value) // 向ServletContext对象的map集合中添加数据
public Object getAttribute(String name) // 从ServletContext对象的map集合中取出数据
public void rmoveAttribute(String name) // 根据name移除数据
修改web.xml文件:
方法:public String getInitParamenter(String name) // 根据配置文件中的key得到value
方法:public String getRealPath(String path)// 根据资源名称得到资源的绝对路径,可以得到当前应用任何位置的任何资源。
方法:public RequestDispatcher getRequestDispatcher(String path) //参数表示要跳转到哪去
、
如果是通过参数传过来的对象,就叫依赖。
通过方法得到的对象,就叫关联。