• java web 学习八(HttpServletResponse对象2)


    一、HttpServletResponse常见应用——生成验证码

    1.1、生成随机图片用作验证码

      生成图片主要用到了一个BufferedImage类,

      

    生成随机图片范例:

    package gacl.response.study;
    
    import java.awt.Color;
    import java.awt.Font;
    import java.awt.Graphics2D;
    import java.awt.image.BufferedImage;
    import java.io.IOException;
    import java.util.Random;
    import javax.imageio.ImageIO;
    import javax.servlet.ServletException;
    import javax.servlet.http.HttpServlet;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    
    public class ResponseDemo03 extends HttpServlet {
    
        public void doGet(HttpServletRequest request, HttpServletResponse response)
                throws ServletException, IOException {
            
            response.setHeader("refresh", "5");//设置refresh响应头控制浏览器每隔5秒钟刷新一次
            //1.在内存中创建一张图片
            BufferedImage image = new BufferedImage(80, 20, BufferedImage.TYPE_INT_RGB);
            //2.得到图片
            //Graphics g = image.getGraphics();
            Graphics2D g = (Graphics2D)image.getGraphics();
            g.setColor(Color.WHITE);//设置图片的背景色
            g.fillRect(0, 0, 80, 20);//填充背景色
            //3.向图片上写数据
            g.setColor(Color.BLUE);//设置图片上字体的颜色
            g.setFont(new Font(null, Font.BOLD, 20));
            g.drawString(makeNum(), 0, 20);
            //4.设置响应头控制浏览器浏览器以图片的方式打开
            response.setContentType("image/jpeg");//等同于response.setHeader("Content-Type", "image/jpeg");
            //5.设置响应头控制浏览器不缓存图片数据
            response.setDateHeader("expries", -1);
            response.setHeader("Cache-Control", "no-cache");
            response.setHeader("Pragma", "no-cache");
            //6.将图片写给浏览器
            ImageIO.write(image, "jpg", response.getOutputStream());
        }
    
        /**
         * 生成随机数字
         * @return
         */
        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;
        }
        
        public void doPost(HttpServletRequest request, HttpServletResponse response)
                throws ServletException, IOException {
            doGet(request, response);
        }
    
    }

    运行结果如下:

      

    二、HttpServletResponse常见应用——设置响应头控制浏览器的行为

    2.1、设置http响应头控制浏览器禁止缓存当前文档内容     

     response.setDateHeader("expries", -1);
     response.setHeader("Cache-Control", "no-cache");
     response.setHeader("Pragma", "no-cache");

    2.2、设置http响应头控制浏览器定时刷新网页(refresh)

     response.setHeader("refresh", "5");//设置refresh响应头控制浏览器每隔5秒钟刷新一次

     2.3、通过response实现请求重定向

      请求重定向指:一个web资源收到客户端请求后,通知客户端去访问另外一个web资源,这称之为请求重定向。

      应用场景:用户登陆,用户首先访问登录页面,登录成功后,就会跳转到某个页面,这个过程就是一个请求重定向的过程

      实现方式:response.sendRedirect(String location),即调用response对象的sendRedirect方法实现请求重定向
      sendRedirect内部的实现原理:使用response设置302状态码和设置location响应头实现重定向

    例如:

    package gacl.response.study;
    import java.io.IOException;
    import javax.servlet.ServletException;
    import javax.servlet.http.HttpServlet;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    
    public class ResponseDemo04 extends HttpServlet {
    
        public void doGet(HttpServletRequest request, HttpServletResponse response)
                throws ServletException, IOException {
            /**
             * 1.调用sendRedirect方法实现请求重定向,
             * sendRedirect方法内部调用了
             * response.setHeader("Location", "/JavaWeb_HttpServletResponse_Study_20140615/index.jsp");
             * response.setStatus(HttpServletResponse.SC_FOUND);//设置302状态码,等同于response.setStatus(302);
             */
            response.sendRedirect("/JavaWeb_HttpServletResponse_Study_20140615/index.jsp");
            
            //2.使用response设置302状态码和设置location响应头实现重定向实现请求重定向
            //response.setHeader("Location", "/JavaWeb_HttpServletResponse_Study_20140615/index.jsp");
            //response.setStatus(HttpServletResponse.SC_FOUND);//设置302状态码,等同于response.setStatus(302);
        }
    
        public void doPost(HttpServletRequest request, HttpServletResponse response)
                throws ServletException, IOException {
            doGet(request, response);
        }
    }

    三、web工程中URL地址的推荐写法

      在JavaWeb开发中,只要是写URL地址,那么建议最好以"/"开头,也就是使用绝对路径的方式,那么这个"/"到底代表什么呢?可以用如下的方式来记忆"/":如果"/"是给服务器用的,则代表当前的web工程,如果"/"是给浏览器用的,则代表webapps目录。

    3.1、"/"代表当前web工程的常见应用场景

    ①.ServletContext.getRealPath(String path)获取资源的绝对路径

    /**
    * 1.ServletContext.getRealPath("/download/1.JPG")是用来获取服务器上的某个资源,
    * 那么这个"/"就是给服务器用的,"/"此时代表的就是web工程
     * ServletContext.getRealPath("/download/1.JPG")表示的就是读取web工程下的download文件夹中的1.JPG这个资源
    * 只要明白了"/"代表的具体含义,就可以很快写出要访问的web资源的绝对路径
    */
    this.getServletContext().getRealPath("/download/1.JPG");

    ②.在服务器端forward到其他页面

     /**
     * 2.forward
      * 客户端请求某个web资源,服务器跳转到另外一个web资源,这个forward也是给服务器用的,
     * 那么这个"/"就是给服务器用的,所以此时"/"代表的就是web工程
     */
     this.getServletContext().getRequestDispatcher("/index.jsp").forward(request, response);

    ③.使用include指令或者<jsp:include>标签引入页面

    <%@include file="/jspfragments/head.jspf" %>

     

    <jsp:include page="/jspfragments/demo.jsp" />

     此时"/"代表的都是web工程。

    3.2、"/"代表webapps目录的常见应用场景

    ①.使用sendRedirect实现请求重定向

     response.sendRedirect("/JavaWeb_HttpServletResponse_Study_20140615/index.jsp");

      服务器发送一个URL地址给浏览器,浏览器拿到URL地址之后,再去请求服务器,所以这个"/"是给浏览器使用的,此时"/"代表的就是webapps目录,"/JavaWeb_HttpServletResponse_Study_20140615/index.jsp"这个地址指的就是"webappsJavaWeb_HttpServletResponse_Study_20140615index.jsp"

      response.sendRedirect("/项目名称/文件夹目录/页面");这种写法是将项目名称写死在程序中的做法,不灵活,万一哪天项目名称变了,此时就得改程序,所以推荐使用下面的灵活写法:

    response.sendRedirect("/JavaWeb_HttpServletResponse_Study_20140615/index.jsp");

      这种写法改成

    response.sendRedirect(request.getContextPath()+"/index.jsp");

      request.getContextPath()获取到的内容就是"/JavaWeb_HttpServletResponse_Study_20140615",这样就比较灵活了,使用request.getContextPath()代替"/项目名称",推荐使用这种方式,灵活方便!

    ②.使用超链接跳转

     <a href="/JavaWeb_HttpServletResponse_Study_20140615/index.jsp">跳转到首页</a>

      这是客户端浏览器使用的超链接跳转,这个"/"是给浏览器使用的,此时"/"代表的就是webapps目录。

      使用超链接访问web资源,绝对路径的写法推荐使用下面的写法改进:

     <a href="${pageContext.request.contextPath}/index.jsp">跳转到首页</a>

      这样就可以避免在路径中出现项目的名称,使用${pageContext.request.contextPath}取代"/JavaWeb_HttpServletResponse_Study_20140615"

    ③.Form表单提交

    <form action="/JavaWeb_HttpServletResponse_Study_20140615/servlet/CheckServlet" method="post">    
            <input type="submit" value="提交">
    </form>

      这是客户端浏览器将form表单提交到服务器,所以这个"/"是给浏览器使用的,此时"/"代表的就是webapps目录。

     对于form表单提交中action属性绝对路径的写法,也推荐使用如下的方式改进:

     <form action="${pageContext.request.contextPath}/servlet/CheckServlet" method="post">
              <input type="submit" value="提交">
     </form>

      ${pageContext.request.contextPath}得到的就是"/JavaWeb_HttpServletResponse_Study_20140615"

      ${pageContext.request.contextPath}的效果等同于request.getContextPath(),两者获取到的都是"/项目名称"

    ④.js脚本和css样式文件的引用

      <%--使用绝对路径的方式引用js脚本--%>
      <script type="text/javascript" src="${pageContext.request.contextPath}/js/index.js"></script>
      <%--${pageContext.request.contextPath}与request.getContextPath()写法是得到的效果是一样的--%>
      <script type="text/javascript" src="<%=request.getContextPath()%>/js/login.js"></script>
      <%--使用绝对路径的方式引用css样式--%>
      <link rel="stylesheet" href="${pageContext.request.contextPath}/css/index.css" type="text/css"/>

    综合范例:

    <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
    
    <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
    <html>
      <head>
        <title>"/"代表webapps目录的常见应用场景</title>
        <%--使用绝对路径的方式引用js脚本--%>
        <script type="text/javascript" src="${pageContext.request.contextPath}/js/index.js"></script>
        <%--${pageContext.request.contextPath}与request.getContextPath()写法是得到的效果是一样的--%>
        <script type="text/javascript" src="<%=request.getContextPath()%>/js/login.js"></script>
        <%--使用绝对路径的方式引用css样式--%>
          <link rel="stylesheet" href="${pageContext.request.contextPath}/css/index.css" type="text/css"/>
      </head>
      
      <body>
          <%--form表单提交--%>
           <form action="${pageContext.request.contextPath}/servlet/CheckServlet" method="post">
               <input type="submit" value="提交">
           </form>
           <%--超链接跳转页面--%>
           <a href="${pageContext.request.contextPath}/index.jsp">跳转到首页</a>
      </body>
    </html>

     四、response细节问题

      getOutputStream和getWriter方法分别用于得到输出二进制数据、输出文本数据的ServletOuputStream、Printwriter对象。
      getOutputStream和getWriter这两个方法互相排斥,调用了其中的任何一个方法后,就不能再调用另一方法。  
      Servlet程序向ServletOutputStream或PrintWriter对象中写入的数据将被Servlet引擎从response里面获取,Servlet引擎将这些数据当作响应消息的正文,然后再与响应状态行和各响应头组合后输出到客户端。 
      Serlvet的service方法结束后,Servlet引擎将检查getWriter或getOutputStream方法返回的输出流对象是否已经调用过close方法,如果没有,Servlet引擎将调用close方法关闭该输出流对象。

    转载:http://www.cnblogs.com/xdp-gacl/p/3791993.html

  • 相关阅读:
    jQuery同步Ajax带来的UI线程阻塞问题及解决办法
    jQuery的deferred对象详解
    原生js,jquery ajax请求以及jsonp的调用
    vue10行代码实现上拉翻页加载更多数据,纯手写js实现下拉刷新上拉翻页不引用任何第三方插件
    js判断手机或Pc端登陆.并跳转到相应的页面
    移动端touch事件封装
    坦然面对:应对前端疲劳
    webpack2 项目
    PPK提供的浏览器类型及版本检测方法
    2013年五大主流浏览器 HTML5 和 CSS3 兼容性大比拼
  • 原文地址:https://www.cnblogs.com/lijia0511/p/5081567.html
Copyright © 2020-2023  润新知