JavaWeb之Servlet

简介: JavaWeb之Servlet

1、什么是JavaWeb?

  1. Servlet JavaEE 规范之一。规范就是接口
  2. Servlet JavaWeb 三大组件之一。三大组件分别是:Servlet 程序、Filter 过滤器、Listener 监听器。
  3. Servlet 是运行在服务器上的一个 java 小程序,它可以接收客户端发送过来的请求,并响应数据给客户端。

2、第一个Servlet程序

2.1 实现步骤

实现一个简单的Servlet程序的步骤:

  1. 编写一个类去实现 Servlet 接口或者Servlet接口的子类
  2. 实现 service 方法,处理请求,并响应数据
  3. web.xml 中去配置 servlet 程序的访问地址

2.2 继承Servlet接口并实现service方法

public class HelloServlet implements Servlet {
/**
* service 方法是专门用来处理请求和响应的
* @param servletRequest 客户端的请求
* @param servletResponse 服务器的响应
*/
@Override
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws
ServletException, IOException {
System.out.println("Hello Servlet");
}
}

2.3 配置web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee
http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
version="4.0">
<!-- servlet 标签给 Tomcat 配置 Servlet 程序 -->
<servlet>
<!--servlet-name 标签 Servlet 程序起一个别名(一般是类名) -->
<servlet-name>HelloServlet</servlet-name>
<!--servlet-class 是 Servlet 程序的全类名-->
<servlet-class>com.lyh.servlet.HelloServlet</servlet-class>
</servlet>
<!--servlet-mapping 标签给 servlet 程序配置访问地址-->
<servlet-mapping>
<!--servlet-name 标签的作用是告诉服务器,我当前配置的地址给哪个 Servlet 程序使用-->
<servlet-name>HelloServlet</servlet-name>
<!--url-pattern 标签配置访问地址 <br/>
/ 斜杠在服务器解析的时候,表示地址为:http://ip:port/工程路径 <br/>
/hello 表示地址为:http://ip:port/工程路径/hello <br/>
-->
<url-pattern>/hello</url-pattern>
</servlet-mapping>
</web-app>

3、Servlet的生命周期

  1. 执行 Servlet 构造器方法(只调用一次)
  2. 执行 init 初始化方法(只调用一次)
  3. 执行 service 方法 (每次访问Servlet对应的网页都会调用)
  4. 执行 destroy 销毁方法  (在 web 工程停止的时候调用)
public class HelloServlet implements Servlet {
    public HelloServlet(){
        System.out.println("1. 构造器方法被调用");
    }
    //这里使用了 ServletConfig这个类
    @Override
    public void init(ServletConfig servletConfig) throws ServletException {
        System.out.println("2. 初始化方法被调用");
    }
    /**
     * service是专门用来处理请求和响应的
     * 当需要查询服务器并且不包含敏感数据时,可以使用GET方法。而当需要向服务器发送大量数据或包含敏感数据时,则应使用POST方法。
     */
    @Override
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
        System.out.println("3 service()方法被调用 Hello Servlet");
    }
    @Override
    public void destroy() {
        System.out.println("4. destroy()方法被调用");
    }
}

4、GET 和 POST

4.1概述

       因为 service方法是专门用来处理客户端请求和返回响应的,所以使用该方法需要我们用户根据自己的需求来实现相关的方法完成对不同类型请求(GET/POST)的处理。

4.2GET/POST的选择

       当需要查询服务器并且不包含敏感数据时,可以使用GET方法。而当需要向服务器发送大量数据或包含敏感数据时,则应使用POST方法。

4.3实现对不同请求的处理

       由于我们继承的是Servlet接口,它没有直接对GET和POST这两种请求类型的区分,所以需要我们自己来实现区分并做不同的处理。

public class HelloServlet implements Servlet {
/**
* service 方法是专门用来处理请求和响应的
*/
    @Override
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws
ServletException, IOException {
        System.out.println("3 service === Hello Servlet 被访问了");
        // 类型转换(因为它有 getMethod()方法)
        HttpServletRequest httpServletRequest = (HttpServletRequest) servletRequest;
        // 获取请求的方式
        String method = httpServletRequest.getMethod();
        if ("GET".equals(method)) {
            doGet();
        } else if ("POST".equals(method)) {
            doPost();
        }
    }
        /**
        * 做 get 请求的操作
        */
        public void doGet(){
            System.out.println("get 请求");
        }
        /**
        * 做 post 请求的操作
        */
        public void doPost(){
            System.out.println("post 请求");
        }
}

