1 案例1-统计网站访问次数

1.1 需求

当用户访问网站的时候,直接提示用户是第几个访问该网站的人;

1.2 技术分析

1:需要使用一个容器来保存所有用户访问的次数;
2:使用servletContext对象即可完成这个事情;

1.3 ServletContext概述

servletContext是一个工程容器对象;(可以简单理解为时tomcat为每一个web工程创建的一个唯一的对象),程序员可以获取这个对象,并操作这个对象;
注意:
该对象只能由tomcat创建,程序员不能自己创建;

1.4 servletContext对象的作用

servletContext有3个方面的作用:
1:作为容器使用,可以保存或获取数据;(可以理解为时map集合)
servletContext的获取方式与使用:
//1:获取servletContext对象的方式1(了解)
ServletContext c1 = this.getServletConfig().getServletContext();
//2:获取servletContext对象的方式2:(掌握)
ServletContext c2 = this.getServletContext();
//3:存数据
c2.setAttribute(“username”,”rose”);
//4:取数据
C2.getAttribute(“username”);
//5:移除值
C2.removeAttribute(“username”);
2:可以操作工程下的WebContent目录下的文件;
(1)getRealPath(“/xxx”); 获取WebContent目录下的xxx文件的绝对路径
(2)getResourceAsStream(“/xxx”); 获取WebContent目录下的xxx文件作为一个字节输入流使用
(3) getMimeType(String file)
MIME:
是网络中用于描述文件数据类型的一个格式;(wc3shooli能查到)
参考代码:

  1. //1:获取ServletContext对象
  2. ServletContext s = this.getServletContext();
  3. //2:获取文件绝对路径
  4. String path = s.getRealPath("/1.txt");
  5. System.out.println("绝对路径:"+path);
  6. //3:获取文件输入流
  7. InputStream in = s.getResourceAsStream("/1.txt");
  8. System.out.println(in);
  9. //4:获取文件mime类型
  10. String type = s.getMimeType("/1.txt");
  11. System.out.println("Mime="+type);

扩展:
如果要读src文件夹下的文件,需要使用ClassLoader;(类加载器)
参考代码:
MyClassLoader.class文件内代码

  1. //注意:路径直接写,不带斜杠,直接写就默认从src文件夹下找文件
  2. InputStream in = MyClassLoader.class.getClassLoader().getResourceAsStream("2.txt");
  3. System.out.println(in);

3:可以获取web.xml配置文件中的全局配置参数;
(1)需要在web.xml中配置
格式:

  1. <context-param>
  2. <param-name>age</param-name>
  3. <param-value>18</param-value>
  4. </context-param>
  5. <context-param>
  6. <param-name>sex</param-name>
  7. <param-value></param-value>
  8. </context-param>

(2)需要使用ServletContext对象获取参数:

        //1:获取servletContext对象
        ServletContext sc = this.getServletContext();
        //2:根据属性名获取对应的配置属性值
        String v1 = sc.getInitParameter("age");
        String v2 = sc.getInitParameter("sex");
        System.out.println(v1);
        System.out.println(v2);
        //3:获取所有的参数名
        Enumeration<String> names = sc.getInitParameterNames();//枚举
        while(names.hasMoreElements()){
            String key = names.nextElement();
            String v = sc.getInitParameter(key);
            System.out.println(key+"===>"+v);
        }

1.5 ServletContext对象的生命周期问题

1:创建时机:(记住)当服务器启动的时候,servletContext对象会被tomcat服务器创建出来,且每一个工程仅有一个对象,所有的servlet对象都可以获取这个servletContext对象并使用;
2:死亡时机:(了解)
(1)tomcat启动状态下,移除工程;
(2)正常停止tomcat服务器;

1.6 案例步骤分析

(1)在web.xml中配置一个全局参数,目的是为了让第一个访问网站的人,可以从容器中获取一个访问次数0;
(2)编写一个servlet,专门用于统计网站访问次数;同时显示该用户是第几个访问网站的人;
1)从servletContext对象中获取访问次数;
2)将次数累加一次;
3)响应给浏览器,访问的次数;
4)将修改后的访问次数更新到servletContext对象中;
参考代码:

//当servlet对象被创建的时候,就立刻将count的值设置为0
    @Override
    public void init() throws ServletException {
        ServletContext c = this.getServletContext();
        String v = c.getInitParameter("count");
        int i=Integer.parseInt(v);
        c.setAttribute("count",i);
    }

