JavaWeb第五天

该文档是:JavaWeb学习

博客连接:https://www.loveuluo.cn

日期:2020-12-04

1. JavaWeb概念

a) 什么是 JavaWeb

JavaWeb 是指,所有通过 Java 语言编写可以通过浏览器访问的程序的总称,叫 JavaWeb。
JavaWeb 是基于请求和响应来开发的。

b) 什么是请求

请求是指客户端给服务器发送数据,叫请求 Request。

c)什么是响应 什么是响应
响应是指服务器给客户端回传数据,叫响应 Response。

d)请求和响应的关系 请求和响应的关系
请求和响应是成对出现的,有请求就有响应。

image-20201204094556273

2. Web资源的分类

web 资源按实现的技术和呈现的效果的不同,又分为静态资源和动态资源两种。

静态资源: html、css、js、txt、mp4 视频 , jpg 图片
动态资源: jsp 页面、Servlet 程序

3. 常用的 Web 服务器

image-20201204094825409

4. Tomcat 服务器和 Servlet 版本的对应关系

image-20201204095739623

5. Tomcat的使用

image-20201204100111674

image-20201204100141165

image-20201204100200841

image-20201204100210290

image-20201204100232579

image-20201204100251010

image-20201204100306291

image-20201204100322899

image-20201204100340064

image-20201204100400194

image-20201204100431002

image-20201204100524533

image-20201204100542591

6. IDEA整合 Tomcat服务器

image-20201204104150803

image-20201204104157217

image-20201204104207033

7. IDEA中动态web工程和Tomcat的操作

7.1 IDEA 中如何创建动态 web 工程

1、创建一个新模块:

image-20201204105145581

2、选择你要创建什么类型的模块:

image-20201204105158192

3、输入你的模块名,点击【Finish】完成创建。

image-20201204105209683

4、创建成功如下图:

image-20201204105220841

7.2Web 工程的目录介绍

image-20201204105253624

7.3 如何给动态 web 工程添加额外 jar 包

提示:也可以直接全部放入lib中,选中全部jar包然后右键add as library。(这种方法比较方便)

1、可以打开项目结构菜单操作界面,添加一个自己的类库:

image-20201204105333863

2、添加你你类库需要的 jar 包文件。

image-20201204105344496

3、选择你添加的类库,给哪个模块使用:

image-20201204105357217

4、选择 Artifacts 选项,将类库,添加到打包部署中:

image-20201204105435345

7.4 如何在 IDEA 中部署工程到 Tomcat 上运行

1、建议修改 web 工程对应的 Tomcat 运行实例名称:

image-20201204105558037

2、确认你的 Tomcat 实例中有你要部署运行的 web 工程模块:

image-20201204105615455

3、你还可以修改你的 Tomcat 实例启动后默认的访问地址:

image-20201204105628881

4、在 IDEA 中如何运行,和停止 Tomcat 实例。

4.1、正常启动 Tomcat 实例:

image-20201204105650565

4.2、Debug 方式启动 Tomcat 运行实例:

image-20201204105659247

4.3、停止 Tomcat 运行实例:

image-20201204105707398

4.4、重启 Tomcat 运行实例:

image-20201204105717144

image-20201204142941268

7.5修改工程访问路径

image-20201204105734200

7.6 修改运行的端口号

image-20201204105830844

7.7 修改运行使用的浏览器 修改运行使用的浏览器

image-20201204105856770

7.8 配置资源热部署

代表修改内容的时候不需要重启服务器,会直接生效。