5、继承HttpServlet

       通过上面的案例我们可以看到,直接继承Servlet接口很不方便,因为它毕竟只是一个接口,只对方法进行了规范并没有实现,我们看看Servlet接口的源码:

5.1 Servlet接口源码

public interface Servlet {
    void init(ServletConfig var1) throws ServletException;
    ServletConfig getServletConfig();
    void service(ServletRequest var1, ServletResponse var2) throws ServletException, IOException;
    String getServletInfo();
    void destroy();
}

      实际开发中我们很少直接继承Servlet接口去实现servlet程序 而是继承它的子类HttpServlet,因为HttpServlet这个抽象子类对Servlet的方法都进行了实现而且对GET/POST请求也添加了专门的方法去实现。

5.2 doGet/doPost

       HttpServlet为GET/POST这两种客户端请求封装了两个方法,使得我们实现自己的业务逻辑更加简单明了。

public class HelloServlet2 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("HelloServlet2 的 doGet 方法");
    }
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("HelloServlet2 的 doPost 方法");
    }
}

每一个Servlet类都需要配置web.xml:

<servlet>
<servlet-name>HelloServlet2</servlet-name>
<servlet-class>com.lyh.servlet.HelloServlet2</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>HelloServlet2</servlet-name>
<url-pattern>/hello2</url-pattern>
</servlet-mapping>

5.3 HttpServlet继承体系

6、ServletConfig

6.1 概述及源码

  • ServletConfig Servlet 程序的配置信息类。
  • Servlet 程序和 ServletConfig 对象都是由 Tomcat 负责创建,我们负责使用。
  • Servlet 程序默认是第一次访问的时候创建,ServletConfig 是每个 Servlet 程序创建时,就创建一个对应的 ServletConfig 对象。
public interface ServletConfig {
    String getServletName();
    ServletContext getServletContext();
    String getInitParameter(String var1);
    Enumeration<String> getInitParameterNames();
}

通过源码,我们可以了解到它的三大作用:

6.2 ServletConfig 类的三大作用

  1. 可以获取 Servlet 程序的别名 servlet-name 的值
  2. 获取初始化参数 init-param
  1. 获取初始化参数可以帮助我们在运行时配置 Servlet 的行为,例如设置数据库连接信息、数据源名称等参数。
  1. 获取 ServletContext 对象(Servlet的上下文对象)

6.3、ServletConfig 的使用

6.3.1、初始化参数<init-param>

需要在我们的Servlet程序的配置信息中添加

<servlet>
        <!--servlet-name 标签 Servlet 程序起一个别名(一般是类名) -->
        <servlet-name>HelloServlet</servlet-name>
        <!--servlet-class 是 Servlet 程序的全类名-->
        <servlet-class>com.lyh.servlet.HelloServlet</servlet-class>
        <!--   初始化参数 可以是多个参数  -->
        <init-param>
            <!--  参数名-->
            <param-name>url</param-name>
            <!--  参数值-->
            <param-value>jdbc:mysql://localhost:3306/test</param-value>
        </init-param>
        <init-param>
            <param-name>username</param-name>
            <param-value>root</param-value>
        </init-param>
    </servlet>
public class HelloServlet implements Servlet {
    //继承了HttpServlet的类在重写init时 一定要调用父类的初始化方法
    super.init(config);
    //这里使用了 ServletConfig这个类
    @Override
    public void init(ServletConfig servletConfig) throws ServletException {
        // 1、可以获取 Servlet 程序的别名 servlet-name 的值
        System.out.println("HelloServlet 程序的别名是:" + servletConfig.getServletName());
        // 2、获取初始化参数 init-param
        System.out.println("初始化参数 username 的值是;" + servletConfig.getInitParameter("username"));
        System.out.println("初始化参数 url 的值是;" + servletConfig.getInitParameter("url"));
        // 3、获取 ServletContext 对象 
        ServletContext context = servletConfig.getServletContext();
    }
}

注意:

继承了HttpServlet的类在重写init时 一定要调用父类的初始化方法

super.init(config);

7、ServletContext

7.1 概述

  1. ServletContext 是一个接口,它表示 Servlet 上下文对象。
  2. 一个 web 工程,只有一个 ServletContext 对象实例。
  3. ServletContext 对象是一个域对象,我们可以通过它来获取整个web工程中在web.xml中配置的所有被<context-param>标记的参数(不管web工程中有多少个Servlet程序)。
  1. ServletContext 是在 web 工程部署启动的时候创建。在 web 工程停止的时候销毁。

