JavaWeb学习6:Servlet

1、什么是Servlet

  • sun公司用于开发动态web的一种技术
  • sun公司在这里API中提供了一个接口,叫做Servlet,如果想开发一个Servlet程序,只需要完成两个步骤:
    1. 编写一个类,实现Servlet接口。
    2. 把开发好的Java类部署到web服务器中。

把实现了Servlet接口的java程序叫做Servlet

2、Servlet工程创建

Servlet接口在sun公司有两个默认的实现类:一个是HttpServ,另一个是GenericServlet。

2.1、构建一个普通的Maven项目

删掉里面的src目录,后续的编程就在这个项目里建立Moudel。这个空的工程就是Maven的主工程。

2.2、 关于Maven父子工程的理解:

父项目中会有:

<modules>
	<module>servlet-1</module>
</modules>

子项目中会有:

<parent>
	<artifactId>javaweb-02-servlet</artifactId>
	<groupId>org.example</groupId>
	<version>1.0-SNAPSHOT</version>
</parent>

父项目中的jar包子项目可以直接使用

2.3、Maven环境优化

  1. 修改web.xml
  2. 将Maven搭建完整

2.4、编写一个Servlet程序

  1. 编写一个普通类
  2. 实现Servlet接口,这里直接继承HttpServlet

image

点击查看Servlet接口代码
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by FernFlower decompiler)
//

package javax.servlet;

import java.io.IOException;

public interface Servlet {
//初始化
    void init(ServletConfig var1) throws ServletException;
//获得Servlet配置
    ServletConfig getServletConfig();

    void service(ServletRequest var1, ServletResponse var2) throws ServletException, IOException;
//获得Servlet信息
    String getServletInfo();
//销毁
    void destroy();
}

GenericServlet.Class对于void service没有做任何处理,直接继承

GenericServlet.Class部分内容
public abstract void service(ServletRequest var1, ServletResponse var2) throws ServletException, IOException;
HttpServlet.class中service方法
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
	String method = req.getMethod();
	long lastModified;
	if (method.equals("GET")) {
		lastModified = this.getLastModified(req);
		if (lastModified == -1L) {
		this.doGet(req, resp);
		} else {
			long ifModifiedSince = req.getDateHeader("If-Modified-Since");
			if (ifModifiedSince < lastModified) {
				this.maybeSetLastModified(resp, lastModified);
				this.doGet(req, resp);
			} else {
				resp.setStatus(304);
			}
		}
	} else if (method.equals("HEAD")) {
		lastModified = this.getLastModified(req);
		this.maybeSetLastModified(resp, lastModified);
		this.doHead(req, resp);
	} else if (method.equals("POST")) {
		this.doPost(req, resp);
	} else if (method.equals("PUT")) {
		this.doPut(req, resp);
	} else if (method.equals("DELETE")) {
		this.doDelete(req, resp);
	} else if (method.equals("OPTIONS")) {
		this.doOptions(req, resp);
	} else if (method.equals("TRACE")) {
		this.doTrace(req, resp);
	} else {
		String errMsg = lStrings.getString("http.method_not_implemented");
		Object[] errArgs = new Object[]{method};
		errMsg = MessageFormat.format(errMsg, errArgs);
		resp.sendError(501, errMsg);
	}
}

在自己创建的类的里面只需要重写HttpServlet中的doPost等方法就可以了

public class HelloServlet extends HttpServlet {
//    由于get或者post只是请求实现的不同方式,可以相互调用,业务逻辑都一样
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //ServletOutputStream outputStream = resp.getOutputStream();
        PrintWriter writer = resp.getWriter(); //响应流

        writer.print("Hello,Servlet");
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        super.doPost(req, resp);
    }
}

2.5、编写Servlet的映射

为什么需要映射:写的是Java程序,但是要通过浏览器访问,而浏览器需要链接web服务器,所以需要在web服务中注册我们的写的Servlet,所以还需要给他一个浏览器能够访问的路径。
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"
         metadata-complete="true">