1.update resources ---- 更新静态的资源,比如html,js,css等 运行模式和调试模式都是立即生效;
2.update classes and resources ---- 更新java,jsp和静态资源( 1. java修改后,会被编译成.class,然后覆盖到target/kao文件夹下,IDE调试模式的情况下,

image-20201204105914890

8. Servlet 技术

8.1 什么是 Servlet

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

8.2 手动实现 Servlet 程序 程序

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

image-20201205090842900

image-20201205090904319

Servlet 程序的示例代码(主要看service方法):

public class HelloServlet implements Servlet {
    @Override
    public void init(ServletConfig servletConfig) throws ServletException {

    }

    @Override
    public ServletConfig getServletConfig() {
        return null;
    }

    /**
     *  service 方法是专门用来处理请求和响应的
     * @param servletRequest
     * @param servletResponse
     * @throws ServletException
     * @throws IOException
     */
    @Override
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException     {
        System.out.println("Hello Servlet被访问了"); //只要访问HelloServlet程序 就会执行service方法
    }

    @Override
    public String getServletInfo() {
        return null;
    }

    @Override
    public void destroy() {
    }
}

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>
        <!--servlet-name 标签 Servlet 程序起一个别名(一般是类名) -->
        <servlet-name>HelloServlet</servlet-name>
        <!--servlet-class 是 Servlet 程序的全类名 -->
        <servlet-class>com.Luo.HelloServlet</servlet-class>
    </servlet>
    <!--servlet-mapping 标签给 servlet 程序配置访问地址 -->
    <servlet-mapping>
        <!--servlet-name 标签的作用是告诉服务器,我当前配置的地址给哪个 Servlet 程序使用 -->
        <servlet-name>HelloServlet</servlet-name><!-- 跟上边的servlet-name中的内容对应,代表给它用的 -->
        <!--url-pattern 标签配置访问地址 <br/>
        / 斜杠在服务器解析的时候,表示地址为: http://ip:port/ 工程路径 <br/>
        /hello 表示地址为: http://ip:port/ 工程路径 /hello <br/>
        -->
        <url-pattern>/hello</url-pattern>
    </servlet-mapping>
</web-app>

常见的错误 1:url-pattern 中配置的路径没有以斜杠打头。

image-20201205091059597

常见错误 2:servlet-name 配置的值不存在:

image-20201205091109670

常见错误 3:servlet-class 标签的全类名配置错误:

image-20201205091119295

8.3 url 地址到 Servlet 程序的访问图解

image-20201205093734103

8.4 Servlet 的生命周期

1、执行 Servlet 构造器方法
2、执行 init 初始化方法
第一、二步,是在第一次访问,的时候创建 Servlet 程序会调用。
3、执行 service 方法
第三步,每次访问都会调用。
4、执行 destroy 销毁方法
第四步,在 web 工程停止的时候调用。

image-20201205095313280

代码情况:

public class HelloServlet implements Servlet {
    public HelloServlet() {
        System.out.println("1 构造器方法");
    }

    @Override
    public void init(ServletConfig servletConfig) throws ServletException {
        System.out.println("2 init初始化方法");
    }

    @Override
    public ServletConfig getServletConfig() {
        return null;
    }

    /**
     *  service 方法是专门用来处理请求和响应的
     * @param servletRequest
     * @param servletResponse
     * @throws ServletException
     * @throws IOException
     */
    @Override
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
        System.out.println("3 service == Hello Servlet被访问了"); //只要访问HelloServlet程序 就会执行service方法
    }

    @Override
    public String getServletInfo() {
        return null;
    }

    @Override
    public void destroy() {
        System.out.println("4 . destroy销毁方法");
    }
}

8.5 GET 和 和 POST 请求的分发处理 请求的分发处理

详情查看代码,比较详细。

image-20201205102813262

image-20201205103154056

代码(主要看service方法):

public class HelloServlet implements Servlet {
    
    public HelloServlet() {System.out.println("1 构造器方法");}
    @Override
    public void init(ServletConfig servletConfig) throws ServletException {ystem.out.println("2 init初始化方法");}
    @Override
    public ServletConfig getServletConfig() {return null;}

    /**
     *  service 方法是专门用来处理请求和响应的
     * @param servletRequest
     * @param servletResponse
     * @throws ServletException
     * @throws IOException
     */
    @Override
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
    //ServletRequest:定义将客户端请求信息提供给某个 servlet 的对象;servlet 容器创建 ServletRequest 对象,并将该对象作为参数传递给该servlet的service方法
        //因为servletRequest没有getMethod()方法,所以要先转换成HttpservletRequest (HttpServletRequest是ServletRequest的子类)
        HttpServletRequest httpServletRequest= (HttpServletRequest) servletRequest;
        String method = httpServletRequest.getMethod();//可以获得请求的方式(是POST还是GET)

