JavaWeb之Servlet

news/2024/11/15 1:17:54/

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这个类@Overridepublic void init(ServletConfig servletConfig) throws ServletException {System.out.println("2. 初始化方法被调用");}/*** service是专门用来处理请求和响应的* 当需要查询服务器并且不包含敏感数据时,可以使用GET方法。而当需要向服务器发送大量数据或包含敏感数据时,则应使用POST方法。*/@Overridepublic void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {System.out.println("3 service()方法被调用 Hello Servlet");}@Overridepublic 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 方法是专门用来处理请求和响应的
*/@Overridepublic 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 {@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {System.out.println("HelloServlet2 的 doGet 方法");}@Overrideprotected 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 的行为,例如设置数据库连接信息、数据源名称等参数。
  3. 获取 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这个类@Overridepublic void init(ServletConfig servletConfig) throws ServletException {// 1、可以获取 Servlet 程序的别名 servlet-name 的值System.out.println("HelloServlet 程序的别名是:" + servletConfig.getServletName());// 2、获取初始化参数 init-paramSystem.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程序)。
  4. 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 {@Overrideprotected 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_explodedSystem.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 {@Overrideprotected 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 {@Overrideprotected 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()方法里有两个参数,分别对应客户端的请求以及我们服务端的响应。

@Overridepublic 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方法下的第一行才有效。

 @Overrideprotected 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 {@Overrideprotected 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 {@Overrideprotected 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 {@Overrideprotected 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");}
}


http://www.ppmy.cn/news/240212.html

相关文章

Flutter入门之Dart中的并发编程、异步和事件驱动详解

Flutter入门之Dart中的并发编程、异步和事件驱动详解 在Dart语言中&#xff0c;开发者可以使用并发编程、异步和事件驱动等机制来提高程序的性能和响应能力。下面将分别对这三个方面进行详细介绍。 并发编程 Dart中的并发编程主要使用的是Isolate的概念。Isolate是指在Dart虚…

Spark RDD计算总分与平均分

文章目录 一&#xff0c;提出任务二&#xff0c;实现思路三&#xff0c;准备工作1、启动HDFS服务2、启动Spark服务3、在本地创建成绩文件4、将成绩文件上传到HDFS 四&#xff0c;完成任务1、在Spark Shell里完成任务&#xff08;1&#xff09;读取成绩文件&#xff0c;生成RDD&…

华为mate8电池虚电校正_vivo手机虚电量如何进行电池校正

在我们使用的智能手机中&#xff0c;电池是一个非常重要的手机部件&#xff0c;如果在使用手机的过程中没有正确的给手机充电&#xff0c;或者让手机电池受到损害的话&#xff0c;手机的电池就会出现虚电量这样的情况&#xff0c;很多用户面对手机虚电量都是束手无策的&#xf…

擦地机器人排行榜_扫地机排名 扫地机器人排行榜

展开全部 在万千型号不一的扫地机器人中能够脱颖而出的产品&#xff0c;一定是32313133353236313431303231363533e78988e69d8331333433633437综合配置和使用体验都很出色的明星机型。今天就一起了解这款优质产品——斐纳TOMEFON智能扫地机器人。对于一台适用于家庭地面清洁的高…

tws蓝牙耳机哪个牌子好?2022蓝牙耳机排行榜

一、南卡lite pro2蓝牙耳机 NANK南卡的出品工艺以及理念一直是行业中的领头羊&#xff0c;不管是性能还是质量上&#xff0c;都是受到一致的好评&#xff0c;而南卡Lite pro2蓝牙耳机更是匠心级别的制造工艺&#xff01;花几百元的价格配置了其千元蓝牙耳机才配置的高通3040蓝牙…

纽扣电池十大品牌

纽扣电池十大品牌 随着消费类电子产品种类的越来越多&#xff0c;外型越来越小&#xff0c;纽扣电池的使用也变得日益广泛。厂家在寻找供应商&#xff0c;顾客在挑选纽扣电池品牌时&#xff0c;面临着越来越多的选择&#xff0c;而纽扣电池因为其工业配套为主&#xff0c;商业…

全球与中国电池级羧甲基纤维素钠(CMC)市场现状及未来发展趋势2022-2028

根据QYR&#xff08;恒州博智&#xff09;的统计及预测&#xff0c;2021年全球电池级羧甲基纤维素钠&#xff08;CMC&#xff09;市场销售额达到了 亿美元&#xff0c;预计2028年将达到 亿美元&#xff0c;年复合增长率&#xff08;CAGR&#xff09;为 %&#xff08;2022-2028&…

y7000电池固件_联想 拯救者Y7000P 怎么拿出电池?

电脑能力主要看处理器和显卡。 办公用途&#xff0c;平面设计类吃处理器的能力。 游戏&#xff0c;3d设计类吃显卡能力。 不管那种需求&#xff0c;内存容量都是需要有合理的选择(类如看机器是否支持扩展)。 不管台式本子都一样。 本子的能力和选择问题下面说法参考。 &#xf…