# Servlet 总结 在 Java Web 程序中,**Servlet**主要负责接收用户请求 `HttpServletRequest`,在`doGet()`,`doPost()`中做相应的处理,并将回应`HttpServletResponse`反馈给用户。**Servlet** 可以设置初始化参数,供 Servlet 内部使用。一个 Servlet 类只会有一个实例,在它初始化时调用`init()`方法,销毁时调用`destroy()`方法**。**Servlet 需要在 web.xml 中配置(MyEclipse 中创建 Servlet 会自动配置),**一个 Servlet 可以设置多个 URL 访问**。**Servlet 不是线程安全**,因此要谨慎使用类变量。 ## 阐述 Servlet 和 CGI 的区别? ### CGI 的不足之处 1,需要为每个请求启动一个操作 CGI 程序的系统进程。如果请求频繁,这将会带来很大的开销。 2,需要为每个请求加载和运行一个 CGI 程序,这将带来很大的开销 3,需要重复编写处理网络协议的代码以及编码,这些工作都是非常耗时的。 ### Servlet 的优点 1,只需要启动一个操作系统进程以及加载一个 JVM,大大降低了系统的开销 2,如果多个请求需要做同样处理的时候,这时候只需要加载一个类,这也大大降低了开销 3,所有动态加载的类可以实现对网络协议以及请求解码的共享,大大降低了工作量。 4,Servlet 能直接和 Web 服务器交互,而普通的 CGI 程序不能。Servlet 还能在各个程序之间共享数据,使数据库连接池之类的功能很容易实现。 补充:Sun Microsystems 公司在 1996 年发布 Servlet 技术就是为了和 CGI 进行竞争,Servlet 是一个特殊的 Java 程序,一个基于 Java 的 Web 应用通常包含一个或多个 Servlet 类。Servlet 不能够自行创建并执行,它是在 Servlet 容器中运行的,容器将用户的请求传递给 Servlet 程序,并将 Servlet 的响应回传给用户。通常一个 Servlet 会关联一个或多个 JSP 页面。以前 CGI 经常因为性能开销上的问题被诟病,然而 Fast CGI 早就已经解决了 CGI 效率上的问题,所以面试的时候大可不必信口开河的诟病 CGI,事实上有很多你熟悉的网站都使用了 CGI 技术。 参考:《javaweb 整合开发王者归来》P7 ## Servlet 接口中有哪些方法及 Servlet 生命周期探秘 Servlet 接口定义了 5 个方法,其中**前三个方法与 Servlet 生命周期相关**: - `void init(ServletConfig config) throws ServletException` - `void service(ServletRequest req, ServletResponse resp) throws ServletException, java.io.IOException` - `void destroy()` - `java.lang.String getServletInfo()` - `ServletConfig getServletConfig()` **生命周期:** **Web 容器加载 Servlet 并将其实例化后,Servlet 生命周期开始**,容器运行其**init()方法**进行 Servlet 的初始化;请求到达时调用 Servlet 的**service()方法**,service()方法会根据需要调用与请求对应的**doGet 或 doPost**等方法;当服务器关闭或项目被卸载时服务器会将 Servlet 实例销毁,此时会调用 Servlet 的**destroy()方法**。**init 方法和 destroy 方法只会执行一次,service 方法客户端每次请求 Servlet 都会执行**。Servlet 中有时会用到一些需要初始化与销毁的资源,因此可以把初始化资源的代码放入 init 方法中,销毁资源的代码放入 destroy 方法中,这样就不需要每次处理客户端的请求都要初始化与销毁资源。 参考:《javaweb 整合开发王者归来》P81 ## GET 和 POST 的区别 这个问题在知乎上被讨论的挺火热的,地址: 。 ![](https://static001.geekbang.org/infoq/04/0454a5fff1437c32754f1dfcc3881148.png) GET 和 POST 是 HTTP 协议中两种常用的请求方法,它们在不同的场景和目的下有不同的特点和用法。一般来说,可以从以下几个方面来区分它们: - 语义上的区别:GET 通常用于获取或查询资源,而 POST 通常用于创建或修改资源。GET 请求应该是幂等的,即多次重复执行不会改变资源的状态,而 POST 请求则可能有副作用,即每次执行可能会产生不同的结果或影响资源的状态。 - 格式上的区别:GET 请求的参数通常放在 URL 中,形成查询字符串(querystring),而 POST 请求的参数通常放在请求体(body)中,可以有多种编码格式,如 application/x-www-form-urlencoded、multipart/form-data、application/json 等。GET 请求的 URL 长度受到浏览器和服务器的限制,而 POST 请求的 body 大小则没有明确的限制。 - 缓存上的区别:由于 GET 请求是幂等的,它可以被浏览器或其他中间节点(如代理、网关)缓存起来,以提高性能和效率。而 POST 请求则不适合被缓存,因为它可能有副作用,每次执行可能需要实时的响应。 - 安全性上的区别:GET 请求和 POST 请求都不是绝对安全的,因为 HTTP 协议本身是明文传输的,无论是 URL、header 还是 body 都可能被窃取或篡改。为了保证安全性,必须使用 HTTPS 协议来加密传输数据。不过,在一些场景下,GET 请求相比 POST 请求更容易泄露敏感数据,因为 GET 请求的参数会出现在 URL 中,而 URL 可能会被记录在浏览器历史、服务器日志、代理日志等地方。因此,一般情况下,私密数据传输应该使用 POST + body。 重点搞清了,两者在语义上的区别即可。不过,也有一些项目所有的请求都用 POST,这个并不是固定的,项目组达成共识即可。 ## 什么情况下调用 doGet()和 doPost() Form 标签里的 method 的属性为 get 时调用 doGet(),为 post 时调用 doPost()。 ## 转发(Forward)和重定向(Redirect)的区别 **转发是服务器行为,重定向是客户端行为。** **转发(Forward)** 通过 RequestDispatcher 对象的 forward(HttpServletRequest request,HttpServletResponse response)方法实现的。RequestDispatcher 可以通过 HttpServletRequest 的 getRequestDispatcher()方法获得。例如下面的代码就是跳转到 login_success.jsp 页面。 ```java request.getRequestDispatcher("login_success.jsp").forward(request, response); ``` **重定向(Redirect)** 是利用服务器返回的状态码来实现的。客户端浏览器请求服务器的时候,服务器会返回一个状态码。服务器通过 `HttpServletResponse` 的 `setStatus(int status)` 方法设置状态码。如果服务器返回 301 或者 302,则浏览器会到新的网址重新请求该资源。 1. **从地址栏显示来说** forward 是服务器请求资源,服务器直接访问目标地址的 URL,把那个 URL 的响应内容读取过来,然后把这些内容再发给浏览器.浏览器根本不知道服务器发送的内容从哪里来的,所以它的地址栏还是原来的地址. redirect 是服务端根据逻辑,发送一个状态码,告诉浏览器重新去请求那个地址.所以地址栏显示的是新的 URL. 2. **从数据共享来说** forward:转发页面和转发到的页面可以共享 request 里面的数据. redirect:不能共享数据. 3. **从运用地方来说** forward:一般用于用户登陆的时候,根据角色转发到相应的模块. redirect:一般用于用户注销登陆时返回主页面和跳转到其它的网站等 4. 从效率来说 forward:高. redirect:低. ## 自动刷新(Refresh) 自动刷新不仅可以实现一段时间之后自动跳转到另一个页面,还可以实现一段时间之后自动刷新本页面。Servlet 中通过 HttpServletResponse 对象设置 Header 属性实现自动刷新例如: ```java Response.setHeader("Refresh","5;URL=http://localhost:8080/servlet/example.htm"); ``` 其中 5 为时间,单位为秒。URL 指定就是要跳转的页面(如果设置自己的路径,就会实现每过 5 秒自动刷新本页面一次) ## Servlet 与线程安全 **Servlet 不是线程安全的,多线程并发的读写会导致数据不同步的问题。** 解决的办法是尽量不要定义 name 属性,而是要把 name 变量分别定义在 doGet()和 doPost()方法内。虽然使用 synchronized(name){}语句块可以解决问题,但是会造成线程的等待,不是很科学的办法。 注意:多线程的并发的读写 Servlet 类属性会导致数据不同步。但是如果只是并发地读取属性而不写入,则不存在数据不同步的问题。因此 Servlet 里的只读属性最好定义为 final 类型的。 参考:《javaweb 整合开发王者归来》P92 ## JSP 和 Servlet 是什么关系 其实这个问题在上面已经阐述过了,Servlet 是一个特殊的 Java 程序,它运行于服务器的 JVM 中,能够依靠服务器的支持向浏览器提供显示内容。JSP 本质上是 Servlet 的一种简易形式,JSP 会被服务器处理成一个类似于 Servlet 的 Java 程序,可以简化页面内容的生成。Servlet 和 JSP 最主要的不同点在于,Servlet 的应用逻辑是在 Java 文件中,并且完全从表示层中的 HTML 分离开来。而 JSP 的情况是 Java 和 HTML 可以组合成一个扩展名为.jsp 的文件。有人说,Servlet 就是在 Java 中写 HTML,而 JSP 就是在 HTML 中写 Java 代码,当然这个说法是很片面且不够准确的。JSP 侧重于视图,Servlet 更侧重于控制逻辑,在 MVC 架构模式中,JSP 适合充当视图(view)而 Servlet 适合充当控制器(controller)。 ## JSP 工作原理 JSP 是一种 Servlet,但是与 HttpServlet 的工作方式不太一样。HttpServlet 是先由源代码编译为 class 文件后部署到服务器下,为先编译后部署。而 JSP 则是先部署后编译。JSP 会在客户端第一次请求 JSP 文件时被编译为 HttpJspPage 类(接口 Servlet 的一个子类)。该类会被服务器临时存放在服务器工作目录里面。下面通过实例给大家介绍。 工程 JspLoginDemo 下有一个名为 login.jsp 的 Jsp 文件,把工程第一次部署到服务器上后访问这个 Jsp 文件,我们发现这个目录下多了下图这两个东东。 .class 文件便是 JSP 对应的 Servlet。编译完毕后再运行 class 文件来响应客户端请求。以后客户端访问 login.jsp 的时候,Tomcat 将不再重新编译 JSP 文件,而是直接调用 class 文件来响应客户端请求。 ![JSP工作原理](https://oss.javaguide.cn/github/javaguide/1.jpeg) 由于 JSP 只会在客户端第一次请求的时候被编译 ,因此第一次请求 JSP 时会感觉比较慢,之后就会感觉快很多。如果把服务器保存的 class 文件删除,服务器也会重新编译 JSP。 开发 Web 程序时经常需要修改 JSP。Tomcat 能够自动检测到 JSP 程序的改动。如果检测到 JSP 源代码发生了改动。Tomcat 会在下次客户端请求 JSP 时重新编译 JSP,而不需要重启 Tomcat。这种自动检测功能是默认开启的,检测改动会消耗少量的时间,在部署 Web 应用的时候可以在 web.xml 中将它关掉。 参考:《javaweb 整合开发王者归来》P97 ## JSP 有哪些内置对象、作用分别是什么 [JSP 内置对象 - CSDN 博客](http://blog.csdn.net/qq_34337272/article/details/64310849) JSP 有 9 个内置对象: - request:封装客户端的请求,其中包含来自 GET 或 POST 请求的参数; - response:封装服务器对客户端的响应; - pageContext:通过该对象可以获取其他对象; - session:封装用户会话的对象; - application:封装服务器运行环境的对象; - out:输出服务器响应的输出流对象; - config:Web 应用的配置对象; - page:JSP 页面本身(相当于 Java 程序中的 this); - exception:封装页面抛出异常的对象。 ## Request 对象的主要方法有哪些 - `setAttribute(String name,Object)`:设置名字为 name 的 request 的参数值 - `getAttribute(String name)`:返回由 name 指定的属性值 - `getAttributeNames()`:返回 request 对象所有属性的名字集合,结果是一个枚举的实例 - `getCookies()`:返回客户端的所有 Cookie 对象,结果是一个 Cookie 数组 - `getCharacterEncoding()`:返回请求中的字符编码方式 = getContentLength()`:返回请求的 Body 的长度 - `getHeader(String name)`:获得 HTTP 协议定义的文件头信息 - `getHeaders(String name)`:返回指定名字的 request Header 的所有值,结果是一个枚举的实例 - `getHeaderNames()`:返回所以 request Header 的名字,结果是一个枚举的实例 - `getInputStream()`:返回请求的输入流,用于获得请求中的数据 - `getMethod()`:获得客户端向服务器端传送数据的方法 - `getParameter(String name)`:获得客户端传送给服务器端的有 name 指定的参数值 - `getParameterNames()`:获得客户端传送给服务器端的所有参数的名字,结果是一个枚举的实例 - `getParameterValues(String name)`:获得有 name 指定的参数的所有值 - `getProtocol()`:获取客户端向服务器端传送数据所依据的协议名称 - `getQueryString()`:获得查询字符串 - `getRequestURI()`:获取发出请求字符串的客户端地址 - `getRemoteAddr()`:获取客户端的 IP 地址 - `getRemoteHost()`:获取客户端的名字 - `getSession([Boolean create])`:返回和请求相关 Session - `getServerName()`:获取服务器的名字 - `getServletPath()`:获取客户端所请求的脚本文件的路径 - `getServerPort()`:获取服务器的端口号 - `removeAttribute(String name)`:删除请求中的一个属性 ## request.getAttribute()和 request.getParameter()有何区别 **从获取方向来看:** `getParameter()`是获取 POST/GET 传递的参数值; `getAttribute()`是获取对象容器中的数据值; **从用途来看:** `getParameter()`用于客户端重定向时,即点击了链接或提交按扭时传值用,即用于在用表单或 url 重定向传值时接收数据用。 `getAttribute()` 用于服务器端重定向时,即在 sevlet 中使用了 forward 函数,或 struts 中使用了 mapping.findForward。 getAttribute 只能收到程序用 setAttribute 传过来的值。 另外,可以用 `setAttribute()`,`getAttribute()` 发送接收对象.而 `getParameter()` 显然只能传字符串。 `setAttribute()` 是应用服务器把这个对象放在该页面所对应的一块内存中去,当你的页面服务器重定向到另一个页面时,应用服务器会把这块内存拷贝另一个页面所对应的内存中。这样`getAttribute()`就能取得你所设下的值,当然这种方法可以传对象。session 也一样,只是对象在内存中的生命周期不一样而已。`getParameter()`只是应用服务器在分析你送上来的 request 页面的文本时,取得你设在表单或 url 重定向时的值。 **总结:** `getParameter()`返回的是 String,用于读取提交的表单中的值;(获取之后会根据实际需要转换为自己需要的相应类型,比如整型,日期类型啊等等) `getAttribute()`返回的是 Object,需进行转换,可用`setAttribute()`设置成任意对象,使用很灵活,可随时用 ## include 指令 include 的行为的区别 **include 指令:** JSP 可以通过 include 指令来包含其他文件。被包含的文件可以是 JSP 文件、HTML 文件或文本文件。包含的文件就好像是该 JSP 文件的一部分,会被同时编译执行。 语法格式如下: <%@ include file="文件相对 url 地址" %> i**nclude 动作:** ``动作元素用来包含静态和动态的文件。该动作把指定文件插入正在生成的页面。语法格式如下: ## JSP 九大内置对象,七大动作,三大指令 [JSP 九大内置对象,七大动作,三大指令总结](http://blog.csdn.net/qq_34337272/article/details/64310849) ## 讲解 JSP 中的四种作用域 JSP 中的四种作用域包括 page、request、session 和 application,具体来说: - **page**代表与一个页面相关的对象和属性。 - **request**代表与 Web 客户机发出的一个请求相关的对象和属性。一个请求可能跨越多个页面,涉及多个 Web 组件;需要在页面显示的临时数据可以置于此作用域。 - **session**代表与某个用户与服务器建立的一次会话相关的对象和属性。跟某个用户相关的数据应该放在用户自己的 session 中。 - **application**代表与整个 Web 应用程序相关的对象和属性,它实质上是跨越整个 Web 应用程序,包括多个页面、请求和会话的一个全局作用域。 ## 如何实现 JSP 或 Servlet 的单线程模式 对于 JSP 页面,可以通过 page 指令进行设置。 `<%@page isThreadSafe="false"%>` 对于 Servlet,可以让自定义的 Servlet 实现 SingleThreadModel 标识接口。 说明:如果将 JSP 或 Servlet 设置成单线程工作模式,会导致每个请求创建一个 Servlet 实例,这种实践将导致严重的性能问题(服务器的内存压力很大,还会导致频繁的垃圾回收),所以通常情况下并不会这么做。 ## 实现会话跟踪的技术有哪些 1. **使用 Cookie** 向客户端发送 Cookie ```java Cookie c =new Cookie("name","value"); //创建Cookie c.setMaxAge(60*60*24); //设置最大时效,此处设置的最大时效为一天 response.addCookie(c); //把Cookie放入到HTTP响应中 ``` 从客户端读取 Cookie ```java String name ="name"; Cookie[]cookies =request.getCookies(); if(cookies !=null){ for(int i= 0;i ``` **优点:** Cookie 被禁时可以使用 **缺点:** 所有页面必须是表单提交之后的结果。 4. HttpSession 在所有会话跟踪技术中,HttpSession 对象是最强大也是功能最多的。当一个用户第一次访问某个网站时会自动创建 HttpSession,每个用户可以访问他自己的 HttpSession。可以通过 HttpServletRequest 对象的 getSession 方 法获得 HttpSession,通过 HttpSession 的 setAttribute 方法可以将一个值放在 HttpSession 中,通过调用 HttpSession 对象的 getAttribute 方法,同时传入属性名就可以获取保存在 HttpSession 中的对象。与上面三种方式不同的 是,HttpSession 放在服务器的内存中,因此不要将过大的对象放在里面,即使目前的 Servlet 容器可以在内存将满时将 HttpSession 中的对象移到其他存储设备中,但是这样势必影响性能。添加到 HttpSession 中的值可以是任意 Java 对象,这个对象最好实现了 Serializable 接口,这样 Servlet 容器在必要的时候可以将其序列化到文件中,否则在序列化时就会出现异常。 ## Cookie 和 Session 的区别 Cookie 和 Session 都是用来跟踪浏览器用户身份的会话方式,但是两者的应用场景不太一样。 **Cookie 一般用来保存用户信息** 比如 ① 我们在 Cookie 中保存已经登录过得用户信息,下次访问网站的时候页面可以自动帮你登录的一些基本信息给填了;② 一般的网站都会有保持登录也就是说下次你再访问网站的时候就不需要重新登录了,这是因为用户登录的时候我们可以存放了一个 Token 在 Cookie 中,下次登录的时候只需要根据 Token 值来查找用户即可(为了安全考虑,重新登录一般要将 Token 重写);③ 登录一次网站后访问网站其他页面不需要重新登录。**Session 的主要作用就是通过服务端记录用户的状态。** 典型的场景是购物车,当你要添加商品到购物车的时候,系统不知道是哪个用户操作的,因为 HTTP 协议是无状态的。服务端给特定的用户创建特定的 Session 之后就可以标识这个用户并且跟踪这个用户了。 Cookie 数据保存在客户端(浏览器端),Session 数据保存在服务器端。 Cookie 存储在客户端中,而 Session 存储在服务器上,相对来说 Session 安全性更高。如果使用 Cookie 的一些敏感信息不要写入 Cookie 中,最好能将 Cookie 信息加密然后使用到的时候再去服务器端解密。