(JavaEE-04)Servlet

Servlet

  • Servlet是sun公司提供的一套专门用于开发动态web资源的技术

  • sun公司在ServletAPI中提供了一个servlet接口,如果需我们需要开发一个动态web资源,需要两步:

  • 编写一个Java类,实现servlet接口

  • 把开发好的Java类部署到web服务器中

demo:使用servlet给浏览器输出“Hello World!~”

  • 首先查看ServletAPI,Servlet开发属于JavaEE技术,在JavaSEAPI中是没有的,需要查看JavaEEAPI,但是JavaEEAPI太过繁多,建议直接查看ServletAPI

  • 根据API可以得知Servlet具体释义以及相关接口和类(GenericServlet, HttpServlet)

  • 在webapps下新建一个web应用,搭建必要的目录结构与文件(WEB-INF、classes、lib、web.xml)

  • 在classes文件夹下新建一个Java源文件(也可以不在这里),文件名叫HelloServlet.java(可以不是这个名字)

  • 继承GenericServlet接口,重写service方法,在service方法中使用response对象编写给客户端发送数据的程序(获取输出流)

  • 添加import和package语句

  • 对该文件进行编译,由于有package,要使用 -d 来进行编译,而且需要使用JavaEE的类,所以需要JavaEE的Jar文件

  • 使用set classpath来设置类文件路径(set classpath=%classpath%;serlvet-api.jar文件的目录)

  • javac -d . HelloServlet.java(进行编译)

  • 将这个Servlet配置到web容器中(配置web.xml,添加servlet元素与servlet-mapping元素)

  • 启动服务器,访问Servlet

使用UML描述Servlet的调用过程

Servlet的生命周期

Servlet会在第一次被访问的时候创建出相应的Servlet对象,为了方便后续请求的访问,Web容器创建一个Servlet对象后,会一直 将其保存在容器中,以后关于该Servlet的请求,都使用该对象处理,一直不会被销毁,直到web容器关闭。也就是说,在一个Servlet运行的过程 中,web容器中有且只有一个该Servlet对象存在,多个请求使用多线程来分别处理,一个请求对应一个线程,多个请求多个线程(Tomcat会给每个 线程创建对应的request对象和response对象,一起交给Servlet对象),这些线程共享Servlet对象!可以得出一个结论:Servlet是非线程安全的,但是request对象和response对象是线程安全的

Servlet的生命周期演示(init方法与destroy方法)

使用Eclipse开发Servlet

  • 怎样创建一个web项目

  • 给Eclipse配置Tomcat服务器(Tomcat使用的虚拟机与web应用编译器的版本问题)

  • 导入web开发的相关jar文件

  • Eclipse帮助我们都干了哪些事情

HttpServlet

HttpServlet指得是能够处理HTTP请求的serlvet,它在原有的Servlet接口上添加了一些HTTP协议处理的方法,它更加强大,在开发中通常继承这个类。
HttpServlet在实现Servlet接口时,重写了service方法,该方法内部会自动判断用户的请求方式,比如是get请求,则会调用HttpServlet的doGet方法,post请求,调用doPost方法。
在开发中,直接覆盖doGet和doPost方法,不用覆盖service方法。

阅读HttpServlet文档,查看HttpServlet源码(ctrl+shift+T)
使用Eclipse创建一个新的Servlet,继承HttpServlet,直接创建Servlet,自动生成配置(一个Servlet可以配置多个映射)