/*
         *     1:从servletContext对象中获取访问次数;
            2:将次数累加一次;
            3:响应给浏览器,访问的次数;
            4:将修改后的访问次数更新到servletContext对象中;
         */
        ServletContext c = this.getServletContext();
        //1:从servletContext对象中获取访问次数;
        Integer i =  (Integer)c.getAttribute("count");
        //2:将次数累加一次;
        i++;
        //3:响应给浏览器,访问的次数;
        response.setContentType("text/html;charset=utf-8");
        response.getWriter().print("<h1>亲,您是第"+i+"个访问的客户.....</h1>");
        //4:将修改后的访问次数更新到servletContext对象中;
        c.setAttribute("count",new Integer(i));

2 案例2-文件下载

2.1 需求

当用户点击超链接的时候,让用户执行下载的动作;

2.2 技术分析

从服务器上将文件通过io流响应给浏览器的过程,就是下载,需要使用response对象;

2.3 Response对象概述

Response对象是tomcat服务器创建的对象,而且是当用户每一次发出请求的时候,tomcat都会创建一个对应的请求对象(明天学)和响应对象;
Response对象的数据类型是:HttpServletResponse 是一个接口类型;

2.4 Response对象的作用

Response对象的作用主要有3部分:操作响应行、响应头、响应体。

2.4.1 操作响应行;

仅操作状态码即可:
setStatus(状态码);
状态码:
1xx;
请求已发送成功;
2xx;
响应已完成;
例如: 200 请求已完事响应也完事了
3xx:
需要浏览器进一步操作;
例如:
302 重定向;
304 查本地缓存;
4xx:
用户操作错误;
例如:
404 地址栏地址错误
5xx:
服务器内部错误
例如:
500 程序员代码错误

2.4.2 操作响应头;

方法:
setHeader(“http协议规定的响应头”,”具体的值”);
常见的响应头:
1: location 配合302状态码可以让浏览器重定向;值格式:”绝对路径”
绝对路径:说的是带工程名。如果去其他地方就带协议,所以建议又带协议又带工程名。
举例:
重定向到www.itcast.cn这个网站。
response.setStatus(302);
response.setHeader(“location”, “http://www.itcast.cn“);
2: refresh 让浏览器定时刷新;值格式: “秒数;url=….”
此处的url填写说明:
(1)url后面直接/…表示工程目录下(得从工程名day029开始写)。举例:url=/day029/MyServlet
(2)不用/表示当前目录下。举例:url=MyServlet
(3)也可以用http://.....。举例:url="https://www.baidu.com
3:Context-Type 告诉浏览器,服务器响应回来的数据类型与编码格式;
举例:
设置中文编码:
response.setCharacterEncoding(“utf-8”);
response.setHeader(“Content-Type”,”text/html”);

response.setHeader(“Content-Type”,”text/html;charset=utf-8”);
实际中的应用:
设置中文编码:
response.setContentType(“text/html;charset=utf-8”);
重定向:
response.sendRedirect(“http://www.itheima.com“);
练习:

package com.itheima.demo03_response;

import java.io.IOException;
import java.util.Date;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
 * 操作响应头
 */
public class MyResponse2 extends HttpServlet {
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //1:设置状态码
        //重定向
        //response.setStatus(302);
        //response.setHeader("location","http://www.itcast.cn");
        //response.getWriter().println(new Date().toLocaleString());
        //2:定时刷新
        //response.setHeader("refresh", "2;url=/day31/r2");


        //3:设置中文乱码问题
        //response.setCharacterEncoding("utf-8");
        //response.setHeader("Content-Type","text/html;charset=utf-8");

        //response.setContentType("text/html;charset=utf-8");
        //response.getWriter().println("你好222!");

        //重定向的优化代码
        response.sendRedirect("http://www.itheima.com");
    }

    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // TODO Auto-generated method stub
        doGet(request, response);
    }

}

2.4.3 操作响应体:

其实就是给浏览器响应数据;
分两种情况:
1:响应的是纯文字;(html标签) 使用字符输出流即可;
response.getWriter();
2:响应文件;(文件下载) 使用字节输出流即可;
response.getOutputStream();
注意事项:
俩流互斥!!!
(一旦使用了其中一个流,就不能再使用另一个流了,否则语法报错;是执行时报错,不是编译时报错)
这两个流,tomcat都会自动关闭;

2.5 文件下载的案例步骤分析

