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:获取ServletContext对象
ServletContext s = this.getServletContext();
//2:获取文件绝对路径
String path = s.getRealPath("/1.txt");
System.out.println("绝对路径:"+path);
//3:获取文件输入流
InputStream in = s.getResourceAsStream("/1.txt");
System.out.println(in);
//4:获取文件mime类型
String type = s.getMimeType("/1.txt");
System.out.println("Mime="+type);
扩展:
如果要读src文件夹下的文件,需要使用ClassLoader;(类加载器)
参考代码:
MyClassLoader.class文件内代码
//注意:路径直接写,不带斜杠,直接写就默认从src文件夹下找文件
InputStream in = MyClassLoader.class.getClassLoader().getResourceAsStream("2.txt");
System.out.println(in);
3:可以获取web.xml配置文件中的全局配置参数;
(1)需要在web.xml中配置
格式:
<context-param>
<param-name>age</param-name>
<param-value>18</param-value>
</context-param>
<context-param>
<param-name>sex</param-name>
<param-value>女</param-value>
</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 项目部分代码
结构
代码
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:给图片绑定一个点击事件,当点击的时候,切换图片;(重新发送请求)