7.2 ServletContext的四大作用

  1. 获取 web.xml 中配置的上下文参数 context-param
  2. 获取当前的工程路径,格式: /工程路径
  3. 获取工程部署后在服务器硬盘上的绝对路径
  4. Map 一样存取数据

7.3 ServletContext的使用

7.3.1 配置web.xml

<!--context-param 是上下文参数(它属于整个 web 工程)-->
<context-param>
<param-name>username</param-name>
<param-value>context</param-value>
</context-param>
<!--context-param 是上下文参数(它属于整个 web 工程)-->
<context-param>
<param-name>password</param-name>
<param-value>root</param-value>
</context-param>

7.3.2 读取工程文件的配置参数以及工程路径

读取配置工程文件web.xml参数用 getInitParameter() 方法。

/**
 * <init-param>不同于ServletConfig的标签<context-param>
 * <init-param>的作用域是它自己对应的Servlet程序
 * <context-param>的作用域是每个Servlet程序
 */
public class ContextServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //1.获取配置文件中的<context-param>参数
        ServletContext context = getServletContext();
        String username = context.getInitParameter("username");
        System.out.println("context-param 参数 username 的值是:" + username);
        //2.获取当前工程路径 /JavaWebStudy_war_exploded
        System.out.println("当前工程路径 ="+context.getContextPath());
        //获取工程部署在服务器上的绝对路径 D:\IdeaProjects\JavaWebStudy\target\JavaWebStudy-1.0-SNAPSHOT\
        //斜杠 / 被服务器解析为 http://127.0.0.1:8080/JavaWebStudy/ 映射到idea的web目录
        System.out.println("工程部署的绝对路径是 ="+context.getRealPath("/"));
        System.out.println("工程下 css 目录的绝对路径是:" + context.getRealPath("/css"));
    }
}

7.3.3 多个Servle程序之间的存取数据

读取数据用 getAttribute() 方法 。

       因为ServletContext的作用域是整个web工程,所以不同Servlet程序之间也可以共享参数以及自己配置参数供其他Servlet读取。

例:写两个Servlet类,Servlet1负责设置参数数据,Servlet2负责读取参数数据。

public class ContextServlet1 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        ServletContext context = getServletContext();
        context.setAttribute("username","root");
        context.setAttribute("password","123456");
    }
}
public class ContextServlet2 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        System.out.println(getServletContext().getAttribute("username"));
        System.out.println(getServletContext().getAttribute("password"));
    }
}

注意:

这两个不一样

  1. 读取配置工程文件web.xml参数用 getInitParameter() 方法
  2. 读取数据用 getAttribute() 方法

8、HttpServletRequest

8.1 概述

我们可以看到service()方法里有两个参数,分别对应客户端的请求以及我们服务端的响应。

@Override
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws
ServletException, IOException {
    //用户业务逻辑代码。。。
}

   每次只要有请求进入 Tomcat 服务器, Tomcat 服务器就会把请求过来的 HTTP 协议信息解析好封装到 Request 对象中。 然后传递到 service 方法(doGet 和 doPost )中给我们使用。我们可以通过 HttpServletRequest 对象,获取到所有请求的信息。

8.2 常用方法

  • getRequestURI()         获取请求的资源路径
  • getRequestURL()       获取请求的统一资源定位符(绝对路径)
  • getRemoteHost()         获取客户端的 ip 地址
  • getHeader()                 获取请求头
  • getParameter()                 获取请求的参数
  • getParameterValues()         获取请求的参数(多个值的时候使用)
  • getMethod()                         获取请求的方式 GET 或 POST
  • setAttribute(key, value);         设置域数据
  • getAttribute(key);                 获取域数据
  • getRequestDispatcher()         获取请求转发对象

8.3、实例

通过html中的表单提交数据,通过Servlet程序读取表单中的数据。

8.3.1、html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<form action="http://localhost:8080/JavaWebStudy_war_exploded/request1" method="get">
  用户名:<input type="text" name="username"><br/>
  密码:<input type="text" name="password"><br/>
  爱好:<input type="checkbox" name="hobby" value="Java">Java
        <input type="checkbox" name="hobby" value="JS">JavaScript
        <input type="checkbox" name="hobby" value="Scala">Scala<br/>
    <input type="submit">
</form>
</body>
</html>