        //一般情况下GET请求和POST请求要干的事是不一样的
        if (method.equals("POST")){ //如果是POST则做什么操作
        //而直接在这里写要做的事可能会让代码太臃肿,因为post请求要做的事不是一行两行 是很多行,所以单独抽出一个doPost方法(),后期维护也更方便
        doPost();
        }else if (method.equals("GET")){ //如果是GET则做什么操作
        //而直接在这里写要做的事可能会让代码太臃肿,因为get请求要做的事不是一行两行 是很多行,所以单独抽出一个doGet方法(),后期维护也更方便
        doGet();
        }
    }

    /**
     * 做 get 请求的操作
     */
    public void doPost(){
        System.out.println("post请求");
        System.out.println("post请求");
        System.out.println("post请求");
    }

    /**
     * 做 post 请求的操作
     */
    public void doGet(){
        System.out.println("get请求");
        System.out.println("get请求");
        System.out.println("get请求");
    }

    @Override
    public String getServletInfo() { return null;}
    @Override
    public void destroy() {System.out.println("4 . destroy销毁方法");}
}

8.6 通过继承 HttpServlet 实现 Servlet 程序

一般在实际项目开发中,都是使用继承 HttpServlet 类的方式去实现 Servlet 程序,因为写起来简单很多。

HttpServlet这个类中已经帮我们分发好了GET请求和POST请求,直接重写doGet和doPost方法即可。

1、编写一个类去继承 HttpServlet 类
2、根据业务需要重写 doGet 或 doPost 方法
3、到 web.xml 中的配置 Servlet 程序的访问地址

Servlet 类的代码:

public class HelloServlet2 extends HttpServlet {
    //使用快捷键重写doGet和doPost方法

    /**
     *  doPost ()在 post 请求的时候调用
     * @param req
     * @param resp
     * @throws ServletException
     * @throws IOException
     */
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("HelloServlet2 的 doGet方法");
    }

    /**
     * doGet ()在 get 请求的时候调用
     * @param req
     * @param resp
     * @throws ServletException
     * @throws IOException
     */
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("HelloServlet2 的 doPost方法");
    }
}

web.xml 中的配置:

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

8.7 使用 IDEA 自动创建 Servlet 程序

菜单:new ->Servlet 程序

image-20201205110606276

配置 Servlet 的信息(勾上这个勾代表使用注解模式开发,本次学习我们使用xml配置开发):

image-20201205110616371

8.8 Servlet 类的继承体系

image-20201205152708218

9. ServletConfig 类

ServletConfig 类从类名上来看,就知道是 Servlet 程序的配置信息类,ServletConfig中封装了初始化配置的信息。

Servlet 程序和 ServletConfig 对象都是由 Tomcat 负责创建,我们负责使用。
Servlet 程序默认是第一次访问的时候创建,ServletConfig 是每个 Servlet 程序创建时,就创建一个对应的 ServletConfig 对象。

ServletConfig 类的三大作用

1、可以获取 Servlet 程序的别名 servlet-name 的值
2、获取初始化参数 init-param(init-param可以再web.xml中进行配置 例子如下)
3、获取 ServletContext 对象

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 -->
    <!-- servlet 标签给 Tomcat 配置 Servlet 程序 -->
    <servlet>
        <!--servlet-name 标签 Servlet 程序起一个别名(一般是类名) -->
        <servlet-name>HelloServlet</servlet-name>
        <!--servlet-class 是 Servlet 程序的全类名 -->
        <servlet-class>com.atguigu.servlet.HelloServlet</servlet-class>
        <!--init-param 是初始化参数 -->
        <init-param>
            <!-- 是参数名 -->
            <param-name>username</param-name>
            <!-- 是参数值 -->
            <param-value>root</param-value>
        </init-param>
        <!--init-param 是初始化参数 -->
        <init-param>
            <!-- 是参数名 -->
            <param-name>url</param-name>
            <!-- 是参数值 -->
            <param-value>jdbc:mysql://localhost:3306/test</param-value>
        </init-param>
    </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>

    <!-- 第二个Servlet -->
    <servlet>
        <servlet-name>HelloServlet2</servlet-name>
        <servlet-class>com.Luo.HelloServlet2</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>HelloServlet2</servlet-name>
        <url-pattern>/hello2</url-pattern>
    </servlet-mapping>
