servlet规范来自于JavaEE规范中的一种。
作用:
①在Servlet规范中,指定动态资源文件开发步骤
②在Servlet规范中,指定Http服务器调用动态资源文件规则
③在Servlet规范中,指定Http服务器管理动态资源文件实例对象规则
Servlet接口来自于Servlet规范下的一个接口,这个接口存在Http服务器提供的jar包内。
Tomcat服务器下的lib文件夹中有一个servlet-api.jar存放Servlet接口(javax.servlet.Servlet接口)。
Servlet规范中认为Http服务器能调用的动态资源文件必须是一个Servlet接口实现类。
①创建一个java类继承于HttpServlet父类(servlet-api.jar包内,javax.servlet.http.HttpServlet抽象类,而该类继承了GenericServlet抽象类,该父类实现了Servlet接口),使之成为一个Servlet接口实现类。
为什么不直接继承Servlet接口?
因为Servlet接口很多方法我们不需要使用,一般只需要使用到service()方法,因此通过
继承其他该接口的实现类,可以减少我们需要重写的方法数量。抽象类的作用就是降低接
口实现类对接口的实现难度,将接口中不需要使用的抽象方法交给抽象类进行完成,这样
接口实现类只需要对接口中需要使用的方法进行重写。
Servlet接口中的方法:
1.init()
2.getServletConfig()
3.getServeltInfo()
4.destory()
5.service()
Tomcat根据Servlet规范调用Servlet接口实现类的规则:
1.Tomcat有权自动创建Servlet接口实现类实例对象
Servlet oneServlet = new OneServlet();
2.Tomcat根据实例对象调用service方法处理当前请求
oneServlet.service();
各个类的关系:
oneServlet extends (abstract)HttpServlet entends (abstract)GenericServlet implements Servlet接口
其中GenericServlet抽象类中实现了Servlet接口中我们不需要用到的四个方法:init()、
getServletConfig()、getServletInfo()、destory()。
HttpServlet抽象类中实现了service()方法。
自己定义的子类均可以继承这五个实现后的方法。
②重写HttpServlet父类中的两个方法,doGet()、doPost()。如果浏览器发送请求的方式为get,则调用的是doGet()方法;如果发送请求的方式为post,则调用的是doPost()方法。
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException{}
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException{}
在HttpServlet抽象类中实现的service()方法会根据浏览器请求的方式来调用doGet()或
doPost()方法。这种通过父类决定何种情况下调用子类中方法,是设计模式中的模板设计模
式。
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);//调用子类中的doGet()方法
} else {
long ifModifiedSince;
try {
ifModifiedSince = req.getDateHeader("If-Modified-Since");
} catch (IllegalArgumentException var9) {
ifModifiedSince = -1L;
}
if (ifModifiedSince < lastModified / 1000L * 1000L) {
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);//调用子类中的doPost()方法
} 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);
}
}
③将Servlet接口实现类注册到Tomcat服务器。
网站--->web文件夹--->WEB-INF文件夹--->web.xml文件
oneServlet
com.bjpowernode.controller.OneServlet
oneServlet
/one
1.网站中所有Servlet接口实现类的实例对象,只能由Http服务器负责创建,开发人员不需要手动创建Servlet接口实现类的实例对象。
2.在默认的请求下,Http服务器接收到对于当前Servlet接口实现类第一次请求时自动创建这个Servlet接口实现类的实例对象。
在手动配置情况下,要求Http服务器在启动时自动创建某个Servlet接口实现类的实例对象,在向服务器注册该类时,加上一句:
oneServlet
com.bjpowernode.controller.OneServlet
30
3.在Http服务器运行期间,一个Servlet接口实现类只能被创建出一个实例对象。多个用户访问时,也只有一个实例对象,相当于一个进程多个线程。
4.在Http服务器关闭时刻,自动将网站中所有的Servlet对象进行销毁。
IDEA快速创建Servlet接口的实现类:
①HttpServletResponse接口来自于Servlet规范中,在Tomcat中存在于servlet-api.jar包中。
②HttpServletResponse接口实现类由http服务器负责提供
③HttpServletResponse接口负责将doGet/doPost方法执行结果写入到响应体交给浏览器
④开发人员习惯于将HttpServletResponse接口修饰的对象称为响应对象。
①将执行结果以二进制形式写入到响应体
②设置响应头中content-type属性值,从而控制浏览器使用对应编译器将响应体二进制数据编译为文字、图片、视频、命令
③设置响应头location属性,将一个请求地址赋值给location,从而控制浏览器向指定服务器发送请求
1.通过HttpServletResponse接口实现类对象response获得需要的输出流。
public class OneServlet extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String result = "Hello World";//执行结果
//---------响应对象将结果写入响应体----------start
//1.通过响应对象,向Tomcat索要输出流
PrintWriter out = response.getWriter();
//2.通过输出流,将执行结果以二进制形式写入到响应体
out.write(result);
//不是自己创建的输出流,不需要自己手动关闭
//---------响应对象将结果写入响应体----------end
}
//doGet方法执行完毕
//Tomcat将响应包推送给浏览器
}
2.out.print()方法:
/**
* 问题描述:浏览器接受到数据是2,不是50
* 问题原因:
* out.write方法可以将“字符”,“字符串”,“ASCII码”写入到响应体
* 参数放入数字,计算机认为是ASCII码,而2的ASCII码就是50
*
* 解决方法:实际开发中,都是通过out.print()将真实数据写入到响应体
*
*/
public class TwoServlet extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
int money = 50;
PrintWriter out = response.getWriter();
//out.write(money);
out.print(money);//以后都使用这个方法即可
}
}
3.通过设置context-type属性来控制浏览器的编译格式。默认情况下context-type属性的值为"test"
/**
* 问题描述:Java
Mysql
Html
* 浏览器在接收到响应结果时,将
作为文字内容在窗口展示出来,没有将
当作HTML标签命令来执行
*
* 问题原因:
* 浏览器在接收到响应包之后,根据响应头中的content-type属性的值,
* 来采用对应的编译器对响应体中二进制内容进行编译处理。
*
* 在默认情况下,content-type属性的值为test 即content-type="test"
* 此时浏览器会采用文本编辑器对响应体二进制数据进行解析。
*
* 解决方案:
* 一定要在得到输出流之前,通过响应对象响应头中content-type属性进行一次重新赋值
* 用于制定浏览器采用正确编译器。
*/
public class ThreeServlet extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String result = "Java
Mysql
Html";//既有文字信息,又有HTML标签命令
String result2 = "
红烧排骨
梅菜肘子
糖醋里脊";
//在获得输出流之前,设置content-type属性
response.setContentType("text/html;charset=utf-8");
PrintWriter out = response.getWriter();
out.print(result);
out.print(result2);
}
}
4.设置location属性设置响应头,向其他服务器发起请求。
public class FourServlet extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String result = "http://www.baidu.com";
//通过响应对象,将地址赋值给响应头中location属性
response.sendRedirect(result);//[响应头 location="http://www.baidu.com"]
}
/**
* 浏览器在接收到响应包之后,如果发现响应头中存在location属性,自动通过地址栏location指定网站发送请求
*
* sendRedirect方法远程控制浏览器请求行为【请求地址,请求方式,请求参数】
*/
}
①HttpServletRequest接口来自于Servlet规范中,在Tomcat中存在于servlet-api.jar包中。
②HttpServletRequest接口实现类由http服务器负责提供
③HttpServletRequest接口负责在doGet/doPost方法运行时读取Http请求协议包中的信息。
④开发人员习惯于将HttpServletRequest接口修饰的对象称为请求对象。
①可以读取Http请求协议包中请求行信息
②可以读取保存在Http请求协议包中请求头或者请求体中请求参数信息
③可以代替浏览器向Http服务器申请资源文件调用
1.读取请求行信息。
URI:资源文件精准定位地址,在请求行没有uri这个属性。实际上uri是url中截取的一个字符
串,这个字符串格式"/网站名/资源文件名",uri用于让Http服务器对被访问的资源文件进行定
位。
public class OneServlet extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//1.通过请求对象,读取请求行中url信息
String url = request.getRequestURL().toString();
//2.通过请求对象,读取请求行中method信息
String method = request.getMethod();
System.out.println("URL = " + url);
System.out.println("method = " + method);
//3.通过请求对象,读取请求行中uri信息
/**
* URI:资源文件精准定位地址,在请求行并没有URI这个属性。
* 实际上是URL中截取的一个字符串,这个字符串格式"/网站名/资源文件名"
* URI用于让Http服务器对被访问的资源文件进行定位
*/
String uri = request.getRequestURI();
System.out.println("URI = " + uri);
}
}
2.读请求头里面的值(get方法)
Title
通过超链接访问TwoServlet并且携带请求参数
public class Two1Servlet extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//1.通过请求对象获得请求头中所有请求参数名
Enumeration paramNames = request.getParameterNames();//将所有请求参数名称保存到一个枚举对象进行返回
while(paramNames.hasMoreElements()){
String paramName = (String)paramNames.nextElement();
System.out.println("请求参数: " + paramName);
//2.通过请求对象读取指定的请求参数的值
String value = request.getParameter(paramName);
System.out.println("value: " + value);
}
}
}
3.读请求体里面的值(post方法)
Title
/**
* 问题:
* 以GET方式发送中文参数内容“张三”时,可以得到正常结果
* 以POST方式发送中文参数内容时,得到一堆乱码?????
*
* 原因:
* 浏览器以GET方式发送请求时,请求参数保存在请求头内,在Http请求协议包到达Http服务器之后,第一件事就是进行解码,
* 请求头二进制内容由Tomcat负责解码,Tomcat9.0默认使用utf-8字符集,可以解释一切国家文字。
*
* 浏览器以POST方式发送请求,请求参数保存在请求体内,在Http请求协议包到大Http服务器之后,第一节是就是进行解码,
* 请求体二进制内容由当前请求对象request负责解码。request默认使用[ISO-8859-1]字符集,一个东欧语系字符集,
* 此时如果请求体参数内容是中文,将无法解码,只能得到乱码
*
* 解决方案:
* 在POST请求方式下,读取请求体的内容之前,应该通知请求对象使用utf-8字符集对请求体内容进行一次重新解码。
*/
public class ThreeServlet extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//通知请求对象使用utf-8字符集
request.setCharacterEncoding("utf-8");
//通过请求对象,读取请求体中参数信息
Enumeration paramNames = request.getParameterNames();
while (paramNames.hasMoreElements()){
String paramName = (String) paramNames.nextElement();
String value = request.getParameter(paramName);
System.out.println("从请求体得到参数值 " + value);
}
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//通过请求对象,读取请求头参数信息
String value = request.getParameter("userName");
System.out.println("从请求头的到参数值:" + value);
}
}
①当浏览器以GET方式发送请求时,请求参数保存在请求头,在Http请求协议包到达Http服务
器之后,第一件事就是进行解码。请求头二进制内容由Tomcat负责解码,Tomcat9.0默认使
用UTF-8字符集,可以解释一切国家语言。
②当浏览器以POST方式发送请求时,请求参数保存在请求体,在Http请求协议包到达Http服
务器之后,第一件事就是进行解码。请求体二进制内容由当前请求对象request负责解码,
request默认使用ISO-8859-1字符集,一个东欧语系字符集,此时如果请求体参数内容是中
文,将无法正常解码,只能获得乱码。
①在Http服务器接收到浏览器发送的Http请求协议包之后,自动为当前的Http请求协议包生成一个请求对象和一个响应对象。
②在Http服务器调用doGet/doPost方法时,负责将请求对象和响应对象作为实参传递到方法,确保doGet/doPost方法正确执行。
③在Http服务器准备推送Http响应协议包之前,负责将本次请求关联的请求对象和响应对象销毁。
请求对象和响应对象生命周期贯穿一次请求的处理过程中。
请求对象和响应对象相当于用户在服务端的代言人。
用户一般可以记住网站名,但记不住网站资源名。因此产生了默认欢迎资源文件。
默认欢迎资源文件:用户发送了一个针对某个网站的默认请求时,此时由Http服务器自动从当前网站返回的资源文件。
正常请求:http://localhost:8080/myWeb/index.html
默认请求:http://localhost:8080/myWeb/
Tomcat服务器对于默认欢迎资源文件的定位规则:
①规则位置:Tomcat安装位置/conf/web.xml
②定义语法:
设置当前网站的默认欢迎资源文件的规则:
①规则位置:网站/web/WEB-INF/web.xml
②定义语法:在web.xml文件中添加。
③网站设置自定义默认文件定位规则后,此时Tomcat自带定位规则将失效。
PS:根据动力节点课程整理,如有侵权,联系删除。