<!--  一个Servlet要对应一个class  -->
<!--  注册Servl  -->
    <servlet>
        <servlet-name>hello</servlet-name>
        <servlet-class>com.jw02.servlet.HelloServlet</servlet-class>
    </servlet>
<!--  Servl的请求路径  -->
    <servlet-mapping>
        <servlet-name>hello</servlet-name>
        <url-pattern>/hello</url-pattern>
    </servlet-mapping>
</web-app>

2.6、配置Tomcat

IDEA中配置Tomcat
image

2.7、启动测试

image
image

3、Servlet原理

Servlet由web服务器调用,web服务器在收到浏览器请求之后,会:
image

4、Mapping问题

4.1、一个Servlet可以指定一个映射路径

<servlet>
	<servlet-name>hello</servlet-name>
	<servlet-class>com.jw02.servlet.HelloServlet</servlet-class>
</servlet>
<servlet-mapping>
	<servlet-name>hello</servlet-name>
	<url-pattern>/hello</url-pattern>
</servlet-mapping>

4.2、一个Servlet可以指定多个映射路径

<servlet>
	<servlet-name>hello</servlet-name>
	<servlet-class>com.jw02.servlet.HelloServlet</servlet-class>
</servlet>
<servlet-mapping>
	<servlet-name>hello</servlet-name>
	<url-pattern>/hello1</url-pattern>
</servlet-mapping>
<servlet-mapping>
	<servlet-name>hello</servlet-name>
	<url-pattern>/hello2</url-pattern>
</servlet-mapping>
<servlet-mapping>
	<servlet-name>hello</servlet-name>
	<url-pattern>/hello3</url-pattern>
</servlet-mapping>

4.3、一个Servlet可以指定通用映射路径

<servlet>
	<servlet-name>hello</servlet-name>
	<servlet-class>com.jw02.servlet.HelloServlet</servlet-class>