</web-app>

9.1 继承了Servlet的类可以获得ServletConfig

每个ServletConfig对应他自己的Servlet的程序,不能得到别人的Servlet信息,他只能得到自己的。

image-20201205161053621

代码(主要看init方法):

public class HelloServlet implements Servlet {
    public HelloServlet() {
        System.out.println("1 构造器方法");
    }

    @Override
    public void init(ServletConfig servletConfig) throws ServletException {
        //可以获取 Servlet 程序的别名 servlet-name 的值
        String servletName = servletConfig.getServletName();
        System.out.println("HelloServlet  程序的别名是:"+servletName);
        //根据名称获取获取初始化参数 init-param
        String username = servletConfig.getInitParameter("username");
        String url = servletConfig.getInitParameter("url");
        System.out.println(" 初始化参数 username  的值是;" + username);
        System.out.println(" 初始化参数 url  的值是;" + url);
    }

    @Override
    public ServletConfig getServletConfig() {
        return null;
    }

    @Override
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException{
    }

    @Override
    public String getServletInfo() {
        return null;
    }

    @Override
    public void destroy() {
        System.out.println("4 . destroy销毁方法");
    }
}

9.2 继承了HttpServlet的类也可以获得ServletConfig

每个ServletConfig对应他自己的Servlet的程序,不能得到别人的Servlet信息,他只能得到自己的。

image-20201205161031469

image-20201205160951483

代码:

public class HelloServlet2 extends HttpServlet {
    //使用快捷键重写doGet和doPost方法

    /**
     *  doPost ()在 post 请求的时候调用
     * @param req
     * @param resp
     * @throws ServletException
     * @throws IOException
     */
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("HelloServlet2 的 doGet方法");
        // 也可以使用
        ServletConfig servletConfig=getServletConfig();
        System.out.println(servletConfig);
        // 获取初始化参数 init-param
        System.out.println(" 初始化参数 username  的值是;" + servletConfig.getInitParameter("username"));
        System.out.println(" 初始化参数 url  的值是;" + servletConfig.getInitParameter("url"));
    }

    /**
     * doGet ()在 get 请求的时候调用
     * @param req
     * @param resp
     * @throws ServletException
     * @throws IOException
     */
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("HelloServlet2 的 doPost方法");
    }
}

9.3 注意点:继承HttpServlet的类重写了init方法

如果继承HttpServlet的类重写了init方法,一定要调用父类的init(ServletConfig操作)【super.init(config);】,否则会空指针异常。原因是:当子类也有init方法,父类也有init方法的时候,调用init方法的时候会调用子类的,而因为父类(GenericServlet)中做了保存config的操作,所以调用父类的init方法,就可以完成config的保存。

image-20201205162644392

public class HelloServlet2 extends HttpServlet {

    @Override
    public void init(ServletConfig config) throws ServletException {
        //调用父类的init方法之后就不会报空指针异常
        super.init(config);
        System.out.println("重写了init初始化方法,假如做了一些工作...");
    }

    /**
     *  doPost ()在 post 请求的时候调用
     * @param req
     * @param resp
     * @throws ServletException
     * @throws IOException
     */
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("HelloServlet2 的 doGet方法");
        // 也可以使用
        ServletConfig servletConfig=getServletConfig();
        System.out.println(servletConfig);
        // 获取初始化参数 init-param
        System.out.println(" 初始化参数 username  的值是;" + servletConfig.getInitParameter("username"));
        System.out.println(" 初始化参数 url  的值是;" + servletConfig.getInitParameter("url"));
    }

    /**
     * doGet ()在 get 请求的时候调用
     * @param req
     * @param resp
     * @throws ServletException
     * @throws IOException
     */
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("HelloServlet2 的 doPost方法");
    }
}