Servlet细节总结

  1. 1**Servlet如果想要外部访问,必须把Servlet程序映射到一个URL地址上,在web.xml中使用`<servlet>`元素和`<servlet-mapping>`元素完成**
    2>`<servlet>`元素用于注册Servlet,它包含两个子元素:`<servlet-name>`和`<servlet-class>`,分别用于注册Servlet的注册名称和Servlet的完整类名
    3>`<servlet-mapping>`元素用于映射一个已经注册的Servet对外访问路径,它包含两个子元素:`<servlet-name>`和`<url-pattern>`,分别用于指定注册名和对外访问路径
  2. 用一个Servlet可以被映射到多个URL上

    1>在Servlet映射到的URL,可以使用通配符来配置,但是只能有**两种固定的格式**:
    2    >   * ` “ * . 扩展名”,如: *.do,*.html(伪静态)`
    3    >   *  `"/* ",如:/action/*`
  3. 使用了通配符后,就会产生一些新的问题,如下:

    1* Servlet1 映射到 /abc/*
    2* Servlet2 映射到 /*
    3* Servlet3 映射到 /abc
    4* Servlet4 映射到 *.do

    那么当这样通配后,对于相似的URL请求会怎么去处理呢?举例来说明:

    *号开头的优先级最低!不以*开头的话,哪个最像选择哪个

  • /abc/a.html,/abc/* 和 /* 都匹配,Servlet引擎将会调用Servlet1

  • /abc,/abc/* 和 /abc 都匹配,Servlet引擎将会调用Servlet3

  • /abc/a.do,/abc/* 和 *.do 都匹配,Serlvet引擎将会调用Servlet1

  • /a.do,/* 和 *.do 都匹配,Servlet引擎将会调用Servlet2

  • /xxx/yyy/a.do,/* 和 *.do 都匹配,Servlet引擎将会调用Servlet2

Servlet是不能独立运行的,它的运行完全是由Servlet引擎来控制和调度的。一个Servlet如果被访问, 不管访问多少次,Web容器中之后一个Servlet对象,被多个请求(线程)共享,但是每次执行service方法,都是根据这次请求重新创建的请求对 象和响应对象。当本次请求完成,请求对象和响应对象都会被销毁(响应对象的销毁在创建了标准的Http响应之后)。

<serlvet>元素可以配置随着web容器的启动而创建该Servlet对象。使用<load-on-startup>元素来完成,在创建过程中,就会调用Servlet的init方法。如果有一些操作需要在服务 器启动的时候就完成,就可以使用这种方式来完成(初始化数据库连接池等)。

如果某个Servlet的映射路径为 “/“,那么这个Servlet就是当前web应用的默认Servlet。凡是在web.xml中找不到映射的URL,它们的访问请求都将交给这个默认的 Servlet处理,也就是说,默认的Servlet用于处理所有其它Servlet都不处理的请求。

1>其实对于Tomcat来说,我们所有的请求都不能直接到达web资源,都会经过一个Servlet。如:http://localhost:8080/testweb/1.html ,如果项目根目录有一个 1.html 文件
2>是可以请求的到的,但其实,并不是直接访问了该资源,Tomcat有一个默认的Servlet,由这个默认的Servlet读取了这个资源,然后返回给客户端。如果没有,就是404
3>如果你自定义了默认的Servlet,将会覆盖掉系统的默认Servlet,所以不建议这么做。(查看Tomcat的web.xml,其实所有的静态资源,都由该Servlet处理)

线程安全

1>Servlet存在线程安全问题,在实际开发中,要根据具体情况来编写解决同步的代码
2>在Servlet中编写程序时,要注意对象的静态属性的处理,不然会引发内容溢出的问题(对象的静态集合属性处理)
3>标准解决方案:同步代码块。非标准解决方案:SingleThreadModel(已经被废弃)

对于Servlet的线程安全问题,的确是一个比较灵活的问题,那么有以下几条开发建议,可以避免Servlet的线程安全问题:

  1. 尽量避免使用成员变量,如果万不得已使用了,就需要同步,但是注意同步可用性最小的代码路径

  2. 要清楚request是线程安全的,HttpSession,ServletContext都不会线程安全的

  3. 使用同步的集合类

  4. 不要在Servlet中创建自己的线程来完成某个功能(增加了复杂度)

  5. 在多个servlet中对外部对象(比如文件)进行修改操作时,一定要加锁,做到互斥的访问效果

ServletConfig对象

  • 在Servlet的配置文件中,可以使用一个或多个<init-param>标签为servlet配置一些初始化参数

  • 当配置了初始化参数后,在web容器在创建servlet实例对象时,会自动将这些初始化参数封装到ServletConfig对象中,并在调 用servlet的init方法时,将ServletConfig对象传递 servlet。进而,在程序中通过ServletConfig对象就可以得到当前servlet的初始化参数信息。

  • ServletConfig一般应用于帮助Servlet存储一些固定信息,如:字符编码、数据库连接、获取配置文件等

  • 如果使用的Servlet继承了HttpServlet,由于在GenericServlet中实现了对于Config对象的处理,所以在自己的Servlet中,直接使用getConfig对象就可以了。

ServletContext对象(重点)

  • Web容器在启动时,它会为每个Web应用都创建一个对应的ServletContext对象,它代表当前web应用。

  • 通过ServletConfig对象是可以获取ServletContext对象的(如果继承了HttpServlet,可以直接使用getServletContext()来获取ServletContext对象)

  • 一个Web应用只有一个ServletContext,所以这个对象将会被所有的Servlet所共享,通过这个特性,我们可以利用ServletContext让多个Servlet进行数据传递

  • 要明白Servlet对象是由web服务器来掌管生死,千万不要自己创建Servlet对象来进行传值

  • ServletContext对象通常被称之为Context域对象

  • 查看ServletContext API文档

  • ServletContext初始化参数

  • 获取Servlet规范的版本号,获取文件的MIME类型表示

  • ServletContext可以做请求转发,将一个Servlet的请求转发给另一个,一般用于(MVC设计模式)

MVC简单介绍

到目前为止,对于客户端的请求,我们都是使用对应的Servlet来做简单处理,但是问题是,一般的,不可能就简单在返回一个字符串或者在控制台打 印一个语句,应该给客户返回一个html页面,那么这个html页面哪里来?如果这个html是个静态资源,那么很简单,但是我们开发的是动态web资 源,该怎么办呢?其实很简单,使用Servlet的response对象发送html的字符流给客户端。这种方法表面上看起来是没有什么问题,但是实际操 作你就会发现,这个做法相当麻烦,而且开发效率很低,应该专门有一种特殊能力的Servlet来处理html的绘制,这种具有特殊能力的Servlet就 是JSP。

MVC简单介绍

JSP其实就是一个特殊的Servlet,它能很好的处理html的问题,使得页面的渲染与程序逻辑的处理可以得以分离。(demo,在Servlet中传值并转发给一个JSP)

在Servlet中获取各种资源文件(重点)

在实际开发中,很多时候需要获取服务器上的一些别的资源来帮助开发,这些资源的获取方式都不太一样。

从web目录中读取资源文件

如果在开发中,需要的文件存储在WebContent(项目目录)中,那就使用ServletContext来读取,相关的方法有:

  • getRealPath(String path),返回一个虚拟路径对应资源的真实路径

  • getResource(String path),返回一个虚拟路径对应资源的URL对象

  • getRescourceAsStream(String path),返回一个虚拟路径对应资源的输入流

  • getResoucePaths(String path),返回一个虚拟路径下的所有对应资源的集合

这些方法中的参数都不应该是资源的绝对路径或相对路径,应该是一个针对当前应用的虚拟路径,应该以 “/” 开头,这个 “/” 就表示web应用所在目录,也成为项目根目录,然后按照资源在应用的目录结构来指定资源的虚拟路径。

从源代码src(类路径)文件夹中读取资源文件

在很多时候,我们的资源文件并不在Web资源目录中,而是在类目录中,这个时候怎么读取呢?

  • 使用类加载器来获取类路径中的资源:getSystemResource(String path),getResource(String path)

  • 使用类本身来获取类路径中的资源:getResource(String path)

  • 使用ServletContext获取项目根目录,然后按照项目层级去读取文件(WEB-INF/classes/a.txt)

  • 根据类加载器来获取资源时,不要添加 “/“,类加载器会从该加载器的类路径根目录来查找 path 指定的资源,加了 “/“,反而找不到。

  • 使用类本身来获取资源时,加 “/” ,意味着从类路径的根目录来查找,不加 “/“,意味着,从当前类所在的文件夹来查找资源

  • web应用和本地方法运行的效果是不一样的,有关于System的资源获取,在web应用中是不起作用的

  • 还有 getResourceAsStream(String path)系列的方法,跟上述的 getResource(String path)方法特点是一致的,只不过一个是返回URL对象,一个是返回流对象

  • 使用类加载器的 getResourceAsStream(String path) 方法来获取资源的输入流时,不能实现动态读取文件变更,应该用类加载器来获取路径,然后用传统的读取文件的方式再次读取文件,这样就能实现实时更新数据了

  • 如果资源文件过大,不要使用类加载器的方式来直接获取,因为这种方式会直接把资源全部加载进内存,容易内存溢出,使用类加载器获取地址,用传统方式的流然后缓冲读取

  • 关于System与no-System的方法有什么区别,我已经总结好了,如果需要进一步了解,参看官方关于Resources的解释

在客户端缓存Servlet的输出

对于不经常变化的数据,在Servlet中可以为其设置合理的缓存时间值,以避免浏览器频繁得向服务器发送请求,提升服务器的性能。

1String data = "aaaaaaaaaaaaaaaaaaaaaa";
2long time = System.currentTimeMillis() + 1 * 24 * 60 * 60 * 1000;
3System.out.println("hehe");
4response.setDateHeader("expires", time);
5response.getWriter().write(data);


转载于:https://my.oschina.net/u/1048681/blog/288412

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值