目录
1.6、通过继承HttpServlet实现Servlet程序
1、Servlet技术
1.1、什么是Servlet
-
Servlet是JavaEE规范之一。规范就是接口
-
Servlet是JavaWeb三大组件之一。三大组件分别是:Servlet程序、Filter过滤器、Listener监听器。
-
Servlet是运行在服务器上的一个java小程序,它可以接收客户端发送过来的请求,并响应数据给客户端。
1.2、手动实现Servlet程序
-
编写一个类去实现Servlet接口
package com.dzg.servlet; import javax.servlet.*; import java.io.IOException; /** * @author DiZhiGang * @version 1.0 */ public class HelloServlet implements Servlet { @Override public void init(ServletConfig servletConfig) throws ServletException { } @Override public ServletConfig getServletConfig() { return null; } /** * servlet方法是专门用来处理请求和响应的 * @param servletRequest * @param servletResponse * @throws ServletException * @throws IOException */ @Override public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException { System.out.println("hello servlet被访问了!"); } @Override public String getServletInfo() { return null; } @Override public void destroy() { } }
-
实现Servlet方法,处理请求,并响应数据
-
到web.xml中去配置Servlet程序的访问地址
<?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.dzg.servlet.HelloServlet</servlet-class> </servlet> <!-- servlet-mapping标签给servlet程序配置访问地址 --> <servlet-mapping> <!-- servlet-name标签的作用是告诉服务器,我当前配置的地址是给那个Servlet程序使用 --> <servlet-name>HelloServlet</servlet-name> <!-- url-pattern标签配置访问地址 /斜杠在服务器解析的时候,表示地址为:http://ip:port/工程路径 /hello 表示地址为 http://ip:port/工程路径/hello --> <url-pattern>/hello</url-pattern> </servlet-mapping> </web-app>
-
运行实例!
-
在上述路径后+/hello:
常见的错误:url-pattern中配置的路径没有以斜杠打头。
错误描述为:Invalid <url-pattern> hello in servlet mapping
1.3、url地址到Servlet程序的访问
1.4、Servlet的生命周期
-
执行Servlet构造器方法
-
执行init初始化方法
第一二步是在第一次访问的时候创建Servlet程序会调用。
-
执行service方法
第三步每次访问都会调用
-
执行destroy销毁方法
第四步,在web工程停止的时候调用。
1.5、GET和POST请求的分发处理
在web文件夹下写一个a.html,写一个表单,加一个提交按钮,方法为get
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<form action="http://localhost:8080/06_servlet/hello" method="get">
<input type="submit">
</form>
</body>
</html>
运行实例后
将get改为post请求后提交
和上图结果相同
我们如何知道请求方式为post还是get
在service方法中获取:
@Override
public 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();
System.out.println(method);
}
运行后访问a.html得到:
所以我们可以判断是post还是get请求,可以用if语句来判断,分别完成对应的代码:
@Override
public 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)){
System.out.println("get请求");
}else if("POST".equals(method)){
System.out.println("post请求");
}
}
当post或者get请求中有很多语句需要执行时,service方法会变得非常臃肿,所以我们可以写两个方法便于维护!在service方法中直接调用这两个方法即可。
/**
* 做get请求的操作
*/
public void doGet(){
System.out.println("get请求");
System.out.println("get请求");
System.out.println("get请求");
}
/**
* 做get请求的操作
*/
public void doPost(){
System.out.println("post请求");
System.out.println("post请求");
System.out.println("post请求");
}
1.6、通过继承HttpServlet实现Servlet程序
一般在实际项目开发中,都是使用继承HttpServlet类的方式去实现Servlet程序。
-
编写一个类去继承HttpServlet类
-
根据业务需要重写doGet或doPost方法
package com.dzg.servlet; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; /** * @author DiZhiGang * @version 1.0 */ public class HelloServlet2 extends HttpServlet { /** * doGet()在get请求的时候调用 * @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()在post请求的时候调用 * @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> <servlet-name>HelloServlet2</servlet-name> <servlet-class>com.dzg.servlet.HelloServlet2</servlet-class> </servlet> <servlet-mapping> <servlet-name>HelloServlet2</servlet-name> <url-pattern>/hello2</url-pattern> </servlet-mapping>
-
将a.html的路径/hello改为/hello2后再次运行
1.7、使用IDEA创建Servlet程序
选择你的包名右键new
记得修改下面的内容:package(类在哪个包下)、class(全类名)和Name(Servlet-name)。对勾不要打,默认使用3.0以上的注解配置
然后类和Servlet标签会自动被创建,只需要再写下servlet-mapping绑定即可
将a.html中的工程路径改为hello3再次运行提交后得到结果
1.8、Servlet类的继承体系
2、ServletConfig类
ServletConfig类从类名上来看,就知道是Servlet程序的配置信息类
Servlet程序和ServletConfig对象都是由Tomcat负责创建,我们负责使用。
Servlet程序默认是第一次访问的时候创建,ServletConfig是每个Servlet程序创建时,就创建一个对应的ServletConfig对象。
2.1、ServletConfig类的三大作用
-
可以获取Servlet程序的别名servlet-name的值
-
获取初始化参数init-param
-
获取ServletContext对象
@Override
public void init(ServletConfig servletConfig) throws ServletException {
System.out.println("2 init初始化方法");
// 1. 可以获取Servlet程序的别名servlet-name的值
System.out.println("HelloServlet程序的别名是:"+servletConfig.getServletName());
// 2. 获取初始化参数init-param
System.out.println("初始化参数url的值是:"+servletConfig.getInitParameter("url"));
System.out.println("初始化参数username的值是:"+servletConfig.getInitParameter("username"));
// 3. 获取ServletContext对象
System.out.println(servletConfig.getServletContext());
}
3、ServletContext类
3.1、什么是ServletContext?
-
ServletContext是一个接口,它表示Servlet上下文对象
-
一个web工程,只有一个ServletContext对象实例
可以创建两个对象分别取其地址,发现相同
-
ServletContext对象是一个域对象
-
ServletContext是在web工程部署启动的时候创建。在wenb工程停止的时候销毁
-
域对象, 是可以像Map一样存取数据的对象,叫域对象
这里的域指的是存取数据的操作范围。
-
存数据 | 取数据 | 删除数据 | |
---|---|---|---|
Map | put() | get() | remove() |
域对象 | setAttribute() | getAttribute() | removeAttribute() |
3.2、ServletContext类的四个作用
-
获取web.xml中配置的上下文参数context-param
先在web.xml中配置
<!-- context-param是上下文参数(它属于整个web工程)--> <context-param> <param-name>username</param-name> <param-value>context</param-value> </context-param> <context-param> <param-name>password</param-name> <param-value>dzg</param-value> </context-param>
然后去获取:
ServletContext context = getServletConfig().getServletContext(); String username = context.getInitParameter("username"); System.out.println("context-param参数username的值是:"+username); System.out.println("context-param参数password的值是:"+context.getInitParameter("password"));
-
获取当前的工程路径,格式:/工程路径
System.out.println(context.getContextPath());
-
获取工程部署后在服务器硬盘上的绝对路径
/** * /被服务器解析地址为http://ip:port/工程名/ 映射到IDEA代码的web目录 */ System.out.println(context.getRealPath("/"));
-
像map一样存取数据
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { ServletContext context = getServletContext(); context.setAttribute("key1","value1"); System.out.println("Context1中获取域数据key1的值是:"+context.getAttribute("key1")); }
4、HTTP协议
4.1、什么是HTTP协议
协议是指双方,或多方,相互约定好,大家都需要遵守规则,叫协议。所谓HTTP协议,就是指,客户端和服务端之间通信时,发送的数据,需要遵守的规则,叫HTTP协议。HTTP协议中的数据又叫报文。
4.2、请求的HTTP协议格式
客户端给服务器发送数据叫请求
服务器给客户端回传数据叫响应
请求又分GET请求和POST请求两种
4.2.1、GET请求
-
请求行
-
请求的方式 GET
-
请求的资源路径[+?+请求参数]
-
请求的协议的版本号 HTTP/1.1
-
-
请求头
-
key:value 组成 不同的键值对,表示不同的含义。
-
4.2.2、POST请求
-
请求行
-
请求的方式 GET
-
请求的资源路径[+?+请求参数]
-
请求的协议的版本号 HTTP/1.1
-
-
请求头
-
key:value 组成 不同的键值对,表示不同的含义。
-
空行
-
-
请求体 --->就是发送给服务器的数据
4.2.3、常用的请求头的说明
-
Accept: 表示客户端可以接收的数据类型
-
Accept-Language: 表示客户端可以接收的语言类型
-
User-Agent: 表示客户端浏览器的信息
-
Host: 表示请求时的服务器ip和端口号
4.2.4、哪些是GET请求,哪些是POST请求
-
GET请求有哪些:
-
form标签 method=get
-
a标签
-
link标签引入css
-
Script标签引入js文件
-
img标签引入图片
-
iframe引入html页面
-
在浏览器地址栏中输入地址后敲回车
-
-
POST请求有哪些:
-
form标签 method=post
-
4.3、响应的HTTP协议格式
-
相应行
-
响应的协议和版本号
-
响应状态码
-
响应状态描述符
-
-
响应头
-
key:value 不同的响应头,有其不同的含义
空行
-
-
响应体 -----> 就是回传给客户端的数据
4.4、常用的响应码说明
-
200 表示请求成功
-
302 表示请求重定向
-
404 表示请求服务器已经收到了,但是你要的数据不存在(请求地址错误)
-
500 表示服务器已经收到请求,但是服务器内部错误(代码错误)
4.5、MIME类型说明
MIME是HTTP协议中数据类型。
MIME的英文全称是Mutipurpose Internet Mail Extensions 多功能Internet邮件扩充服务。MIME类型的格式是“大类型/小类型”,并与某一种文件的扩展名相对应。
常见的MIME类型
超文本标记语言文本 | .html,.html text/html |
---|---|
普通文本 | .txt text/plain |
RTF文本 | .rtf application/rtf |
GIF图形 | .gif image/gif |
JPEG图形 | .jpeg,.jpg image/jpeg |
au声音文件 | .au audio/basic |
MIDI音乐文件 | .mid,.midi audio/midi,audio/x-midi |
RealAudio音乐文件 | .ra, .ram audio/x-pn-realaudio |
MPEG文件 | .mpg,.mpeg video/mpeg |
AVI文件 | .avi video/x-msvideo |
GZIP文件 | .gz application/x-gzip |
TAR文件 | .tar application/x-tar |
5、HttpServletRequest类
5.1、HttpServletRequest类有什么作用。
每次只要有请求进入Tomcat服务器,Tomcat服务器就会把请求过来的HTTP协议信息解析好封装到Request对象中。然后传递到service方法(doGet和doPost)中给我们使用。我们可以通过HttpServletRequest对象,获取到所有请求的信息。
5.2、HttpServletRequest类的常用方法
-
getRequestURI() 获取请求的资源路径
-
getRequestURL() 获取请求的统一资源定位符(绝对路径)
-
getRemoteHost() 获取客户端的ip地址
-
getHeader() 获取请求头
-
getParamter() 获取请求的参数
-
getParamterValues() 获取请求的参数(多个值的时候使用)
-
getMethod() 获取请求的方式GET或POST
-
setAttribute(key,value); 设置域数据
-
getAttribute(key); 获取域数据
-
getRequestDispatcher() 获取请求转发对象
举例几个:
public class RequestAPIServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // - getRequestURI() 获取请求的资源路径 System.out.println("URI =>"+req.getRequestURI()); // - getRequestURL() 获取请求的统一资源定位符(绝对路径) System.out.println("URL =>"+req.getRequestURL()); // - getRemoteHost() 获取客户端的ip地址 /** * 在IDEA中,使用localhost或127.0.0.1访问时,得到的客户端ip地址是127.0.0.1或者0:0:0:0:0:0:0:1 * 使用真实ip访问时,得到的也是真实的ip */ System.out.println("客户端ip地址 =>"+req.getRemoteHost()); // - getHeader() 获取请求头 System.out.println("获取请求头User-Agent ==>"+req.getHeader("User-Agent")); // - getMethod() 获取请求的方式GET或POST System.out.println("请求的方式 ==>"+req.getMethod()); } }
5.3、如何获取请求参数
public class ParameterServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//获取请求参数
String username = req.getParameter("username");
String password = req.getParameter("password");
String[] hobby = req.getParameterValues("hobby");
System.out.println("用户名:"+username);
System.out.println("密码:"+password);
System.out.println("兴趣爱好:"+ Arrays.asList(hobby));
}
}
post请求一样,但是post请求体默认输入中文时会乱码
//设置请求体的字符集为utf-8,从而解决post请求的中文乱码问题,此方法必须在请求参数前使用
req.setCharacterEncoding("UTF-8");
5.4、请求的转发
什么是请求的转发?
请求转发是指:服务器收到请求后,从一次资源跳转到另一个资源的操作叫做请求转发。
可以去创建两个类Servlet1和Servlet2来用代码实现一下
Servlet1:
public class Servlet1 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//获取请求的参数(办事的材料)查看
String username = req.getParameter("username");
System.out.println("在Servlet1(柜台1)中查看参数(材料):"+username);
//给材料,盖一个章,并传递到Servlet2(柜台2)去查看
req.setAttribute("key1","柜台1的章");
//问路:Servlet2(柜台2)怎么走
/**
* 请求转发必须要以/打头,/表示地址为:http://ip:port/工程名/,映射到IDEA代码的web目录
*/
RequestDispatcher requestDispatcher = req.getRequestDispatcher("/servlet2");
//走向servlet2(柜台2)
requestDispatcher.forward(req,resp);
}
}
Servlet2:
public class Servlet2 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//获取请求的参数(办事的材料)查看
String username = req.getParameter("username");
System.out.println("在Servlet2(柜台2)中查看参数(材料):"+username);
//查看柜台1是否有盖章
Object key1 = req.getAttribute("key1");
System.out.println("柜台1是否有章:"+key1);
//处理自己的业务
System.out.println("servlet2处理自己的业务");
}
}
记得在请求时添加参数?后加参数=xxx
请求转发的特点:
-
浏览器地址栏没有变化
-
他们是一次请求
-
他们共享Request域中的数据
-
可以转发到WEB-INF目录下
-
是否可以访问工程以外的资源
5.5、base标签的作用
<!-- base标签设置页面相对路径工作时参照的地址
href 属性就是参数的地址值-->
<base href="http://localhost:8080/07_servlet/a/b/">
5.6、Web中的相对路径和绝对路径
在javaweb中,路径分为相对路径和绝对路径两种:
-
相对路径是:
. 表示当前目录
.. 表示上一级目录
资源名 表示当前目录/资源名
-
绝对路径:
http://ip:port/工程路径/资源路径
5.7、web中/斜杠的不同意义
在web中/是一种绝对路径。
/如果被浏览器解析,得到的路径是http://ip:port
-
<a href=" /">斜杠 < /a >
/如果被服务器解析,得到的路径是http://ip:port/工程路径
-
<url-pattern>/servlet1</url-pattern>
-
servletContext.getRealPath("/");
-
request.getRequestDispatcher("/").
特殊情况:response.sendRediect("/"); 把斜杠发送给浏览器解析。得到http://ip:port/
6、HttpServletResponse类
6.1、HttpServletResponse类的作用
HttpServletResponse类和HttpServletRequest类一样。每次请求进来,Tomcat服务器都会创建一个Response对象传递给Servlet程序去使用。HttpServletRequest表示请求过来的信息,HttpServletResponse表示所有响应的信息。
我们如果需要设置返回给客户端的信息,都可以通过HttpServletResponse对象来进行设置。
6.2、两个输出流的说明
-
字节流 getOutputStream(); 常用于下载(传递二进制数据)
-
字符流 getWriter(); 常用于回传字符串(常用)
两个流只能同时使用一个
使用了字节流,就不能再使用字符流,反之亦然,否则会报错。
6.3、如何往客户端回传数据
要求:往客户端回传 字符串 数据。
public class ResponseIOServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//要求:往客户端回传 字符串 数据。
PrintWriter writer = resp.getWriter();
writer.println("response's content!!!");
}
}
6.4、响应的乱码解决
不光要改服务器的字符集编码,还需要通过响应头,设置浏览器也使用UTF-8字符集
-
第一种方法:
public class ResponseIOServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println(resp.getCharacterEncoding());
//设置服务器字符集为UTF-8
resp.setCharacterEncoding("UTF-8");
System.out.println(resp.getCharacterEncoding());
//通过响应头,设置浏览器也使用UTF-8字符集
resp.setHeader("Context-Type","text/html;charset=UTF-8");
//要求:往客户端回传 字符串 数据。
PrintWriter writer = resp.getWriter();
writer.println("response's content!!!");
}
}
-
第二种方法://此方法一定要在获取流对象前使用才有效!!!
//它会同时设置服务器和客户端都使用UTF-8字符集,还设置了响应头。 resp.setContentType("text/html;charset=UTF-8");
6.5、请求重定向
请求重定向,是指客户端给服务器发请求,然后服务器告诉客户端说。我给你一些地址,你去新地址访问,叫做请求重定向。(因为之前的地址可能已经被废弃)。
6.5.1、请求重定向的第一种方案:
//设置响应状态码302,表示重定向,(已搬迁)
resp.setStatus(302);
//设置响应头、说明 新的地址在哪里
resp.setHeader("Location","http://localhost:8080/07_servlet/response2");
创建两个类Response1/2执行上述步骤:
public class Response1 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("response1到此一游");
//设置响应状态码302,表示重定向,(已搬迁)
resp.setStatus(302);
//设置响应头、说明 新的地址在哪里
resp.setHeader("Location","http://localhost:8080/07_servlet/response2");
}
}
public class Response2 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println(req.getAttribute("key1"));
resp.getWriter().write("response2's result!");
}
}
请求重定向的特点:
-
浏览器地址栏会发生变化;
-
是两次请求;
-
不共享Request域中数据;
-
不能访问WEB-INF下的资源
-
可以访问工程外的资源
6.5.2、请求重定向的第二种方案(推荐使用):
只需一行代码即可。
resp.sendRedirect("http://localhost:8080");
如有错误请评论指正!!!记录不易,希望可以点赞哈!