10. ServletContext 类

10.1 什么是 ServletContext

1、ServletContext 是一个接口,它表示 Servlet 上下文对象
2、一个 web 工程,只有一个 ServletContext 对象实例。
3、ServletContext 对象是一个域对象。
4、ServletContext 是在 web 工程部署启动的时候创建。在 web 工程停止的时候销毁。

什么是域对象?
域对象,是可以像 Map 一样存取数据的对象,叫域对象。
这里的域指的是存取数据的操作范围,整个 web 工程。

Map和域对象对比存数据取数据删除数据
Mapput()get()remove()
域对象setAttribute()getAttribute()removeAttribute();

10.2 ServletContext 类的四个作用 类的四个作用

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

提示:ServletContext是在web工程部署启动的时候创建。在web工程停止的时候销毁

10.2.1 演示1~3个作用

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

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>
<!-- 第三个Servlet -->
<servlet>
    <servlet-name>ContextServlet</servlet-name>
    <servlet-class>com.Luo.ContextServlet</servlet-class>
</servlet>
<servlet-mapping>
    <servlet-name>ContextServlet</servlet-name>
    <url-pattern>/contextServlet</url-pattern>
</servlet-mapping>

和第3个作用有关的配图:

image-20201205171037417

输出的结果为:

image-20201205172226417

代码:

public class ContextServlet extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

    }

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //获取servletContext (也可以直接调用getServletContext();方法获取)
        ServletContext servletContext = getServletConfig().getServletContext();
        // 第1个作用:获取 web.xml 中配置的上下文参数 context-param
        String username = servletContext.getInitParameter("username");
        System.out.println("context-param  参数 username  的值是:" + username);
        System.out.println("context-param  参数 password  的值是:" + servletContext.getInitParameter("password"));

        // 第2个作用:获取当前的工程路径,格式 : / 工程路径
        System.out.println( " 当前工程路径:" + servletContext.getContextPath() );

        // 第3个作用:获取工程部署后在服务器硬盘上的绝对路径
        // 斜杠被服务器解析地址为: http://ip:port/工程名/ 映射到IDEA代码的web目录
        System.out.println(" 工程部署的路径是:" + servletContext.getRealPath("/"));
        System.out.println(" 工程下 css  目录的绝对路径是:" + servletContext.getRealPath("/css"));
        System.out.println(" 工程下 imgs  目录 1.jpg  的绝对路径是:" + servletContext.getRealPath("/imgs/1.jpg"));
    }
}

10.2.2 演示第4个作用

4、像 Map 一样存取数据

提示:ServletContext是在web工程部署启动的时候创建。在web工程停止的时候销毁(重启也会销毁)

演示访问的顺序是,先访问ContextServlet1再访问ContextServlet2再次访问ContextServlet1:

image-20201205192920599

可以发现结果,第一次访问ContextServlet1,在往域对象中保存数据之前获取key1的值的时候还是null,保存之后成功获取到,此时域对象中已经存在数据,所以ContextServlet2也成功访问到,第二次访问ContextServlet1的时候,也在保存之前访问到key1的值(因为数据已经存在域对象之中)。

image-20201205192831216

ContextServlet1代码:

public class ContextServlet1 extends HttpServlet {

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 获取ServletContext对象
        ServletContext context = getServletContext();
        System.out.println("在ContextServlet1中打印context对象:"+context);
        System.out.println(" 保存之前:Context1取获取key1的值是:"+context.getAttribute("key1"));
        // 往域对象中存储数据
        context.setAttribute("key1","value1");
        // 获取数据
        System.out.println("Context1中获取域数据key1的值是:"+context.getAttribute("key1"));
    }
}

ContextServlet2代码:

public class ContextServlet2 extends HttpServlet {

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        ServletContext context = getServletContext();
        System.out.println("在ContextServlet2中打印context对象:"+context);
        System.out.println("Context2中获取域数据key1的值是:"+context.getAttribute("key1"));
    }
}

11. HTTP协议

小教程,如何用谷歌或者火狐查看HTTP协议:

