1. 简介
Servlet 运行在服务端的Java小程序,是sun公司提供一套规范(接口),用来处理客户端请求、响应给浏览器的动态资源。但servlet的实质就是java代码服务器执行,通过java的API动态的向客户端输出内容 Server Applet Servlet运行于支持Java的应用服务器中。从实现上讲,Servlet可以响应任何类型的请求,但绝大多数情况下Servlet只用来扩展基于HTTP协议的Web服务器
2. Servlet接口和类
- GenericServlet抽象类
它对Servlet接口中的部分方法(init和destroy)添加了实现,使得开发时只需要考虑针对service方法的业务实现即可。
- HttpServlet类
2. 创建方式
2.1. 实现接口Servlet
@WebServlet("/hs2")
public class HelloServlet2 implements Servlet{
@Override
public void destroy() {
}
@Override
public ServletConfig getServletConfig() {
return null;
}
@Override
public String getServletInfo() {
return null;
}
@Override
public void init(ServletConfig arg0) throws ServletException {
}
@Override
public void service(ServletRequest request, ServletResponse response) throws
ServletException, IOException {
System.out.println("OK");
System.out.println("OK");
response.setContentType("text/html;charset=UTF‐8");
response.getWriter().println("我是第二种创建方式");
}
}
2.2. 继承GenericServlet类
@WebServlet(value="/hs3")
public class HelloServlet3 extends GenericServlet{
@Override
public void service(ServletRequest request, ServletResponse response) throws
ServletException, IOException {
System.out.println("OK");
response.setContentType("text/html;charset=UTF‐8");
response.getWriter().println("我是第三种创建方式");
}
}
2.3. 实现接口HttpServlet
@WebServlet("/hs1")
public class HelloServlet extends HttpServlet {
private static final long serialVersionUID = 1L;
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws
ServletException, IOException {
response.getWriter().print("我是Servlet创建的第一种方式");
}
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws
ServletException, IOException {
doGet(request, response);
}
}
3. 配置方式
3.1. 注解
Servlet3.0及以后
@WebServlet("资源路径")
public @interface WebServlet {
String name() default "";//相当于<Servlet-name>
String[] value() default {};//代表urlPatterns()属性配置
String[] urlPatterns() default {};//相当于<url-pattern>
int loadOnStartup() default -1;//相当于<load-on-startup>
WebInitParam[] initParams() default {};
boolean asyncSupported() default false;
String smallIcon() default "";
String largeIcon() default "";
String description() default "";
String displayName() default "";
}
@WebServlet(value="/hello",loadOnStartup=1)
public class HelloServlet extends HttpServlet {
private static final long serialVersionUID = 1L;
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws
ServletException, IOException {
response.getWriter().print("OK");
}
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws
ServletException, IOException {
doGet(request, response);
}
}
3.2. web.xml配置
Servlet所有版本都支持
<?xml version="1.0" encoding="UTF‐8"?>
<web‐app xmlns:xsi="http://www.w3.org/2001/XMLSchema‐instance"
xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/webapp_
3_1.xsd" version="3.1">
<display‐name>Web_Day11</display‐name>
<servlet>
<servlet‐name>hello2</servlet‐name>
<servlet‐class>com.qf.web.servlet.HelloServlet</servlet‐class>
<!‐‐启动的优先级,数字越小越先起作用 ‐‐>
<load‐on‐startup>1</load‐on‐startup>
</servlet>
<!‐‐映射配置 ‐‐>
<servlet‐mapping>
<servlet‐name>hello2</servlet‐name>
<url‐pattern>/hello2</url‐pattern>
</servlet‐mappin>
<welcome‐file‐list>
<welcome‐file>login.html</welcome‐file>
</welcome‐file‐list>
</web‐app>
4. 生命周期
- 被创建:执行init方法,只执行一次,默认情况下,第一次被访问时,Servlet被创建可以配置执行Servlet的创建时机。在<servlet>标签下配置
- 第一次被访问时,创建 <load-on-startup>的值为负数
- 在服务器启动时,创建<load-on-startup>的值为0或正整数
- 在服务器启动时,创建<load-on-startup>的值为0或正整数,Servlet是单例的 多个用户同时访问时,可能存在线程安全问题。解决:尽量不要在Servlet中定义成员变量。 即使定义了成员变量,也不要对修改值,提供服务:每次访问Servlet时,Service方法都会被调用一次。
- 被销毁:执行destroy方法,只执行一次,Servlet被销毁时执行。服务器关闭时,Servlet被销毁只有服务器正常关闭时,才会执行destroy方法。destroy方法在Servlet被销毁之前执行,一般用于释放资源
5. 乱码
5.1. GET中文乱码
在Tomcat7及以下客户端以UTF‐8的编码传输数据到服务器端,而服务器端的request对象使用的是ISO8859‐1这个字符编码来接收数据,服务器和客户端沟通的编码不一致因此才会产生中文乱码的。解决办法:在接收到数据后,先获取request对象以ISO8859‐1字符编码接收到的原始数据的字节数组,然后通过字节数组以指定的编码构建字符串,解决乱码问题。
Tomcat8的版本中GET基本就不会乱码了,因为服务器对url的编码格式可以进行自动转换
Url 256个字节 可以传输多个字节
String name=request.getParameter("name");
name=new String(name.getBytes("ISO8859‐1"),"UTF‐8");
5.2. POST乱码
由于客户端是以UTF‐8字符编码将表单数据传输到服务器端的,因此服务器也需要设置以UTF‐8字符编码进行接收,要想完成此操作,服务器可以直接使用从ServletRequest接口继承而来的"setCharacterEncoding(charset)"方法进行统一的编码设置.
客户端----服务器: request.setCharacterEncoding(charset) 把来自客户端的请求body的编码转化为utf-8
浏览器识别不到返回的中文是什么编码格式,就会默认使用GB2312,如果返回的是UTF‐8格式的那么在浏览器上就会显示乱码的问题
1. 设置缓冲区的编码
response.setCharacterEncoding("UTF-8");
2. 通知浏览器使用utf-8解码
response.setHeader("Content-Type", "text/html;charset=UTF-8");
6. 重定向和转发
6.1. 重定向
重定向就是通过各种方法将各种网络请求重新定个方向转到其它位
客户浏览器发送http请求
web服务器接受后发送302状态码响应及对应新的location给客户浏览器
客户浏览器发现是302响应,则自动再发送一个新的http请求,请求url是新的location地址
服务器根据此请求寻找资源并发送给客户。在这里location可以重定向到任意URL,既然是浏览器重新发出了请求,则就没有什么request传递的概念了。在客户浏览器路径栏显示的是其重定向的路径,客户可以观察到地址的变化的。
特点:
1,重定向是客户端行为。
2,重定向是浏览器做了至少两次的访问请求的。
3,重定向浏览器地址改变。
4,重定向2次跳转之间传输的信息会丢失(request范围)。
5,重定向可以指向任何的资源,包括当前应用程序中的其他资源,同一个站点上的其他应用程序中的资源,其他站点的资源。注意:传递给HttpServletResponse.sendRedirect 方法的相对URL以“/”开头,它是相对于整个WEB站点的根目录
6.2. 转发
客户浏览器发送http请求
web服务器接受此请求
调用内部的一个方法在容器内部完成请求处理和转发动作
将目标资源发送给客户。在这里,转发的路径必须是同一个web容器下的url,其不能转向到其他的web路径上去,中间传递的是自己的容器内的request。在客户浏览器路径栏显示的仍然是其第一次访问的路径,也就是说客户是感觉不到服务器做了转发的
特点:
1,转发是服务器行为
2,转发是浏览器只做了一次访问请求
3,转发浏览器地址不变
4,转发2次跳转之间传输的信息不会丢失,所以可以通过request进行数据的传递
5,转发只能将请求转发给同一个WEB应用中的组件
注意:如果创建RequestDispatcher 对象时指定的相对URL以“/”开头
request.getRequestDispatcher("LoginServlet").forward(rerequest, response);
7. ServletConfig
- config.getServletName(); 获取配置中的servlet名字
<servlet>
<servlet-name>abc</servlet-name>
<servlet-class>com.itheima.servlet.QuickStratServlet</servlet-class>
<init-param>
<param-name>url</param-name>
<param-value>jdbc:mysql:///mydb</param-value>
</init-param>
<load-on-startup>3</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>abc</servlet-name>
<url-pattern>/quickStratServlet</url-pattern>
</servlet-mapping>
2、获得该servlet的初始化的参数
String initParameter = config.getInitParameter("url");
3、获得Servletcontext对象
ServletContext servletContext = config.getServletContext();
String realPath = servletContext.getRealPath("/");