6、Servlet
6.1、简介
- sun 公司开发动态 web 的一门技术
- 该公司在这些 api 中提供一个接口叫做:Servlet
- 开发一个 Servlet 程序,只需要完成两个小步骤:
- 编写一个类,实现 Servlet 接口
- 把开发好的 java 类部署到 web 服务器当中
把实现了 Servlet 接口的 java 程序叫做:Servlet
6.2、HelloServlet
Servlet 接口 sun 公司有两个默认的实现类:HttpServlet、GenericServlet
-
构建一个maven项目
-
构建子项目
-
导入maven依赖:官网https://mvnrepository.com/
<dependency> <groupId>javax.servlet</groupId> <artifactId>javax.servlet-api</artifactId> <version>4.0.1</version> </dependency> <dependency> <groupId>javax.servlet.jsp</groupId> <artifactId>jsp-api</artifactId> <version>2.2</version> </dependency>
注意:有时候可能会爆红,则要手动导入
-
编写一个 Servlet 程序
-
编写一个普通类
-
实现 Servlet 接口,这里只继承 HttpServlet
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("HelloServlet"); } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { super.doPost(req, resp); } }
-
-
编写 Servlet 的映射
由于我们写的是java程序,但是要通过浏览器访问,而浏览器需要连接web服务器,所有我们需要在web服务中注册我们写的 Servlet,还需要给他一个浏览器能够访问的路径
<!--注册Servlet--> <servlet> <servlet-name>HelloServlet</servlet-name> <servlet-class>com.mei.servlet.HelloServlet</servlet-class> </servlet> <!--Servlet请求路径--> <servlet-mapping> <servlet-name>HelloServlet</servlet-name> <url-pattern>/hello</url-pattern> </servlet-mapping>
-
配置Tomcat
-
启动测试
6.3、Servlet运行原理
Servlet 是由web服务器调用,web收到浏览器请求之后,会:
6.4、Mapping问题
-
一个Servlet可以指定一个映射路径
-
一个Servlet可以指定多个映射路径
-
一个Servlet可以指定通用映射路径
<!--Servlet请求路径--> <servlet-mapping> <servlet-name>HelloServlet</servlet-name> <url-pattern>/hello/*</url-pattern> </servlet-mapping>
-
默认请求路径
<!--Servlet请求路径 会把首页index.xml干掉 --> <servlet-mapping> <servlet-name>HelloServlet</servlet-name> <url-pattern>/*</url-pattern> </servlet-mapping>
-
指定一些后缀或者前缀等等
<!--可以自定义后缀实现请求映射 注意点:*前面不能加项目映射路径 --> <!--Servlet请求路径--> <servlet-mapping> <servlet-name>HelloServlet</servlet-name> <url-pattern>*.happy</url-pattern> </servlet-mapping>
-
优先级问题
指定了固定的映射路径优先级高,如果找不到就会走默认的请求
6.5、ServletContext
web 容器在启动的时候,它会为每一个web程序都创建一个对应的ServletContext对象,它代表了当前的web应用;
6.5.1、共享数据
-
我在这个servlet保存的数据,可以在另外一个servlet中拿到
-
创建一个放置数据的类
public class HelloServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //this.getInitParameter() 初始化参数 //this.getServletConfig() Servlet配置 //this.getServletContext() Servlet上下文 ServletContext context = this.getServletContext(); String username = "msh"; //数据 context.setAttribute("username",username); //将一个数据保存到了 ServletContext 中,名字为 username,值为 username } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { doGet(req, resp); } }
-
创建一个读取数据的类
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"); resp.setCharacterEncoding("utf-8"); resp.getWriter().print(username); } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { doGet(req, resp); } }
-
配置web.xml
<servlet> <servlet-name>hello</servlet-name> <servlet-class>com.mei.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.mei.servlet.GetServlet</servlet-class> </servlet> <servlet-mapping> <servlet-name>getc</servlet-name> <url-pattern>/getc</url-pattern> </servlet-mapping>
6.5.2、获取初始化参数
<!--配置一些web应用初始化的参数-->
<context-param>
<param-name>url</param-name>
<param-value>jdbc:mysql://localhost:3306/mybatis</param-value>
</context-param>
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext context = this.getServletContext();
String url = context.getInitParameter("url");
resp.getWriter().print(url);
}
结果:
6.5.3、请求转发
<servlet>
<servlet-name>sd</servlet-name>
<servlet-class>com.mei.servlet.ServletDemo2</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>sd</servlet-name>
<url-pattern>/sd</url-pattern>
</servlet-mapping>
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext context = this.getServletContext();
//RequestDispatcher requestDispatcher = context.getRequestDispatcher("/gp"); //转发的请求路径
//requestDispatcher.forward(req,resp); //调用forward实现请求转发
context.getRequestDispatcher("/gp").forward(req,resp);
}
结果(和上面的一样):
6.5.4、读取资源文件
properties
<servlet>
<servlet-name>sd1</servlet-name>
<servlet-class>com.mei.servlet.ServletDemo3</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>sd1</servlet-name>
<url-pattern>/sd1</url-pattern>
</servlet-mapping>
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);
}
结果:
6.6、HttpServletResponse
web服务器接收到客户端的http请求,针对这个请求,分别创建一个代表请求的HttpServletRequest对象,代表响应的一个HttpServletResponse;
- 如果需要获取客户端请求过来的参数:找HttpServletRequest
- 如果要给客户端响应一些信息:找HttpServletResponse
简单分类
-
负责向浏览器发送数据的方法
ServletOutputStream getOutputStream() throws IOException; PrintWriter getWriter() throws IOException;
-
负责向浏览器发送响应头的方法
//ServletResponse的方法 void setCharacterEncoding(String var1); void setContentLength(int var1); void setContentLengthLong(long var1); void setContentType(String var1); //HttpServletResponse的方法 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);
6.6.1、下载文件
- 要获取下载文件的路径
- 下载的文件名是啥
- 设置想办法让浏览器能够支持下载我们需要的东西
- 获取下载文件的输入流
- 创建缓冲区
- 获取OutputStream对象
- 将FileOutputStream流写入到buffer缓冲区
- 使用OutputStream将缓冲区中的数据输出到客户端
实现:
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//1. 要获取下载文件的路径
//String realPath = this.getServletContext().getRealPath("/首页.png"); // getRealPath:获取绝对地址
String realPath = this.getServletContext().getRealPath("/WEB-INF/classes/首页.png");
System.out.println("下载文件的路径:"+realPath);
//2. 下载的文件名是啥
String fileName = realPath.substring(realPath.lastIndexOf("\\") + 1);// substring:截取字符串
//3. 设置想办法让浏览器能够支持(Content-Disposition)下载我们需要的东西,中文文件名URLEncoder.encode(string,编码)编码,否则有可能乱码
resp.setHeader("Content-Disposition","attachment;filename="+ URLEncoder.encode(fileName, "UTF-8")); // web下载文件的头信息:"Content-Disposition" attachment:附件
//4. 获取下载文件的输入流
FileInputStream inputStream = new FileInputStream(realPath); // 把文件变成流
//5. 创建缓冲区
int len = 0;
byte[] buffer = new byte[1024];
//6. 获取OutputStream对象
ServletOutputStream outputStream = resp.getOutputStream();
//7. 将FileOutputStream流写入到buffer缓冲区,使用OutputStream将缓冲区中的数据输出到客户端
while((len = inputStream.read(buffer,0,buffer.length)) != -1){
outputStream.write(buffer,0,len);
}
//8. 关闭流,保证安全
outputStream.flush();
inputStream.close();
outputStream.close();
}
<servlet>
<servlet-name>Filedown</servlet-name>
<servlet-class>com.mei.servlet.FileServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>Filedown</servlet-name>
<url-pattern>/down</url-pattern>
</servlet-mapping>
结果:
6.6.2、验证码功能
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//如何让浏览器5秒自动刷新一次
resp.setHeader("refresh","3");
//在内存中创建一个图片
BufferedImage image = new BufferedImage(80, 20, BufferedImage.TYPE_INT_RGB);
//得到图片
Graphics2D g = (Graphics2D) image.getGraphics(); //相当于创建了一个画笔
//设置图片的背景颜色
g.setColor(Color.white); //填充的颜色
g.fillRect(0,0,80,20); //填充的位置
//给图片写数据
g.setColor(Color.BLUE);
g.setFont(new Font(null,Font.BOLD,20)); //字体
g.drawString(makeNum(),0,20);
//告诉浏览器这个请求用图片的方式打开
resp.setContentType("image/jpeg");
//网站存在缓存,不让浏览器缓存
resp.setDateHeader("expires",-1);
resp.setHeader("Cache-Control","no-cache");
resp.setHeader("Pragma","no-cache");
//把图片写给浏览器
ImageIO.write(image,"jpeg",resp.getOutputStream());
}
//生成随机数
private String makeNum(){
Random random = new Random();
String num = random.nextInt(9999999) + ""; //随机数
StringBuffer sb = new StringBuffer();//把随机数往外写
for (int i = 0; i < 7-num.length(); i++) {
sb.append("0");
}
num = sb.toString() + num;
return num;
}
<servlet>
<servlet-name>ImageServlet</servlet-name>
<servlet-class>com.mei.servlet.ImageServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>ImageServlet</servlet-name>
<url-pattern>/img</url-pattern>
</servlet-mapping>
结果:
6.6.3、重定向
void sendRedirect(String var1) throws IOException;
测试:
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.sendRedirect("/img"); //重定向到 http://localhost:8080/img
}
6.6.4、用户登录
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//处理请求
String username = req.getParameter("username");// 获取请求的参数
String password = req.getParameter("password");// 获取请求的参数
System.out.println("username:"+username+";password:"+password);
//重定向一定要注意路径
resp.sendRedirect("/success.jsp");
}
<form action="${pageContext.request.contextPath}/login" method="get">
用户名:<input type="text" name="username"> <br>
密码:<input type="password" name="password"> <br>
<input type="submit" value="登录">
</form>
<servlet>
<servlet-name>test</servlet-name>
<servlet-class>com.mei.servlet.RequestTest</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>test</servlet-name>
<url-pattern>/login</url-pattern>
</servlet-mapping>
6.7、HttpServletRequest
6.7.1、获取前端的参数,请求转发
第一个和第四个比较重要,第一个返回 String,第四个返回 String[]
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 解决后端接收中文乱码问题
req.setCharacterEncoding("utf-8");
resp.setCharacterEncoding("utf-8");
String username = req.getParameter("username");
String password = req.getParameter("password");
String[] hobbies = req.getParameterValues("hobbies");
System.out.println("username:"+username+",password:"+password+",hobbies:"+Arrays.toString(hobbies));
// 通过请求转达
req.getRequestDispatcher("/success.jsp").forward(req,resp);
}
<div style="text-align: center">
<%-- 这里表单提交的意思,以 post 方式提交表单,提交到我们的 login 请求--%>
<form action="${pageContext.request.contextPath}/login" method="post">
用户名:<input type="text" name="username"> <br>
密码:<input type="password" name="password"> <br>
爱好:<input type="checkbox" name="hobbies" value="开心">😀
<input type="checkbox" name="hobbies" value="流泪">😭
<input type="checkbox" name="hobbies" value="研究">🧐
<input type="checkbox" name="hobbies" value="调皮">😝
<br>
<input type="submit" value="登录">
</form>
</div>
界面:
后端显示:
重定向和转发的区别:
- 请求转发的时候,url不会产生变化
- 重定向,url会发生变化
7、Cookie、Session
7.1、会话
会话:用户打开一个浏览器,点击了很多超链接,访问了多个web资源,关闭浏览器,这个过程称之为会话
**有状态会话:**来过一次之后,就知道你来过了
一个网站,怎么证明你来过?
客户端 服务端
- 服务端给客户端一个 信件,客户端下次访问服务端带上信件就可以了:cookie
- 服务器等级来过了,下次你来的时候我匹配你:session
7.2、保存会话的两种技术
cookie
- 客户端技术,响应,请求
session
- 服务端技术,利用这个技术,可以保存用户的会话信息,我们可以把数据放在session中
7.3、Cookie
- 从请求中拿到cookie信息
- 服务端响应客户端cookie
Cookie[] cookies = req.getCookies(); // 获得cookie
cookie.getName(); // 获得cookie中的key
cookie.getValue(); // 获得cookie中的value
new Cookie("lastLoginTime", System.currentTimeMillis()+""); // 新建一个cookie
cookie.setMaxAge(24*60*60); // 设置cookie的有效期
resp.addCookie(cookie); // 响应给客户端一个cookie
编码与解码
-
将字符转换为字节的方式称为编码
URLEncoder.encode("Happy","UTF-8")
-
将字节转换为字符的方式称为解码
URLDecoder.decode(cookie.getValue(),"utf-8")
7.4、Session(重点)
什么是session?
- 服务器会给每一个用户(浏览器)创建一个Session对象
- 一个Session独占一个浏览器,只要浏览器没关闭,就会一直存在
- 用户登录之后,整个网站访问它都可以访问 --> 保存用户信息,保存购物车信息…
Session和Cookie的区别
- Cookie是把用户的数据写给用户的浏览器,浏览器保存
- Session把用户的数据写到用户独占Session中,服务器端保存(保存重要的信息,减少服务器资源的浪费)
- Session对象由服务器创建
使用场景
- 登录用户信息
- 购物车信息
- 整个网站中经常会用到的数据
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
req.setCharacterEncoding("utf-8");
resp.setCharacterEncoding("utf-8");
resp.setContentType("text/html;charset=utf-8");
// 得到Session
HttpSession session = req.getSession();
// 给Session中存东西
session.setAttribute("name",new Person("Happy",1));
// 获取Session的id
String id = session.getId();
// 判断session是不是新创建的
if(session.isNew()){
resp.getWriter().write("session创建成功,ID:"+id);
}else{
resp.getWriter().write("session已经在服务器中存在了,ID:"+id);
}
}
// 得到Session
HttpSession session = req.getSession();
Person person = (Person) session.getAttribute("name");
System.out.println(person.toString());
// 注销Session
HttpSession session = req.getSession();
session.removeAttribute("name");
session.invalidate();
会话自动过期:web.xml
<!--设置Session默认的失效时间-->
<session-config>
<!--15分钟后session自动失效-->
<session-timeout>15</session-timeout>
</session-config>