谷歌浏览器如何查看 HTTP 协议:

image-20201205205136107

火狐浏览器如何查看 HTTP 协议:

image-20201205205201299

11.1 什么是 HTTP 协议

image-20201205193746862

11.2 请求的 HTTP 协议格式

客户端给服务器发送数据叫请求。
服务器给客户端回传数据叫响应。

请求又分为 GET 请求,和 POST 请求两种

11.2.1 GET请求

1、请求行
(1) 请求的方式 GET
(2) 请求的资源路径[+?+请求参数]
(3) 请求的协议的版本号 HTTP/1.1
2、请求头
key : value 组成 不同的键值对,表示不同的含义。

image-20201205195245373

11.2.2 POST请求

1、请求行
(1) 请求的方式 POST
(2) 请求的资源路径[+?+请求参数]
(3) 请求的协议的版本号 HTTP/1.1
2、请求头
1) key : value 不同的请求头,有不同的含义
空行
3、请求体 ===>>> 就是发送给服务器的数据

image-20201205202025005

11.2.3 常用请求头的说明

Accept: 表示客户端可以接收的数据类型
Accpet-Languege: 表示客户端可以接收的语言类型
User-Agent: 表示客户端浏览器的信息
Host: 表示请求时的服务器 ip 和端口号

11.2.4 哪些是GET请求,哪些是POST请求

GET 请求有哪些:

1、form 标签 method=get
2、a 标签
3、link 标签引入 css
4、Script 标签引入 js 文件
5、img 标签引入图片
6、iframe 引入 html 页面
7、在浏览器地址栏中输入地址后敲回车

POST 请求有哪些:

1、form 标签 method=post

11.3 响应的 HTTP 协议格式

11.3.1 响应实例

1、响应行
(1) 响应的协议和版本号
(2) 响应状态码
(3) 响应状态描述符
2、响应头
(1) key : value 不同的响应头,有其不同含义
空行
3、响应体 ---->>> 就是回传给客户端的数据

image-20201205203633060

11.3.2 常见的响应码说明

200 表示请求成功
302 表示请求重定向 (下边会详细了解)
404 表示请求服务器已经收到了,但是你要的数据不存在(请求地址错误)
500 表示服务器已经收到请求,但是服务器内部错误(代码错误)

11.3.3 MIME 类型说明 类型说明

MIME 是 HTTP 协议中数据类型。
MIME 的英文全称是"Multipurpose Internet Mail Extensions" 多功能 Internet 邮件扩充服务。MIME 类型的格式是“大类型/小
类型”,并与某一种文件的扩展名相对应。

常见的 MIME 类型:

image-20201205205014906

12. HttpServletRequest 类

ServletContext、HttpSession和HttpServletRequest的区别和联系:

1. ServletContext:范围最大,应用程序级别的,整个应用程序都能访问;

