HttpServlet源码分析
这个是专门为Http协议准备的,比GenericServlet更适合Http协议的开发
Servlet包下的:
Servlet核心接口,ServletConfig配置信息接口,ServletContext上下文信息接口,Request请求接口,Response响应接口
ServletException异常类,GenericServlet标准类(抽象的)
Http包下的:
HttpServlet核心接口,HttpRequest请求接口,HttpResponse响应接口
HttpRequest请求接口封装了什么:(Request对象)
封装了请求协议的全部内容,Tomcat服务器封装的,将请求协议的全部内容全部解析出来,然后将它们封装到request对象
也就是说,我们只要面向request这个对象编程,就能获取请求协议的全部数据
我们只需要掉方法即可
Response对象同理,它是专门负责向浏览器端传送相应信息的,封装在内部,我们通过操作这个java对象处理完后
再拆包为response信息回传给浏览器
生命周期:
第一次请求:Tomcat通过反射机制创建Servlet对象,调用init方法,并将config信息赋值完成初始化
然后调用service方法处理请求
第二次请求:调用service方法处理请求
第三次请求:调用service方法处理请求
服务器关闭:调用destroy方法做销毁之前的准备动作,然后清理对象,最后关闭服务器
//没有提供init方法,意味着一定会执行父类的init方法
//HttpServlet也没有init方法,会继续执行父类genericServlet的init方法
//父类genericServlet的init方法会调用我们没有重写的无参数的init方法
//这里也没有service方法,执行父类HttpServlet的service方法
//因为父类genericServlet的service方法是抽象的,被HttpServlet给重写了
//调用的是没有http的request和response的service方法
//因为这个才是最原始被重写的方法
//它将这个请求和响应信息强制类型转换为了http的request和response(向下转型,父类转子类)
//这个没有http的service方法会调用httpServlet自己的service方法
//然后调用重载的service方法
//这个重载的方法两个参数都是带有http的
//每次发送请求都会调用这个service方式
//然后回调去req.getMethod()方法,可以获取是get还是post方法或是put方法,都是大写的
//返回值是大写的GET或是POST等七种方法之一
//如果请求方式是其中某一个请求,则执行do某某
//这个方法起到的就是模板方法的骨架作用
//我们编写的HelloServlet直接继承HttpServlet可以直接重写service方法
//但是会享受不到405错误提示,以及所有http协议专属的东西(其实又变成了GenericServlet的近似物)
如果我们发送了get或是post请求,但是没有对应的方法,就会导致405错误
httpServlet的service方法跳转到了doget或dopost方法内,但是我们使用的子类没有重写该方法
导致执行了父类的doget方法,而父类的该方法执行就会抛出405错误
如果自己重写service方法就无法享受到这个405的资源和提示了
第一,我们后续就直接写继承HttpServlet类即可
第二,重写doget或者dopost方法
第三步,将servlet对象配置到我们的web.xml中进行管理
第四步准备表单,form表单准备请求路径即可
HttpServlet源码分析
这个是专门为Http协议准备的,比GenericServlet更适合Http协议的开发
Servlet包下的:
Servlet核心接口,ServletConfig配置信息接口,ServletContext上下文信息接口,Request请求接口,Response响应接口
ServletException异常类,GenericServlet标准类(抽象的)
Http包下的:
HttpServlet核心接口,HttpRequest请求接口,HttpResponse响应接口
HttpRequest请求接口封装了什么:(Request对象)
封装了请求协议的全部内容,Tomcat服务器封装的,将请求协议的全部内容全部解析出来,然后将它们封装到request对象
也就是说,我们只要面向request这个对象编程,就能获取请求协议的全部数据
我们只需要掉方法即可
Response对象同理,它是专门负责向浏览器端传送相应信息的,封装在内部,我们通过操作这个java对象处理完后
再拆包为response信息回传给浏览器
生命周期:
第一次请求:Tomcat通过反射机制创建Servlet对象,调用init方法,并将config信息赋值完成初始化
然后调用service方法处理请求
第二次请求:调用service方法处理请求
第三次请求:调用service方法处理请求
服务器关闭:调用destroy方法做销毁之前的准备动作,然后清理对象,最后关闭服务器
//没有提供init方法,意味着一定会执行父类的init方法
//HttpServlet也没有init方法,会继续执行父类genericServlet的init方法
//父类genericServlet的init方法会调用我们没有重写的无参数的init方法
//这里也没有service方法,执行父类HttpServlet的service方法
//因为父类genericServlet的service方法是抽象的,被HttpServlet给重写了
//调用的是没有http的request和response的service方法
//因为这个才是最原始被重写的方法
//它将这个请求和响应信息强制类型转换为了http的request和response(向下转型,父类转子类)
//这个没有http的service方法会调用httpServlet自己的service方法
//然后调用重载的service方法
//这个重载的方法两个参数都是带有http的
//每次发送请求都会调用这个service方式
//然后回调去req.getMethod()方法,可以获取是get还是post方法或是put方法,都是大写的
//返回值是大写的GET或是POST等七种方法之一
//如果请求方式是其中某一个请求,则执行do某某
//这个方法起到的就是模板方法的骨架作用
//我们编写的HelloServlet直接继承HttpServlet可以直接重写service方法
//但是会享受不到405错误提示,以及所有http协议专属的东西(其实又变成了GenericServlet的近似物)
如果我们发送了get或是post请求,但是没有对应的方法,就会导致405错误
httpServlet的service方法跳转到了doget或dopost方法内,但是我们使用的子类没有重写该方法
导致执行了父类的doget方法,而父类的该方法执行就会抛出405错误
如果自己重写service方法就无法享受到这个405的资源和提示了
第一,我们后续就直接写继承HttpServlet类即可
第二,重写doget或者dopost方法
第三步,将servlet对象配置到我们的web.xml中进行管理
第四步准备表单,form表单准备请求路径即可
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Login</title>
</head>
<body>
<form action="/LoginServlet" method="post">
<h1>用户名</h1><input name="name" type="text">
<h1>密码</h1><input name="password" type="password">
<input type="submit" value="submit">
</form>
</body>
</html>
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Login</title>
</head>
<body>
<form action="/LoginServlet" method="post">
<h1>用户名</h1><input name="name" type="text">
<h1>密码</h1><input name="password" type="password">
<input type="submit" value="submit">
</form>
</body>
</html>
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Index</title>
</head>
<body>
<h1>get请求</h1>
<a href="/HelloServlet">hello</a>
<h1>Post请求</h1>
<form action="/HelloServlet" method="post">
<input type="submit" value="hello">
</form>
</body>
</html>
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Index</title>
</head>
<body>
<h1>get请求</h1>
<a href="/HelloServlet">hello</a>
<h1>Post请求</h1>
<form action="/HelloServlet" method="post">
<input type="submit" value="hello">
</form>
</body>
</html>
package com.bjpowernode.javaweb.servlet;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
public class LoginServlet extends HttpServlet
{
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.setContentType("text/html;charset=UTF-8");
PrintWriter out = resp.getWriter();
out.println("<h1>登陆成功</h1>");
}
}
package com.bjpowernode.javaweb.servlet;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
public class LoginServlet extends HttpServlet
{
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.setContentType("text/html;charset=UTF-8");
PrintWriter out = resp.getWriter();
out.println("<h1>登陆成功</h1>");
}
}
package com.bjpowernode.javaweb.servlet;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
public class HelloServlet extends HttpServlet
{
//通过无参数构造方法创建对象
public HelloServlet() {}
//改为当前端发送get请求,执行get的
//前端发送post请求,执行post的
//因此我们可以通过405错误判断定位到发送的前端请求与后端接收方法不一致
//通过源码可知,只要HttpServlet的doPost和doGet执行了必然405错误
//子类一错就跳到父类报错
//前端发送的请求方法和后端的接受应当一致
//前端的请求应该与后端一致
//不要同时重写两个方法,或是不同方法互相调用
//这样确实避免了405,但是不科学
//如果我们同时重写了doGet和doPost,还不如直接重写service方法
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
PrintWriter out = response.getWriter();
out.println("<h1>doGet</h1>");
}
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
PrintWriter out = response.getWriter();
out.println("<h1>doPost</h1>");
}
// @Override
// protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// response.setContentType("text/html;charset=UTF-8");
// PrintWriter out = response.getWriter();
// out.println("<h1>Hello</h1>");
// }
}
package com.bjpowernode.javaweb.servlet;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
public class HelloServlet extends HttpServlet
{
//通过无参数构造方法创建对象
public HelloServlet() {}
//改为当前端发送get请求,执行get的
//前端发送post请求,执行post的
//因此我们可以通过405错误判断定位到发送的前端请求与后端接收方法不一致
//通过源码可知,只要HttpServlet的doPost和doGet执行了必然405错误
//子类一错就跳到父类报错
//前端发送的请求方法和后端的接受应当一致
//前端的请求应该与后端一致
//不要同时重写两个方法,或是不同方法互相调用
//这样确实避免了405,但是不科学
//如果我们同时重写了doGet和doPost,还不如直接重写service方法
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
PrintWriter out = response.getWriter();
out.println("<h1>doGet</h1>");
}
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
PrintWriter out = response.getWriter();
out.println("<h1>doPost</h1>");
}
// @Override
// protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// response.setContentType("text/html;charset=UTF-8");
// PrintWriter out = response.getWriter();
// out.println("<h1>Hello</h1>");
// }
}
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="https://jakarta.ee/xml/ns/jakartaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="https://jakarta.ee/xml/ns/jakartaee https://jakarta.ee/xml/ns/jakartaee/web-app_6_0.xsd"
version="6.0">
<servlet>
<servlet-name>HelloServlet</servlet-name>
<servlet-class>com.bjpowernode.javaweb.servlet.HelloServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>HelloServlet</servlet-name>
<url-pattern>/HelloServlet</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>LoginServlet</servlet-name>
<servlet-class>com.bjpowernode.javaweb.servlet.LoginServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>LoginServlet</servlet-name>
<url-pattern>/LoginServlet</url-pattern>
</servlet-mapping>
</web-app>
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="https://jakarta.ee/xml/ns/jakartaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="https://jakarta.ee/xml/ns/jakartaee https://jakarta.ee/xml/ns/jakartaee/web-app_6_0.xsd"
version="6.0">
<servlet>
<servlet-name>HelloServlet</servlet-name>
<servlet-class>com.bjpowernode.javaweb.servlet.HelloServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>HelloServlet</servlet-name>
<url-pattern>/HelloServlet</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>LoginServlet</servlet-name>
<servlet-class>com.bjpowernode.javaweb.servlet.LoginServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>LoginServlet</servlet-name>
<url-pattern>/LoginServlet</url-pattern>
</servlet-mapping>
</web-app>