JavaWeb(二)

6、Servlet

6.1、简介

  • sun 公司开发动态 web 的一门技术
  • 该公司在这些 api 中提供一个接口叫做:Servlet
  • 开发一个 Servlet 程序,只需要完成两个小步骤:
    • 编写一个类,实现 Servlet 接口
    • 把开发好的 java 类部署到 web 服务器当中

把实现了 Servlet 接口的 java 程序叫做:Servlet

6.2、HelloServlet

Servlet 接口 sun 公司有两个默认的实现类:HttpServletGenericServlet

  1. 构建一个maven项目

  2. 构建子项目

  3. 导入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>
    

    注意:有时候可能会爆红,则要手动导入

  4. 编写一个 Servlet 程序

    1. 编写一个普通类

    2. 实现 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);
          }
      }
      
  5. 编写 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>
    
  6. 配置Tomcat

  7. 启动测试

    在这里插入图片描述

在这里插入图片描述

6.3、Servlet运行原理

Servlet 是由web服务器调用,web收到浏览器请求之后,会:

在这里插入图片描述

6.4、Mapping问题

  1. 一个Servlet可以指定一个映射路径

  2. 一个Servlet可以指定多个映射路径

  3. 一个Servlet可以指定通用映射路径

    <!--Servlet请求路径-->
    <servlet-mapping>
        <servlet-name>HelloServlet</servlet-name>
        <url-pattern>/hello/*</url-pattern>
    </servlet-mapping>
    
  4. 默认请求路径

    <!--Servlet请求路径
    	会把首页index.xml干掉
    -->
    <servlet-mapping>
        <servlet-name>HelloServlet</servlet-name>
        <url-pattern>/*</url-pattern>
    </servlet-mapping>
    
  5. 指定一些后缀或者前缀等等

    <!--可以自定义后缀实现请求映射
    	注意点:*前面不能加项目映射路径
    -->
    <!--Servlet请求路径-->
    <servlet-mapping>
        <servlet-name>HelloServlet</servlet-name>
        <url-pattern>*.happy</url-pattern>
    </servlet-mapping>
    
  6. 优先级问题

    指定了固定的映射路径优先级高,如果找不到就会走默认的请求

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

简单分类

  1. 负责向浏览器发送数据的方法

    ServletOutputStream getOutputStream() throws IOException;
    PrintWriter getWriter() throws IOException;
    
  2. 负责向浏览器发送响应头的方法

    //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、下载文件
  1. 要获取下载文件的路径
  2. 下载的文件名是啥
  3. 设置想办法让浏览器能够支持下载我们需要的东西
  4. 获取下载文件的输入流
  5. 创建缓冲区
  6. 获取OutputStream对象
  7. 将FileOutputStream流写入到buffer缓冲区
  8. 使用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资源,关闭浏览器,这个过程称之为会话

**有状态会话:**来过一次之后,就知道你来过了

一个网站,怎么证明你来过?

客户端 服务端

  1. 服务端给客户端一个 信件,客户端下次访问服务端带上信件就可以了:cookie
  2. 服务器等级来过了,下次你来的时候我匹配你:session

7.2、保存会话的两种技术

cookie

  • 客户端技术,响应,请求

session

  • 服务端技术,利用这个技术,可以保存用户的会话信息,我们可以把数据放在session中

7.3、Cookie

  1. 从请求中拿到cookie信息
  2. 服务端响应客户端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

编码与解码

  1. 将字符转换为字节的方式称为编码

    URLEncoder.encode("Happy","UTF-8")
    
  2. 将字节转换为字符的方式称为解码

    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>
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值