1:文件下载就是将服务器上的文件通过网络响应给浏览器;
2:按照响应方式不同,分两种方式;
方式1:
直接通过超链接完成下载;
好处:
代码极其简单;(仅需要超链接即可)
缺点:
如果浏览器能识别这个文件,则不会下载,会直接打开文件;(其实也可以下载,右键点击另存为就行了)
方式2:
通过编写代码,完成具体的文件下载;
好处:
无论浏览器是否识别这个文件,都会进行下载;
缺点:
代码复杂!

2.6 代码文件下载的步骤

设置两个头和两个流:
两个头:文件下载的两个头
设定接收程序处理数据的方式
Content-Disposition: attachment;filename= 文件名
设定实体内容的MIME类型
Content-Type:查询文件的MIME类型
两个流:
文件字节输入流;
ServletContext.getResourceAsStream(“/文件名”);
网络字节响应流;
response.getOutputStream();
工具:
使用Commons-io工具类的copy(输入流,输出流);

2.8 项目部分代码

结构
QQ拼音截图未命名.png
代码
MyDownload2.java

package com.itheima.demo04_download;

import java.io.IOException;
import java.io.InputStream;

import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.commons.io.IOUtils;

/**
 * 代码方式完成文件下载  (解决中文乱码)
 */
public class MyDownload2 extends HttpServlet {
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        /*
         * 步骤:
         *         1:设置两个头
         *             1:获取用户想下载的文件名称;
         *             1.5:处理文件名的中文乱码问题(传过来的参数的中文乱码问题);
         *                 固定格式:
         *                 String 不乱的名称= new String(乱码后的字符串.getBytes("iso8859-1"),"utf-8");
         *             1.6:解决响应中的中文文件名乱码问题;
         *                 使用工具类,根据不同的浏览器,采用不同的解决方案即可;    
         *         2:获取文件的mime类型;
         *                 
         *             3:设置两个头具体的值
         *         2:获取连个流;
         *             1:获取文件输入流
         *             2:获取输出流
         *         3:使用工具复制两个流中的数据;
         *             IoUtils.copy(....);
         */
        //1:获取用户想下载的文件名称;
        String name = request.getParameter("filename");
        //1.5:解决请求中的参数乱码问题
        String realName = new String(name.getBytes("iso8859-1"),"utf-8");
        //2:获取文件的mime类型;
        String mimeType = this.getServletContext().getMimeType(realName);
        //2.5:解决响应的中文文件名乱码问题  使用工具解决
        String dname = DownLoadUtils.getName(request.getHeader("User-Agent"), realName);
        //3:设置两个头具体的值
        response.setHeader("Content-Disposition", "attachment;filename="+dname);
        response.setHeader("Content-Type",mimeType);
        //4:获取文件输入流
        InputStream in = this.getServletContext().getResourceAsStream("/download/"+realName);
        //5::获取输出流
        ServletOutputStream out = response.getOutputStream();
        //6:使用工具复制两个流中的数据;
        IOUtils.copy(in, out);
        in.close();
    }
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doGet(request, response);
    }

}

DownLoadUtils.Java(不用记,知道这个工具就行)

package com.itheima.demo04_download;

import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;

import sun.misc.BASE64Encoder;

public class DownLoadUtils {
    public static String getName(String agent, String filename) throws UnsupportedEncodingException {
        if (agent.contains("Firefox")) {
            // 火狐浏览器
            BASE64Encoder base64Encoder = new BASE64Encoder();
            filename = "=?utf-8?B?" + base64Encoder.encode(filename.getBytes("utf-8")) + "?=";
        } else {
            // 其它浏览器
            filename = URLEncoder.encode(filename, "utf-8");
        }
        return filename;
    }
}

3 案例3-验证码

3.1 需求

当用户在页面上访问的时候,显示一个验证码图片;

4.2 技术分析

1:需要使用图片生成的技术(JavaME)动态的生成一张图片;
2:将图片响应给浏览器的img标签即可;
3:给图片绑定一个点击事件,当点击的时候,切换图片;(重新发送请求)

4.3 案例的代码实现

1:从网上复制一个生成图片验证码的servlet代码;(百度:servlet生成验证码)
2:web.xml中自己配置servlet路径;
3:在页面中使用img标签,将标签的src属性指向servlet路径即可;
4:给图片绑定一个点击事件,当点击的时候,切换图片;(重新发送请求)