8.3.2、Servlet类

Servlet只需要实现doGet方法即可。

注意:

1.当参数不只一个的时候(比如多选框),我们需要用的是getParameterValues方法,它会返回一个数组对象。使用getParameter来读取多选框的话,最终只会返回一个值(用户最后选的值,先选的值会被覆盖)。

2.中文乱码问题:我们需要设置请求体的字符集编码格式,设置编码格式的代码只有放到doGet/doPost方法下的第一行才有效。

 @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //设置请求体的字符集为 UTF-8 解决post的请求乱码问题
        //必须放在获取请求参数getParameter()方法之前调用才有效果
        req.setCharacterEncoding("UTF-8");
        System.out.println("URI => "+req.getRequestURI());
        System.out.println("URL => "+req.getRequestURL());
        System.out.println("客户端Host的ip => "+req.getRemoteHost());
//        我的真实ip 10.49.xx.xxx 可通过网络管理查看ipv4地址 就是本地真实地址
        System.out.println("请求头 => "+req.getHeader("User-Agent"));
        System.out.println("请求的方法 => "+req.getMethod());
//        参数名称就是 <input>标签中的 name属性
        String username = req.getParameter("username");
        String password = req.getParameter("password");
        String[] hobby = req.getParameterValues("hobby");
        System.out.println("请求的参数 username => "+username);
        System.out.println("请求的参数 password => "+password);
        System.out.print("请求的参数 hobby => ");
        //打印数组的两种方式 1. Arrays.toString  Arrays.asList(数组名)
        System.out.println(Arrays.toString(hobby));
//        System.out.println(Arrays.asList(hobby));
    }

8.4、请求转发

8.4.1 特点:

  1. 浏览器地址没有变化
  2. 是一次请求
  3. 共享request域的数据
  4. 可以转发到WEB-INF目录下的资源
  5. 不可以访问外部资源(比如www.baidu.com)

核心代码:

req.getRequestDispatcher("转发路径").forward(req,resp);

实例:客户端请求 /forwardc的资源, 由ForwardC这个Servlet程序转发到

http://localhost:8080/JavaWebStudy_war_exploded/a/b/c.html

需要再我们上面的html中添加超链接(我们的ForwardC对应的工程url为"/forwardc")。

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <a href="a/b/c.html">跳转到c.html</a></br>
    <a href="http://localhost:8080/JavaWebStudy_war_exploded/forwardC">通过请求转发跳转到c.html</a></br>
    <a href="http://localhost:8080/JavaWebStudy_war_exploded/redirect">请求重定向到c.html</a>
</body>
</html>
public class ForwardC extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("经过了ForwardC请求转发到了 c.html");
        //转发到http://localhost:8080/JavaWebStudy_war_exploded + /a/b/c.html
        //这里的斜杠 / 是协议//ip:端口号/工程路径的意思
        req.getRequestDispatcher("/a/b/c.html").forward(req,resp);
    }
}

c.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>c页面</title>
<!--  有了base标签 浏览器就不会根据当前的浏览器地址的路径去判断 ../../后的路径了  -->
  <base href="http://localhost:8080/JavaWebStudy_war_exploded/a/b/">
</head>
<body>
C页面
<a href="../../index.html">回到首页</a></br>
</body>
</html>

9、HttpServletResponse

9.1 HttpServletResponse 类的作用

HttpServletResponse 类和 HttpServletRequest 类一样。每次请求进来, Tomcat 服务器都会创建一个 Response 对象传 递给 Servlet 程序去使用。 HttpServletRequest 表示请求过来的信息, HttpServletResponse 表示所有响应的信息, 我们如果需要设置返回给客户端的信息,都可以通过 HttpServletResponse 对象来进行设置。

9.1.1 两个输出流的说明。

  • 字节流   getOutputStream()  常用于下载(传递二进制数据)
  • 字符流  getWriter();  常用于回传字符串(常用)

两个流同时只能使用一个。

使用了字节流,就不能再使用字符流,反之亦然,否则就会报错。

public class ResponseIOServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //回传中文字符串同样会乱码,需要设置编码格式
        //1.1这里设置的是我们服务器的编码格式 如果客户的浏览器不是UTF-8仍然是乱码
//        resp.setCharacterEncoding("UTF-8");
        //1.2通过响应头来设置浏览器编码