2. HttpSession–次之,会话级别的,在当前的浏览器中都能訪问[不论是在同一浏览器开多少窗体,都能够访问,可是换个浏览器就不行了,就必须又一次创建session;

3. HttpServletRequest–范围最小,请求级别,请求结束,变量的作用域也结束【也就是仅仅是一次访问,访问结束,这个也结束】[

12.1 HttpServletRequest 类有什么作用

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

12.2 HttpServletRequest 类的常用方法

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

下边代码演示了i.getRequestURI(),ii.getRequestURL(),iii.getRemoteHost(),iv.getHeader(),vii.getMethod()这五个方法。

image-20201206145357072

image-20201206145408665

示例代码:

public class RequestAPIServlet extends HttpServlet {

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse res) throws ServletException, IOException {
        // i.getRequestURI() 获取请求的资源路径
        System.out.println("URI => " + req.getRequestURI());
        // ii.getRequestURL() 获取请求的统一资源定位符(绝对路径)
        System.out.println("URL => " + req.getRequestURL());
        // iii.getRemoteHost() 获取客户端的 ip 地址
        /*
         * 在 IDEA 中,使用 localhost 访问时,得到的客户端 ip 地址是 ===>>> 127.0.0.1<br/>
         * 在 IDEA 中,使用 127.0.0.1 访问时,得到的客户端 ip 地址是 ===>>> 127.0.0.1<br/>
         * 在 IDEA 中,使用 真实 ip 访问时,得到的客户端 ip 地址是 ===>>> 真实的客户端 ip 地址 <br/>
        */
        System.out.println("客户端 ip 址 地址 => " + req.getRemoteHost());
        // iv.getHeader() 获取请求头
        System.out.println("请求头 User-Agent ==>> " + req.getHeader("User-Agent"));
        // vii.getMethod() 获取请求的方式 GET 或 POST
        System.out.println( "请求的方式 ==>> " + req.getMethod() );
    }
}

12.3 HttpServletRequest常用方法之获取请求参数

image-20201206152427641

image-20201206152457119

表单:

<body>
<form action="http://localhost:8080/07_servlet/parameterServlet" method="get">
    用户名:<input type="text" name="username"><br/>
    密码:<input type="password" name="password"><br/>
    兴趣爱好:<input type="checkbox" name="hobby" value="cpp">C++
    <input type="checkbox" name="hobby" value="java">Java
    <input type="checkbox" name="hobby" value="js">JavaScript<br/>
    <input type="submit">
</form>
</body>

代码:

public class ParameterServlet extends HttpServlet {

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 获取请求参数
        // 获取单个值 (根据表单中的name=?获取值)
        String username = request.getParameter("username");
        String password = request.getParameter("password");
        // 获取多个值 (根据表单中的name=?获取值)
        String[] hobbies = request.getParameterValues("hobby");

        System.out.println("用户名:"+username);
        System.out.println("密码:" + password);
        System.out.println("兴趣爱好:" + Arrays.asList(hobbies));
    }
}

12.4 POST 请求的中文乱码解决

获取POST请求参数如果填上中文的时候可能会存在乱码,如何解决:

@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    // 设置请求体的字符集为 UTF-8 ,从而解决 post 请求的中文乱码问题
    req.setCharacterEncoding("UTF-8");
    System.out.println("-------------doPost------------");
    
    // 获取请求参数
    // 获取单个值 (根据表单中的name=?获取值)
    String username = req.getParameter("username");
    String password = req.getParameter("password");
    // 获取多个值 (根据表单中的name=?获取值)
    String[] hobbies = req.getParameterValues("hobby");

    System.out.println("用户名:"+username);
    System.out.println("密码:" + password);
    System.out.println("兴趣爱好:" + Arrays.asList(hobbies));
}

12.5 请求的转发

什么是请求的转发?
请求转发是指,服务器收到请求后,从一次资源跳转到另一个资源的操作叫请求转发。

ServletContext、HttpSession和HttpServletRequest的区别和联系:

1. ServletContext:范围最大,应用程序级别的,整个应用程序都能访问;

2. HttpSession–次之,会话级别的,在当前的浏览器中都能訪问[不论是在同一浏览器开多少窗体,都能够访问,可是换个浏览器就不行了,就必须又一次创建session;

3. HttpServletRequest–范围最小,请求级别,请求结束,变量的作用域也结束【也就是仅仅是一次访问,访问结束,这个也结束】[

image-20201206162937487

image-20201206165914381

image-20201206165910389

Servlet1 代码:

public class Servlet1 extends HttpServlet {

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 获取请求的参数(办事的材料)查看
        String username = request.getParameter("username");
        System.out.println("在Servlet1(柜台 1 )中查看参数(材料):"+username);

        // 给材料盖一个章,并传递到 Servlet2(柜台 2)去查看
        System.out.println("柜台1盖章完毕");
        request.setAttribute("key1","柜台1的章"); //把资源放入HttpServletRequest域中

        // 问路:Servlet2(柜台 2)怎么走
        // 请求转发必须要以斜杠打头, / 斜杠表示地址为:http://ip:port/工程名/,映射到 IDEA 代码的 web 目录
        RequestDispatcher requestDispatcher = request.getRequestDispatcher("/servlet2");

        // 走向Servlet2(柜台 2)
        System.out.println("前往柜台2");
        requestDispatcher.forward(request,response); //同时把request和response传给Servlet2
        //Servlet2(柜台2)执行完毕后,会接着执行forward()后下边的代码。
        System.out.println("又回到柜台1");
    }
}

Servlet2 代码:

public class Servlet2 extends HttpServlet {

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 获取请求的参数(办事的材料)查看 (这里的request是从Servlet1中传过来的)
        String username = request.getParameter("username");
        System.out.println("在Servlet2(柜台2)中查看参数(材料):" + username);

        // 查看柜台1是否有盖章
        Object key1 = request.getAttribute("key1");
        System.out.println("查看柜台1是否有盖章:" + key1);

        // 处理自己的业务
        System.out.println("Servlet2 处理自己的业务中...");
        System.out.println("Servlet2 处理完毕自己的业务");
    }
}

