Servlet
简介
Servlet是JavaEE的一个组件,也是JavaEE中的一种规范,为了遵循这一规范所有的类都必须实现javax.servlet.Servlet接口
使用IDEA开发Servlet
- 新建一个项目:
- 新建好的项目的结构
- 接着在src中新建一个包,用于存放java代码,在这个包中new
- 新建的servlet中IDEA已经帮你写好需要继承的HttpSevlet类,但因为没有导包,致使 类报错,所以我们需要给其导包(快捷键alt+回车) 点击Add JavaEE6 …
- 点击后会出现一个提示框,接着点击Download即可,IDEA会自动下载所需的jar包,并放在lib的文件夹下
6. 此时就可以编写Servlet了
package com.igor.servlet;
import java.io.IOException;
public class ServletDemo extends javax.servlet.http.HttpServlet {
protected void doPost(javax.servlet.http.HttpServletRequest request, javax.servlet.http.HttpServletResponse response) throws javax.servlet.ServletException, IOException {
}
protected void doGet(javax.servlet.http.HttpServletRequest request, javax.servlet.http.HttpServletResponse response) throws javax.servlet.ServletException, IOException {
//让浏览器打印一句Hello Servlet
response.getWriter().println("Hello Servlet");
}
}
- 接着配置Servlet,打开下面的Web目录下的WEB-INF目录,有一个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">
<servlet>
<servlet-name>ServletDemo</servlet-name>
<servlet-class>com.igor.servlet.ServletDemo</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>ServletDemo</servlet-name>
<url-pattern>/servlet</url-pattern>
</servlet-mapping>
</web-app>
配置Tomcat,以及运行
见:https://blog.youkuaiyun.com/weixin_43958460/article/details/96269640 后面的使用IDEA搭建Web开发环境,在搭建完成后运行Tomcat,Tomcat运行后,在运行的初始页面后加上上面web.xml中servlet-apping标签下,url-pattern中的地址,比如:/servlet
此时页面跳转,并显示"Hello Servlet"。
Servlst的生命周期
可以用this查看其中的方法:
.
Servlet运行过程
在刚下的演示中,我们用一句代码演示了前后端的交互,其中的流程又是怎样的的?
- 当服务器跳转到"localhost:8080/javaweb/servlet"时,浏览器给服务器发出请求,服务器找到web.xml文件,然后在servlet-mapping标签中通过"/servlet"找到对应servlet-name中的ServletDemo,接着找到servlet标签下与ServletDemo相同的servlet-name,从而找到servlet-class中Servlet类的地址,进而将该Servlet类加载进JVM。
- JVM加载字节码,该Servlet类中的doGet方法中的response会被执行,响应浏览器的请求,同时给浏览器传输"Hello Servlet"的资源,浏览器接收后打印在页面上。
Servlet实现类
Servlet映射路径问题
1.一般情况下我们都会指定一个URL , 一个URL对应一个请求
2.通配问题
/* :不报错
*.do
Reponse需要掌握
重定向
将这个请求,转换到另一个地址;
比如,我们请求 : http://localhost:8080/servlet01/dasdasdasda/aaaa.do 他会跳转到百度首页;
//通过响应对象重定向页面
resp.sendRedirect("404.jsp");
ServletContext
通过servletContext对象实现数据共享
通过this获得ServletContext,实现数据的共享,ServletContext中的资源会被存储到服务器中,所有的Servlet都可以访问的到。
通过servletContext对象读取网站配置文件
-
新建一个properties文件
driver=com.mysql.jdbc.Driver username=root password=123456 url=jdbc:mysql://localhost:3306/smbms
-
编写servlet类
package com.kuang.servlet; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.FileInputStream; import java.io.IOException; import java.util.Properties; //读取properties配置文件 public class ServletDemo03 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { doPost(req,resp); } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //读取配置文件 //1.获得配置文件的路径 String realPath = this.getServletContext().getRealPath("/WEB-INF/classes/resources/database.properties"); System.out.println("取得的路径为:"+realPath); Properties properties = new Properties(); FileInputStream is = new FileInputStream(realPath); properties.load(is);//把文件流加载到配置文件的对象中; String driver = properties.getProperty("driver"); String username = properties.getProperty("username"); String password = properties.getProperty("password"); String url = properties.getProperty("url"); //响应到网页 resp.getWriter().println(driver); resp.getWriter().println(username); resp.getWriter().println(password); resp.getWriter().println(url); //======================================= System.out.println(driver); System.out.println(username); System.out.println(password); System.out.println(url); } }
-
配置web.xml
<servlet> <servlet-name>Demo03</servlet-name> <servlet-class>com.kuang.servlet.ServletDemo03</servlet-class> </servlet> <servlet-mapping> <servlet-name>Demo03</servlet-name> <url-pattern>/q3</url-pattern> </servlet-mapping>
-
访问查看
localhost:8080/demo02/q3
-
项目结构
简单验证码
package com.kuang.servlet;
import javax.imageio.ImageIO;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.util.Map;
//验证码
public class ServletDemo04 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doPost(req,resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//resp.getWriter().println(newRandom());
//自动刷新网页
resp.setHeader("refresh","2");
//验证码是一个图片 , 我们需要制作一个图片
BufferedImage image = new BufferedImage(100,30,BufferedImage.TYPE_3BYTE_BGR);
//图片写入一些东西
Graphics2D graphics = image.createGraphics();
graphics.setColor(Color.red);
String num = String.valueOf(newRandom());
graphics.drawString(num,10,10);
//想办法让浏览器知道我们给的是一张图片
resp.setContentType("image/jpg");
//让网站去打开图片
ImageIO.write(image,"jpg",resp.getOutputStream());
}
//生成随机数
public int newRandom(){
int num = (int)(Math.random()*9+1)*100000;
return num;
}
/*
@Test
public void test(){
ServletDemo04 servletDemo04 = new ServletDemo04();
for (int j = 0; j < 100; j++) {
int i = servletDemo04.newRandom();
System.out.println(i);
}
}
*/
}
乱码问题解决方案:
- 所有能设置编码的地方都统一设置 : 数据库,数据库字段,Servlet程序,请求,响应 , 前端,浏览器。
JSP取得项目的路径
${pageContext.request.contextPath}
Request对象
package com.kuang.servlet;
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 RequestDemo01 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//查看request对象的方式
System.out.println(request.getContextPath()); //获得web项目路径
System.out.println(request.getRequestURI()); //请求的URL路径
//Remote 远程
System.out.println(request.getRemoteUser()); //获得远程的用户名
System.out.println(request.getRequestedSessionId()); //获得SessionId;
System.out.println(request.getServletPath()); //请求servlet的url
System.out.println(request.getLocalAddr()); //获得本地地址 防火墙
System.out.println(request.getLocalName()); //获得本地名字
System.out.println(request.getLocalPort()); //获得访问的端口号
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request,response);
}
}
Request对象接收前端用户提交的参数
-
写前端注册页面
-
表单的提交方式 get/post
-
表单提交的位置 action:服务项目的路径/servlet对应的请求url
<%@ page contentType="text/html;charset=UTF-8" language="java" %> <html> <head> <title>注册</title> </head> <body> <%-- JSP的注释 --%> <%-- Form表单 action:代表要提交的路径,表单要提交到哪里 ,可以提交到一个Servlet //提交到Servlet需要,获取到Web项目的路径下的Servlet method: 提交表单的方式 --%> <h1>注册</h1> <%-----------action的值用的是EL表达式,指跳转到servlet配置的地址----------------%> <form action="${pageContext.request.contextPath}/r2" method="post"> <p>用户名:<input type="text" name="username"></p> <p>密码:<input type="password" name="password"></p> <p> <input type="submit"> <input type="reset"> </p> </form> </body> </html>
-
-
写处理前端提交信息的Servlet
-
接收前端传递的控件信息, request.getParameter(“控件的name‘);
package com.kuang.servlet; import javax.print.attribute.HashPrintRequestAttributeSet; 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 RequestDemo02 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { doPost(req,resp); } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //通过HttpServletRequest获取前端用户提交的信息 //req.getParameterValues(); 获得前端传递的控件信息,通过控件的name;但是可以接受多个参数的控件,比如(多选框....) //req.getParameter(); 获得前端传递的控件信息,通过控件的name; req.setCharacterEncoding("utf-8"); resp.setCharacterEncoding("utf-8"); String username = req.getParameter("username"); String password = req.getParameter("password"); //连接数据库,比较数据库中的信息是否匹配 if (username.equals("admin")&&password.equals("1234567")){ System.out.println("登录成功!"); //跳转到成功页面 }else { System.out.println("登录失败!"); //提示重新注册 } System.out.println(username); System.out.println(password); } }
-
-
配置web.xml
-
配置对应的XML文件
<servlet> <servlet-name>RequsetDemo02</servlet-name> <servlet-class>com.kuang.servlet.RequestDemo02</servlet-class> </servlet> <!--=====================================================--> <servlet-mapping> <servlet-name>RequsetDemo02</servlet-name> <url-pattern>/r2</url-pattern> </servlet-mapping>
-
-
提交测试
- 检查英文提交情况,服务器是否接收成功!
- 提交中文,看看乱码情况
- 设置编码统一,解决乱码问题;
Request实现请求转发
package com.kuang.servlet;
import javax.servlet.RequestDispatcher;
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 FormServlet1 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doPost(req,resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//处理乱码
req.setCharacterEncoding("UTF-8");
//处理前端的请求
String username = req.getParameter("username");
String password = req.getParameter("password");
System.out.println("接收到的用户名:"+username+"\n密码:"+password+"\n爱好:");
String[] hobbies = req.getParameterValues("hobby");
for (String hobby : hobbies) {
System.out.println(hobby);
}
//重定向:服务器告诉客户端你应该请求另外一个地址; (地址栏会变)
//resp.sendRedirect("url");
//转发:服务器把这个请求转向另外一个Servlet去处理; (地址栏不会变)
//RequestDispatcher ,需要使用RequestDispatcher来进行处理,我们需要获得这个类
//参数就是他要转发到的页面
/*
RequestDispatcher requestDispatcher = req.getRequestDispatcher("/success.jsp");
requestDispatcher.forward(req,resp);
*/
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
req.getRequestDispatcher("/success.jsp").forward(req,resp);
}
}
转发和重定向的区别 (面试常见题:笔试)
重定向:服务器告诉客户端,你去请求另外一个地址;客户端行为
-
地址栏会变
-
不能携带参数
转发:服务器自己转发到服务器上的另一个请求; 服务器行为
- 地址栏不会变
- 可以携带参数
会话
会话字面意思:一次交流或交谈
在Web中,会话表示从浏览器打开某个网站,在这个网站中无论操作了什么,知道关闭浏览器,这一个过程,称之为一个会话。
怎么样算会话结束:
- 客户端关闭了
- 服务端销毁了
为什么要处理会话:
长期保持会话,无论用户关闭少次浏览器,这个会话都要存在;
比喻:
你昨天来了教室,我今天需要知道你昨天来了教室;
- 你留下一个标记,让我知道你来过
- 你告诉我你来这边,我需要记录你来了;
你:客户端 我 :服务器
Javaweb中针对这两个不同的端,诞生两个小机制
cookie :在客户端留下一点东西,客户端下次带过来,我就知道你来过了。
比如说:我的店铺非常大,怎么证明你来过?
你开一个会员吧 , 我给你一张VIP卡,你下次带卡来,我就知道你来过了
Session:在服务器端登记你来过。
比如说:你去理发店。怎么证明你来过?
理发店收会员,在他的笔记本上记录了你的消息,下次你来直接报上大名就可以了。
Cookie
构造器 :
Cookie cookie = new Cookie(String name,String value);
Cookie cookie = new Cookie(String name,String value);
Cookie cookie = new Cookie(String name,String value);
服务器响应cookie给客户端
Response.addCookie(Cookie);
Response.addCookie(Cookie);
Response.addCookie(Cookie);
服务查看用户带来的请求是否有cookie
Cookie[] cookies = Request.getCookie();
//可以使用cookie来验证用户是否来过
//判断cookies是否为空,然后遍历即可
Cookie.getName();
Cookie.getValue();
cookie测试用户是否来过
package com.westos.serlvet;
import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
public class CookieTest extends HttpServlet {
boolean flag = false;
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doGet(request, response);
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//解决乱码问题:
request.setCharacterEncoding("UTF-8");
response.setCharacterEncoding("UTF-8");
//检查请求的人是否带了Cookie
//通过用户的请求,检查它是否有cookie
Cookie[] cookies = request.getCookies();
System.out.println("cookie:"+cookies);
if (flag){//如果flag是真,来过
if (cookies!=null){//你有cookie
for (int i = 0; i < cookies.length ; i++) {
Cookie cookie = cookies[i];
if (cookie.getName().equals("lastLoginTime")){
response.getWriter().println("你上一次来的时间为:"+cookie.getValue());
System.out.println("刷新了cookie");
}
}
}
}else { //如果flag是真,没来过
response.getWriter().println("你是第一次来,还想要会员套餐");
System.out.println("给该用户一个cookie");
}
//建立一个cookie
Cookie cookie = new Cookie("lastLoginTime",System.currentTimeMillis()+"");
//设置cookie保存的时间 //cookie.setMaxAge();
//把这个cookie发给客户端
response.addCookie(cookie);
//response.addCookie(new Cookie("lastLoginTime",System.currentTimeMillis()+""));
flag = true;
}
}
Session
只要客户端一旦连接上服务器,服务器就会自动产生Session;
一个连接对应一个session,session可以在一个会话中传递信息;
通过setAttribute设置值。
通过getAttribute获得值。
由服务器端控制,服务器如果重启了,信息就会丢失!
package com.westos.serlvet;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;
public class SessionTest extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doPost(req,resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//解决乱码问题:
req.setCharacterEncoding("UTF-8");
resp.setCharacterEncoding("UTF-8");
//Session由服务器自动创建,如何获得session
//HttpSession 得到的sessioin对象
HttpSession session = req.getSession();
//得到sessionID,一次会话,一个seesionID;
String id = session.getId();
resp.getWriter().println("获得的sessionId:\n"+id);
String name = "qinjiang";
//向session中存入一个值;
session.setAttribute("name",name);
resp.getWriter().println("存入信息成功:"+name);
}
}
package com.westos.serlvet;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;
//获得session存入的值
public class SessionTest2 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doPost(req,resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//解决乱码问题:
req.setCharacterEncoding("UTF-8");
resp.setCharacterEncoding("UTF-8");
//获得
HttpSession session = req.getSession();
System.out.println("得到的SessionID:"+session.getId());
String name = (String) session.getAttribute("name");
resp.getWriter().println("得到的session存入的信息"+name);
//Servlet < Seesion < WebContext
}
}
会话注销方法一:
session.invalidate();//通过代码注销会话
会话注销方法二:
<!--session-config可以设置会话自动过期,时间分钟为单位-->
<session-config>
<session-timeout>1</session-timeout>
</session-config>