//        resp.setHeader("Content-Type","text/html; charset=UTF-8");
        //2.设置相应内容的格式 这种方法可以同时设置服务器和浏览器的编码格式
        resp.setContentType("text/html; charset=UTF-8");
        /**
         * 字符流和字节流只能用一个
         */
        //字符流 回传字符串 展示到html上
        PrintWriter writer = resp.getWriter();
        writer.write("Hello Response");
    }
}

9.2 请求重定向

9.2.1 特点:

  1. 浏览器地址会发生变化
  2. 两次请求
  3. 不共享request域中的数据
  4. 不能访问WEB-INF目录下的资源
  5. 可以访问外部资源(比如www.baidu.com)

9.2.2 实现

/**
 * 请求重定向
 */
public class RedirectServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("Servlet程序接收到用户请求");
        //设置重定向的两种方法
        //设置响应状态码 302 ,表示重定向,(已搬迁)
        resp.setStatus(302);
        //1. 设置响应头,说明 新的地址在哪里 不推荐
       // resp.setHeader("Location", "http://localhost:8080");
        //2.第二种方法 推荐使用
        resp.sendRedirect("http://localhost:8080/JavaWebStudy_war_exploded/a/b/c.html");
    }
}


相关文章
|
6月前
|
缓存 Java 应用服务中间件
javaweb实训第三天上午——Servlet(1)
Servlet基础 课程介绍 Servlet基础; (掌握) 基本结构 生命周期 执行流程 接收参数 响应数据
84 0
|
2月前
|
Java 应用服务中间件 Maven
JavaWeb基础5——HTTP,Tomcat&Servlet
JavaWeb技术栈、HTTP、get和post区别、响应状态码、请求响应格数据式、IDEA使用Tomcat、报错解决、Servlet的体系结构、IDEA使用模板创建Servlet
JavaWeb基础5——HTTP,Tomcat&Servlet
|
5月前
|
SQL druid Java
javaweb案例实训之基于jsp和servlet的用户管理开发[增删改查及登录注销]
javaweb案例实训之基于jsp和servlet的用户管理开发[增删改查及登录注销]
34 0
|
5月前
|
SQL 数据可视化 数据库
基于jsp+servlet的javaweb实现最基本的用户注册登陆注销功能
基于jsp+servlet的javaweb实现最基本的用户注册登陆注销功能
23 0
|
6月前
|
存储 安全 JavaScript
JavaWeb之Servlet(下)
本文主要介绍了JavaWeb开发中关于请求、响应、会话和Servlet3.0注解的相关知识。首先讲解了请求(Request)的常用方法,以及处理请求乱码问题的几种方式。接着讨论了响应(Response)的方法,,以及设置全局初始化参数。然后介绍了会话(Session)的概念,包括其生命周期、如何设置和获取初始化参数,并对比了全局初始化参数和Servlet3.0的`@WebServlet`注解。文中给出了具体的代码示例和测试结果,帮助读者更好地理解和掌握这些概念。
|
6月前
|
Java 应用服务中间件 程序员
JavaWeb之Servlet(上)
本篇博文介绍了JavaWeb中Servlet的基本概念和应用。Servlet是用于交互式浏览和修改数据、生成动态Web内容的Java程序,通常运行在支持Java的应用服务器上。文章讲解了Servlet的工作模式,包括客户端请求、服务器调用Servlet以及响应返回客户端的过程。此外,还提到了Servlet API,特别是doGet和doPost方法用于处理请求和响应。作者通过创建第一个Servlet的步骤,展示了如何实现Servlet接口并配置web.xml文件。最后,文章简述了Servlet的工作原理和生命周期,强调了Servlet的单例特性以及初始化、服务和销毁的过程。
|
6月前
|
存储 前端开发 Java
JavaWeb:servlet+jsp+mybatis商品管理增删改查
商品管理通常包括增加(添加)、删除、修改和查询商品信息
172 1
JavaWeb:servlet+jsp+mybatis商品管理增删改查
|
6月前
|
安全 Java
javaweb实训第四天下午——员工管理系统-JSP&Servlet&JDBC综合练习-CRUD
1.课程介绍 Servlet细节; (掌握) 员工信息相关的CRUD; (掌握) 部门信息相关的CRUD; (掌握) 2.Servlet细节 2.1.多种匹配方式
94 0
|
6月前
|
编解码 应用服务中间件 数据库
|
3月前
|
缓存 安全 Java
Java服务器端技术:Servlet与JSP的集成与扩展
Java服务器端技术:Servlet与JSP的集成与扩展
33 3