扩展:base标签

image-20201206193402248

<!DOCTYPE html>
<html lang="zh_CN">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <!--base 标签设置页面相对路径工作时参照的地址
        href 属性就是参数的地址值
    -->
    <base href="http://localhost:8080/07_servlet/a/b/">
</head>
<body>
    这是 a 下的 b 下的 c.html 页面<br/>
    <a href="../../index.html">跳回首页</a><br/>
</body>
</html>

扩展:Web 中的相对路径和绝对路径

image-20201206193806901

扩展:web中 / 斜杠的不同意义

image-20201206193924214

13. HttpServletResponse 类

13.1 HttpServletResponse 类的作用

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

13.2 HttpServletResponse 中两个输出流的说明

image-20201206194914110

13.3 如何往客户端回传数据

要求 : 往客户端回传字符串数据。

image-20201207090035398

public class ResponseIOServlet extends HttpServlet {

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 获得字符流
        PrintWriter writer = response.getWriter();
        // 往客户端传输字符数据
        writer.write("Hello response!");
    }
}

13.4 响应中文的乱码解决

image-20201207091351281

解决方法一(不推荐使用了解即可):

// 设置服务器字符集为 UTF-8
resp.setCharacterEncoding("UTF-8");
// 通过响应头,设置浏览器也使用 UTF-8 字符集
resp.setHeader("Content-Type", "text/html; charset=UTF-8");

解决方法二(推荐使用):

public class ResponseIOServlet extends HttpServlet {

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 它会同时设置服务器和客户端都使用 UTF-8 字符集,还设置了响应头
        // 此方法一定要在获取流对象之前调用才有效
        response.setContentType("text/html; charset=UTF-8");
        // 获得字符流
        PrintWriter writer = response.getWriter();
        // 往客户端传输字符数据
        writer.write("你好,response");
    }
}

13.5 请求重定向

请求重定向,是指客户端给服务器发请求,然后服务器告诉客户端说。我给你一些地址。你去新地址访问。叫请求
重定向(因为之前的地址可能已经被废弃)。

image-20201207110607083

请求重定向的第一种方案:(不推荐使用了解即可)

// 设置响应状态码 302 ,表示重定向,(已搬迁)
resp.setStatus(302);
// 设置响应头,说明 新的地址在哪里
resp.setHeader("Location", "http://localhost:8080");

请求重定向的第二种方案(推荐使用):

因为如果要重定向,302状态是固定的,所以直接省去了,这个方案简化了代码 ,直接填写要重定向的地址就行。

访问该地址:

image-20201207111825160

会重定向到(地址栏的地址会发生改变):

image-20201207111850295

控制台状态:访问1时状态码为302(代表重定向),响应头中有需要重定向到那的地址。

image-20201207121021508

image-20201207121142267

Response1代码:

public class Response1 extends HttpServlet {

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        System.out.println("Response1曾到此一游");

        //重定向到填写的链接
        response.sendRedirect("http://localhost:8080/07_servlet/response2");
    }
}

Response2代码:

public class Response2 extends HttpServlet {

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        response.setContentType("text/html; charset=UTF-8");
        PrintWriter writer = response.getWriter();
        writer.write("我的是Response2!");
    }
}
最后修改:2020 年 12 月 07 日 12 : 34 PM
如果觉得我的文章对你有用,请随意赞赏