今日内容:
1. Servlet
2. HTTP协议
3. Request
一、Servlet:
- 概念
- 步骤
- 执行原理
- 生命周期
- Servlet3.0 注解配置
- Servlet的体系结构
HttpServlet(抽象类) extend GenericServlet(抽象类) implements Servlet(接口)
案例GenericServlet:将Servlet接口中其他的方法做了默认空实现,只将service()方法作为抽象。将来定义Servlet类时,可以继承GenericServlet,实现service()方法即可 HttpServlet:对http协议的一种封装,简化操作 1. 定义类继承HttpServlet 2. 复写doGet/doPost方法
//实现servlet @WebServlet("/demo2") public class ServletDemo2 implements Servlet { @Override public void init(ServletConfig servletConfig) throws ServletException { } @Override public ServletConfig getServletConfig() { return null; } @Override public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException { System.out.println("直接实现servlet"); } @Override public String getServletInfo() { return null; } @Override public void destroy() { } } //继承GenericServlet @WebServlet("/demo3") public class ServletDemo3 extends GenericServlet { @Override public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException { System.out.println("继承GenericServlet"); } } //继承HttpServlet @WebServlet("/demo4") public class ServletDemo4 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("do get"); } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("do post"); } }
- Servlet相关配置
①urlpartten:Servlet访问路径
一个Servlet可以定义多个访问路径 : @WebServlet({"/d4","/dd4","/ddd4"})
②路径定义规则:1. /xxx:路径匹配 2. /xxx/xxx:多层路径,目录结构 3. *.do:扩展名匹配,任意扩展
二、HTTP:
- 概念:Hyper Text Transfer Protocol 超文本传输协议
- 传输协议:定义了,客户端和服务器端通信时,发送数据的格式
- 特点:
①基于TCP/IP的高级协议
②默认端口号:80
③基于请求/响应模型的:一次请求对应一次响应
④无状态的:每次请求之间相互独立,不能交互数据 - 历史版本:
①1.0:每一次请求响应都会建立新的连接
②1.1:复用连接 - 请求消息数据格式
①请求行
HTTP协议有7中请求方式,常用的有2种请求方式 请求url 请求协议/版本 GET /login.html HTTP/1.1
GET:
POST:1. 请求参数在请求行中,在url后。 2. 请求的url长度有限制的 3. 不太安全
②请求头:客户端/浏览器告诉服务器一些信息1. 请求参数在请求体中 2. 请求的url长度没有限制的 3. 相对安全
请求头名称: 请求头值
常见的请求头:
③请求空行1. User-Agent:浏览器告诉服务器,我访问你使用的浏览器版本信息 可以在服务器端获取该头的信息,解决浏览器的兼容性问题 2. Referer:http://localhost/login.html 告诉服务器,我(当前请求)从哪里来? 作用: 1. 防盗链: 2. 统计工作:
空行,就是用于分割POST请求的请求头,和请求体的。
④请求体(正文):封装POST请求消息的请求参数的POST /login.html HTTP/1.1 Host: localhost User-Agent: Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:60.0) Gecko/20100101 Firefox/60.0 Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8 Accept-Language: zh-CN,zh;q=0.8,zh-TW;q=0.7,zh-HK;q=0.5,en-US;q=0.3,en;q=0.2 Accept-Encoding: gzip, deflate Referer: http://localhost/login.html Connection: keep-alive Upgrade-Insecure-Requests: 1 username=zhangsan
- 响应消息数据格式
三、 Request
request对象和response对象的原理
①request和response对象是由服务器创建的。我们来使用它们
②request对象是来获取请求消息,response对象是来设置响应消息request对象继承体系结构:
ServletRequest -- 接口
| 继承
HttpServletRequest -- 接口
| 实现
org.apache.catalina.connector.RequestFacade 类(tomcat)-
request功能:
①获取请求行数据
请求行:GET /anfly/requestdemo?name=anfly HTTP/1.1@WebServlet("/requestdemo") public class RequestDemo1 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //取请求方式 :GET String method = req.getMethod(); System.out.println(method); //2. (*)获取虚拟目录:/anfly String contextPath = req.getContextPath(); System.out.println(contextPath); //获取Servlet路径: /demo1 String servletPath = req.getServletPath(); System.out.println(servletPath); //获取get方式请求参数:name=anfly String queryString = req.getQueryString(); System.out.println(queryString); //(*)获取请求URI-统一资源标识符:/day14/demo1 String requestURI = req.getRequestURI(); System.out.println(requestURI); //(*)获取请求URL:统一资源定位符:http://localhost/day14/demo1/day14/demo1 StringBuffer requestURL = req.getRequestURL(); System.out.println(requestURL); //获取协议及版本:HTTP/1.1 String protocol = req.getProtocol(); System.out.println(protocol); //获取客户机的IP地址: String remoteAddr = req.getRemoteAddr(); System.out.println(remoteAddr); } }
②获取请求头数据
//获取所有的请求头名称 Enumeration<String> headerNames = req.getHeaderNames(); while (headerNames.hasMoreElements()){ String name = headerNames.nextElement(); System.out.println(name+":"+req.getHeader(name)+"/n"); }
③获取请求体数据:只有POST请求方式,才有请求体,在请求体中封装了POST请求的请求参数
步骤:1. 获取流对象 * BufferedReader getReader():获取字符输入流,只能操作字符数据 * ServletInputStream getInputStream():获取字节输入流,可以操作所有类型数据 * 在文件上传知识点后讲解 2. 再从流对象中拿数据
代码
//注册界面 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>注册</title> </head> <body> <form action="/anfly/demo5" method="post"> <input type="text" placeholder="请输入用户名" name="name"><br> <input type="text" placeholder="请输入密码" name="pwd"><br> <input type="submit" value="注册"><br> </form> </body> </html> //获取请求体 @WebServlet("/demo5") public class ServletDemo5 extends HttpServlet { @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //获取请求体 //1. 获取字符流 BufferedReader br = req.getReader(); //2. 读取数据 String line = null; while ((line = br.readLine())!=null){ System.out.println(line); } } }
④获取请求参数通用方式:不论get还是post请求方式都可以使用下列方法来获取请求参数
1. String getParameter(String name):根据参数名称获取参数值 username=zs&password=123 2. String[] getParameterValues(String name):根据参数名称获取参数值的数组 hobby=xx&hobby=game 3. Enumeration<String> getParameterNames():获取所有请求的参数名称 4. Map<String,String[]> getParameterMap():获取所有参数的map集合
中文乱码问题:
get方式:tomcat 8 已经将get方式乱码问题解决了 post方式:会乱码 解决:在获取参数前,设置request的编码:request.setCharacterEncoding("utf-8");
⑤请求转发:一种在服务器内部的资源跳转方式
步骤:1. 通过request对象获取请求转发器对象:RequestDispatcher getRequestDispatcher(String path) 2 . 使用RequestDispatcher对象来进行转发:forward(ServletRequest request, ServletResponse response)
代码:
//第一个servlet @WebServlet("/demo6") public class ServletDemo6 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("ServletDemo6"); req.getRequestDispatcher("/demo7").forward(req,resp); } } //第二个servlet @WebServlet("/demo7") public class ServletDemo7 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("ServletDemo7"); } }
特点:
1. 浏览器地址栏路径不发生变化 2. 只能转发到当前服务器内部资源中。 3. 转发是一次请求
⑥共享数据:
域对象:一个有作用范围的对象,可以在范围内共享数据
request域:代表一次请求的范围,一般用于请求转发的多个资源中共享数据
方法:1. void setAttribute(String name,Object obj):存储数据 2. Object getAttitude(String name):通过键获取值 3. void removeAttribute(String name):通过键移除键值对
代码:
//第一个servlet @WebServlet("/demo6") public class ServletDemo6 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("ServletDemo6"); //存储数据 req.setAttribute("msg","hello"); req.getRequestDispatcher("/demo7").forward(req,resp); } } //第二个servlet @WebServlet("/demo7") public class ServletDemo7 extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //获取数据 Object msg = req.getAttribute("msg"); System.out.println(msg); System.out.println("ServletDemo7"); } }
⑦获取ServletContext:
ServletContext getServletContext()
四、案例:用户登录
- 用户登录案例需求:
1.编写login.html登录页面
username & password 两个输入框
2.使用Druid数据库连接池技术,操作mysql,day14数据库中user表
3.使用JdbcTemplate技术封装JDBC
4.登录成功跳转到SuccessServlet展示:登录成功!用户名,欢迎您
5.登录失败跳转到FailServlet展示:登录失败,用户名或密码错误
- 开发步骤
①创建项目,导入html页面,配置文件,jar包
②创建数据库环境//HTML代码,web下边创建login.html文件 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>登录</title> </head> <body> <form action="/anfly/loginservlet" method="post"> 账号<input name="name" type="text"><br> 密码<input name="pwd" type="text"><br> <input value="登录" type="submit"><br> </form> </body> </html> //配置文件:src下边创建文件 druid.properties driverClassName=com.mysql.jdbc.Driver url=jdbc:mysql://127.0.0.1:3306/login username=root password=123 # 初始化连接数 initialSize=5 # 最大连接数 maxActive=10 # 超时 maxWait=3000 //jar包 - 在web\WEB-INF\lib\下添加 commons-logging-1.2.jar druid-1.0.9.jar mchange-commons-java-0.2.12.jar mysql-connector-java-5.1.37-bin.jar spring-beans-5.0.0.RELEASE.jar spring-core-5.0.0.RELEASE.jar spring-jdbc-5.0.0.RELEASE.jar spring-tx-5.0.0.RELEASE.jar
③创建包cn.itcast.domain,创建类User,和数据库user表字段对应create database login; create table user(id int primary key auto_increment,name varchar(20) not null,pwd varchar(20)); insert into user vlaues(1,"anfly",'123');
④创建包cn.itcast.util,编写工具类JDBCUtilspackage cn.itcast.domain; /** * 用户实体类 */ public class User { private int id; private String name; private String pwd; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getPwd() { return pwd; } public void setPwd(String pwd) { this.pwd = pwd; } @Override public String toString() { return "User{" + "id=" + id + ", name='" + name + '\'' + ", pwd='" + pwd + '\'' + '}'; } }
⑤创建包cn.itcast.dao,创建类UserDao,作用是操作数据库。提供login方法package cn.itcast.utils; import com.alibaba.druid.pool.DruidDataSourceFactory; import javax.sql.DataSource; import java.io.InputStream; import java.sql.Connection; import java.sql.SQLException; import java.util.Properties; /** * JDBC工具类 使用Durid连接池 */ public class JDBCUtil { private static DataSource ds; static { try { //1.加载配置文件 Properties pro = new Properties(); //使用ClassLoader加载配置文件,获取输入流 InputStream is = JDBCUtil.class.getClassLoader().getResourceAsStream("druid.properties"); pro.load(is); //2. 初始化连接池对象 ds = DruidDataSourceFactory.createDataSource(pro); } catch (Exception e) { e.printStackTrace(); } } //1.获取连接池对象 public static DataSource getDs() { return ds; } //2.获取连接connection对象 public static Connection getConnection() throws SQLException { return ds.getConnection(); } }
⑥编写cn.itcast.web.servlet.LoginServlet类package cn.itcast.dao; import cn.itcast.domain.User; import cn.itcast.utils.JDBCUtil; import org.springframework.dao.DataAccessException; import org.springframework.jdbc.core.BeanPropertyRowMapper; import org.springframework.jdbc.core.JdbcTemplate; /** * 操作数据库中user表的类 */ public class UserDao { //申明JdbcTemplate对象共用 private JdbcTemplate jt = new JdbcTemplate(JDBCUtil.getDs()); /** * 登录方法 * @param loginUser 只有用户名和密码 * @return 包含用户全部数据 */ public User login(User loginUser){ try { //1. 编写SQL String sql = "select * from user where name = ? and pwd = ?"; //2.调用query方法 User user = jt.queryForObject(sql, new BeanPropertyRowMapper<User>(User.class), loginUser.getName(), loginUser.getPwd()); return user; } catch (DataAccessException e) { e.printStackTrace(); return null; } } }
⑦编写FailServlet和SuccessServlet类package cn.itcast.web.servlet; import cn.itcast.dao.UserDao; import cn.itcast.domain.User; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; @WebServlet("/loginservlet") public class LoginServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //设置编码 req.setCharacterEncoding("utf-8"); //2. 获取请求参数 String name = req.getParameter("name"); String pwd = req.getParameter("pwd"); //3. 封装user对象 User loginUser = new User(); loginUser.setPwd(pwd); loginUser.setName(name); //4.调用userdao中login方法 UserDao userDao = new UserDao(); User user = userDao.login(loginUser); //5. 判断user if (user == null) { //登录失败,转发到失败 req.getRequestDispatcher("/failservlet").forward(req, resp); } else { //登录成功 //存储数据 req.setAttribute("user", user); //转发 req.getRequestDispatcher("/successservlet").forward(req, resp); } } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { this.doGet(req, resp); } }
⑧login.html中form表单的action路径的写法:虚拟目录+Servlet的资源路径package cn.itcast.web.servlet; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; @WebServlet("/failservlet") public class FailServlet extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //给页面写一句话 //设置编码 response.setContentType("text/html;charset=utf-8"); //输出 response.getWriter().write("登录失败,用户名或密码错误"); } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doPost(request, response); } } package cn.itcast.web.servlet; import cn.itcast.domain.User; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; @WebServlet("/successservlet") public class SuccessServlet extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //给页面写一句话 //获取request共享域中的对象 User user = (User) request.getAttribute("user"); //输出 if (user != null) { response.setContentType("text/html;charset=utf-8"); //输出 response.getWriter().write("登录成功!"+user.getName()+",欢迎您"); } } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doPost(request,response); } }
⑨BeanUtils工具类,简化数据封装
用于封装JavaBean的
a.JavaBean:标准的Java类
b.概念:1. 要求: 1. 类必须被public修饰 2. 必须提供空参的构造器 3. 成员变量必须使用private修饰 4. 提供公共setter和getter方法 2. 功能:封装数据
成员变量:
属性:setter和getter方法截取后的产物
例如:getUsername() --> Username--> username
c.方法:
d.使用:在LoginServlet中获取参数的时候1. setProperty() 2. getProperty() 3. populate(Object obj , Map map):将map集合的键值对信息,封装到对应的JavaBean对象中
//添加jar包:commons-beanutils-1.8.0.jar //2. 获取请求参数 String name = req.getParameter("name"); String pwd = req.getParameter("pwd"); //3. 封装user对象 User loginUser = new User(); loginUser.setPwd(pwd); loginUser.setName(name); //可以简化为: try { //获取所有请求参数 Map<String, String[]> map = req.getParameterMap(); //创建user对象 User loginUser = new User(); //调用BeanUtils的populate()把map数据封装到loginUser中 BeanUtils.populate(loginUser, map); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); }