</servlet>
<servlet-mapping>
	<servlet-name>hello</servlet-name>
	<url-pattern>/hello/*</url-pattern>
	<!--http://localhost:8080/s1/hello/weqeqweqe-->
	<!--后面无论是什么都可以请求成功-->
</servlet-mapping>

如果没有写/hello/*,而是直接写/*就会跳过index.jsp,直接显示Hello,Servlet。不要这样子写。

4.4、指定一些后缀或者前缀等等...

http://localhost:8080/s1/abc.wuyu

<servlet>
	<servlet-name>hello</servlet-name>
	<servlet-class>com.jw02.servlet.HelloServlet</servlet-class>
</servlet>	<servlet-mapping>
	<servlet-name>hello</servlet-name>
	<url-pattern>*.wuyu</url-pattern>
</servlet-mapping>

4.5、优先级问题

写一个Error404页面

点击查看ErroeServlet代码
package com.jw02.servlet;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;

public class ErroeServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        resp.setContentType("text/html");
        resp.setCharacterEncoding("utf-8");

        PrintWriter writer = resp.getWriter();
        writer.print("<h1>404</h1>");
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req,resp);
    }
}
点击查看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"
         metadata-complete="true">

<!--  一个Servlet要对应一个class  -->
<!--  注册Servl  -->
    <servlet>
        <servlet-name>hello</servlet-name>
        <servlet-class>com.jw02.servlet.HelloServlet</servlet-class>
    </servlet>
<!--  Servl的请求路径  -->
    <servlet-mapping>
        <servlet-name>hello</servlet-name>
        <url-pattern>/hello</url-pattern>
    </servlet-mapping>
<!--404-->
    <servlet>
        <servlet-name>error</servlet-name>
        <servlet-class>com.jw02.servlet.ErroeServlet</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>error</servlet-name>
        <url-pattern>/*</url-pattern>
    </servlet-mapping>
</web-app>

网页刚刚进入的时候,会打开404,因为/也属于/*的范围。输入/hello以后,会打开hello页面。
指定了固有的映射路径优先级最高,如果找不到就会走默认的的处理请求(/*)


把原先的先去掉,改成新的,不然两个都会打包,会比较慢。
image

5、ServletContext

web容器在启动的时候,它会为每个web程序都创建一个对应的ServletContext,它代表了当前的web应用;

5.1、共享数据

我在这个Servlet中保存的数据,可以在另一个servlet中拿到

image

  1. image

  1. image

  1. image

点击查看web.xml相关部分代码
<servlet>
	<servlet-name>hello</servlet-name>
	<servlet-class>com.jw02.servlet.HelloServlet</servlet-class>
</servlet>
<servlet-mapping>
	<servlet-name>hello</servlet-name>
	<url-pattern>/hello</url-pattern>
</servlet-mapping>

<servlet>
	<servlet-name>getc</servlet-name>
	<servlet-class>com.jw02.servlet.GetServlet</servlet-class>
</servlet>
<servlet-mapping>
	<servlet-name>getc</servlet-name>
	<url-pattern>/getc</url-pattern>
</servlet-mapping>
点击查看HelloServlet代码
package com.jw02.servlet;

import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

public class HelloServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//        this.getInitParameter()初始化参数
//        this.getServletConfig()Servlet配置
//        this.getServletContext()Servlet上下文
//        this.getServletName()名字,返回字符串
//        this.getServletInfo()信息,返回字符串
        ServletContext context = this.getServletContext();
        String username = "张张";
        //数据
        context.setAttribute("username",username);
        //将一个数据保存在ServletContext,名字为“username”值为username的值
    }
}
点击查看GetServlet代码
package com.jw02.servlet;

import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

public class GetServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        ServletContext context = this.getServletContext();
        String username = (String) context.getAttribute("username");
        resp.setContentType("text/html;charset=utf-8");//中文不乱码
        resp.getWriter().print("名字:"+username);
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req,resp);
    }
}

5.2、获取初始化参数

image

点击查看web.xml部分代码
<!--  配置一些web应用初始化参数  -->
<context-param>
	<param-name>url</param-name>
	<param-value>jdbc:mysql://localhost:3306/mybatis</param-value>
</context-param>
<servlet>
	<servlet-name>gp</servlet-name>
	<servlet-class>com.jw02.servlet.ServletDemo03</servlet-class>
</servlet>
<servlet-mapping>
	<servlet-name>gp</servlet-name>
	<url-pattern>/gp</url-pattern>
</servlet-mapping>
点击查看ServletDemo03代码
package com.jw02.servlet;

import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

public class ServletDemo03 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        ServletContext context = this.getServletContext();

        String url = context.getInitParameter("url");
        resp.getWriter().print(url);
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req,resp);
    }
}

5.3、请求转发

image

会转发到另一个页面,路径不会变化。

重定向路径会发生变化
image

image

点击查看web.xml部分代码
<servlet>
	<servlet-name>sd4</servlet-name>
	<servlet-class>com.jw02.servlet.ServletDemo04</servlet-class>
</servlet>
<servlet-mapping>
	<servlet-name>sd4</servlet-name>
	<url-pattern>/sd4</url-pattern>
</servlet-mapping>
点击查看ServletDemo04代码
public class ServletDemo04 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        ServletContext context = this.getServletContext();
        System.out.println("ServletDemo04");
        //转发的请求路径
        RequestDispatcher requestDispatcher = context.getRequestDispatcher("/gp");
        requestDispatcher.forward(req,resp);//调用forward实现请求转发
//        context.getRequestDispatcher("/gp").forward(req,resp);

    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req,resp);
    }
}

5.4、读取资源文件

Properties 类表示了一个持久的属性集。Properties 可保存在流中或从流中加载。属性列表中每个键及其对应值都是一个字符串。

resources下新建db.properties
运行后生成在
target/servlet-2-1.0-SNAPSHOT/WEB-INF/classes/db.properties

  • 在java目录下新建properties
  • 在resources目录下新建properties

都被打包到了同一个路径下:classes,俗称这个路径为classpath

image

点击查看web.xml部分代码
<servlet>
	<servlet-name>sd5</servlet-name>
	<servlet-class>com.jw02.servlet.ServletDemo05</servlet-class>
	</servlet>
<servlet-mapping>
	<servlet-name>sd5</servlet-name>
	<url-pattern>/sd5</url-pattern>
</servlet-mapping>
点击查看db.properties代码
username=root
password=123456
点击查看ServletDemo05代码
public class ServletDemo05 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        InputStream is = this.getServletContext().getResourceAsStream("/WEB-INF/classes/db.properties");
        Properties prop = new Properties();
        prop.load(is);
        String username = prop.getProperty("username");
        String password = prop.getProperty("password");
        resp.getWriter().print(username+":"+password);
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req, resp);
    }
}

6、HttpServletResponse

web服务器接收到客户端的http请求,会针对这个请求分别创建一个代表请求HttpServletRequest对象,一个代表响应HttpServletResponse对象。

  • 如果要获取客户端请求过来的参数:找HttpServletRequest
  • 如果要获取客户端响应过来的参数:找HttpServletResponse

6.1、简单分类

负责向浏览器发送数据的方法

ServletOutputStream getOutputStream() throws IOException;
PrintWriter getWriter() throws IOException;

负责向浏览器发送响应头的方法

void setCharacterEncoding(String var1);
void setContentLength(int var1);
void setContentLengthLong(long var1);
void setContentType(String var1);
void setDateHeader(String var1, long var2);
void addDateHeader(String var1, long var2);
void setHeader(String var1, String var2);
void addHeader(String var1, String var2);
void setIntHeader(String var1, int var2);
void addIntHeader(String var1, int var2);
点击查看响应的状态码
int SC_CONTINUE = 100;
int SC_SWITCHING_PROTOCOLS = 101;
int SC_OK = 200;
int SC_CREATED = 201;
int SC_ACCEPTED = 202;
int SC_NON_AUTHORITATIVE_INFORMATION = 203;
int SC_NO_CONTENT = 204;
int SC_RESET_CONTENT = 205;
int SC_PARTIAL_CONTENT = 206;
int SC_MULTIPLE_CHOICES = 300;
int SC_MOVED_PERMANENTLY = 301;
int SC_MOVED_TEMPORARILY = 302;
int SC_FOUND = 302;
int SC_SEE_OTHER = 303;
int SC_NOT_MODIFIED = 304;
int SC_USE_PROXY = 305;
int SC_TEMPORARY_REDIRECT = 307;
int SC_BAD_REQUEST = 400;
int SC_UNAUTHORIZED = 401;
int SC_PAYMENT_REQUIRED = 402;
int SC_FORBIDDEN = 403;
int SC_NOT_FOUND = 404;
int SC_METHOD_NOT_ALLOWED = 405;
int SC_NOT_ACCEPTABLE = 406;
int SC_PROXY_AUTHENTICATION_REQUIRED = 407;
int SC_REQUEST_TIMEOUT = 408;
int SC_CONFLICT = 409;
int SC_GONE = 410;
int SC_LENGTH_REQUIRED = 411;
int SC_PRECONDITION_FAILED = 412;
int SC_REQUEST_ENTITY_TOO_LARGE = 413;
int SC_REQUEST_URI_TOO_LONG = 414;
int SC_UNSUPPORTED_MEDIA_TYPE = 415;
int SC_REQUESTED_RANGE_NOT_SATISFIABLE = 416;
int SC_EXPECTATION_FAILED = 417;
int SC_INTERNAL_SERVER_ERROR = 500;
int SC_NOT_IMPLEMENTED = 501;
int SC_BAD_GATEWAY = 502;
int SC_SERVICE_UNAVAILABLE = 503;
int SC_GATEWAY_TIMEOUT = 504;
int SC_HTTP_VERSION_NOT_SUPPORTED = 505;

6.2、常见应用

  1. 向浏览器输出信息
  2. 下载文件

6.3、

响应:

7、HttpServletRequest

posted @ 2022-10-17 16:38  饺子少蘸醋  阅读(21)  评论(0编辑  收藏  举报