• servlet的生命周期:
    创建: 默认第一次接收请求
    销毁: 服务器关闭
    request的生命周期:
    创建: 接收请求时
    销毁: 请求结束时
    范围: 一次请求的过程

一。request的请求转发:
通常是Servlet 转发到 jsp 时使用
一个Servlet事情做了一半, 就需要转发到另一个资源继续完成。

//直接用request设置域对象
        request.setAttribute("name","zhangsan");
//        //获得转发器
//        RequestDispatcher requestDispatcher = request.getRequestDispatcher("/Demo1");
//        //通过转发器转发对象
//        requestDispatcher.forward(request,response);
        //完成请求转发简写
        request.getRequestDispatcher("/Demo2").forward(request,response);
        //请求转发后就发出请求了,下面再写代码就无效了
        --------------------------------------------
        String name=(String) request.getAttribute("name");
        response.getWriter().println(name);

二。response的重定向:
重定向特点:
1.客户端发送了两次请求
2.浏览器地址栏会发生改变, 变成第二次请求地址
3.不能使用request域来分享数据
重定向适用场景:
第一个资源已经做完了一件事情, 需要做下一件事情

// 1.设置响应行信息 - 状态码 为重定向
        response.setStatus(302);
        // response.setHeader("name", "zhangsan");
        // 2.设置响应头内容 - 重定向地址
//        response.setHeader("location", "http://www.baidu.com");
        response.setHeader("location", "/web03_war_exploded/Demo01Servlet");
        // 重定向后, 代码也不会执行
        ------------
        简化写法
          // 获得虚拟目录
        String path = request.getContextPath();
        // 重定向
        response.sendRedirect(path + "/Demo01Servlet");

三。案例1: 图片验证码
通过response的输出流, 写出一张图片

1.验证码程序

@WebServlet("/CheckImgServlet")
public class CheckImgServlet extends HttpServlet {

	// 集合中保存所有成语
	private List<String> words = new ArrayList<String>();

	@Override
	public void init() throws ServletException {
		// 初始化阶段,读取new_words.txt
		// web工程中读取 文件,必须使用绝对磁盘路径
		String path = getServletContext().getRealPath("/WEB-INF/new_words.txt");
		try {
			BufferedReader reader = new BufferedReader(new FileReader(path));
			String line;
			while ((line = reader.readLine()) != null) {
				words.add(line);
			}
			reader.close();
		} catch (IOException e) {
			e.printStackTrace();
		}
	}

	public void doGet(HttpServletRequest request, HttpServletResponse response)
			throws ServletException, IOException {
		// 禁止缓存
		// response.setHeader("Cache-Control", "no-cache");
		// response.setHeader("Pragma", "no-cache");
		// response.setDateHeader("Expires", -1);

		int width = 120;
		int height = 30;

		// 步骤一 绘制一张内存中图片
		BufferedImage bufferedImage = new BufferedImage(width, height,
				BufferedImage.TYPE_INT_RGB);

		// 步骤二 图片绘制背景颜色 ---通过绘图对象
		Graphics graphics = bufferedImage.getGraphics();// 得到画图对象 --- 画笔
		// 绘制任何图形之前 都必须指定一个颜色
		graphics.setColor(getRandColor(200, 250));
		graphics.fillRect(0, 0, width, height);

		// 步骤三 绘制边框
		graphics.setColor(Color.WHITE);
		graphics.drawRect(0, 0, width - 1, height - 1);

		// 步骤四 四个随机数字
		Graphics2D graphics2d = (Graphics2D) graphics;
		// 设置输出字体
		graphics2d.setFont(new Font("宋体", Font.BOLD, 18));

		Random random = new Random();// 生成随机数
		int index = random.nextInt(words.size());
		String word = words.get(index);// 获得成语

		// 定义x坐标
		int x = 10;
		for (int i = 0; i < word.length(); i++) {
			// 随机颜色
			graphics2d.setColor(new Color(20 + random.nextInt(110), 20 + random
					.nextInt(110), 20 + random.nextInt(110)));
			// 旋转 -30 --- 30度
			int jiaodu = random.nextInt(60) - 30;
			// 换算弧度
			double theta = jiaodu * Math.PI / 180;

			// 获得字母数字
			char c = word.charAt(i);

			// 将c 输出到图片
			graphics2d.rotate(theta, x, 20);
			graphics2d.drawString(String.valueOf(c), x, 20);
			graphics2d.rotate(-theta, x, 20);
			x += 30;
		}

		// 将验证码内容保存session
		request.getSession().setAttribute("checkcode_session", word);

		// 步骤五 绘制干扰线
		graphics.setColor(getRandColor(160, 200));
		int x1;
		int x2;
		int y1;
		int y2;
		for (int i = 0; i < 30; i++) {
			x1 = random.nextInt(width);
			x2 = random.nextInt(12);
			y1 = random.nextInt(height);
			y2 = random.nextInt(12);
			graphics.drawLine(x1, y1, x1 + x2, x2 + y2);
		}

		// 将上面图片输出到浏览器 ImageIO
		graphics.dispose();// 释放资源

		//将图片写到response.getOutputStream()中
		ImageIO.write(bufferedImage, "jpg", response.getOutputStream());

	}

	public void doPost(HttpServletRequest request, HttpServletResponse response)
			throws ServletException, IOException {
		doGet(request, response);
	}

	/**
	 * 取其某一范围的color
	 *
	 * @param fc
	 *            int 范围参数1
	 * @param bc
	 *            int 范围参数2
	 * @return Color
	 */
	private Color getRandColor(int fc, int bc) {
		// 取其随机颜色
		Random random = new Random();
		if (fc > 255) {
			fc = 255;
		}
		if (bc > 255) {
			bc = 255;
		}
		int r = fc + random.nextInt(bc - fc);
		int g = fc + random.nextInt(bc - fc);
		int b = fc + random.nextInt(bc - fc);
		return new Color(r, g, b);
	}

}

2.html文件使用图片

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="jsku/jquery-1.11.0.min.js"></script>
    <script src="jsku/jquery.validate.min.js"></script>
    <script>
        $(function () {
            $("#check").click(function () {

                this.src = "/web03_war_exploded/CheckImgServlet?random="+Math.random();
            })
        })
    </script>
</head>
<body>
<form action="/web03_war_exploded/write01" method="get">
<img src="/web03_war_exploded/CheckImgServlet" id="check" />
</form>
</body>
</html>

四。网页文件下载

@WebServlet("/ServletDownload")
public class ServletDownload extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //获得请求的filename
        String filename = request.getParameter("filename");
        //获得绝对路径
        String path=this.getServletContext().getRealPath("/download/"+filename);
        //获得文件输入流
        BufferedInputStream rd=new BufferedInputStream(new FileInputStream(path));
        //获得文件输出流
        OutputStream ot=response.getOutputStream();
       //告诉客户端文件的类型,通过文件类型获取文件后缀
        response.setContentType(this.getServletContext().getMimeType(filename));//设置响应类型setContentType() ;this.getServletContext().getMimeType(filename)获得文件的MIME类型
        //浏览器响应头不支持中文,所以需要处理filename
        String agent = request.getHeader("user-agent");//获得user-agent即浏览器类型
        String neNmae=DownLoadUtils.getFileName(agent,filename);//通过自己写的DownLoadUtils类的getFileName()方法处理中文:就是URLEncoder.encode
        //设置响应头,告诉客户端文件以附件形式打开-下载
        response.setHeader("Content-Disposition","attachment;filename="+neNmae);//Content-Disposition响应头:文件名
        //复制文件
        int len=0;
        byte[] buffer=new byte[1024];
        while ((len=rd.read(buffer))>0){
            ot.write(buffer,0,len);
        }
        rd.close();
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }
}

网页不支持中文,中文转换encode代码

public class DownLoadUtils {

    public static String getFileName(String agent, String filename) throws UnsupportedEncodingException {
        if (agent.contains("MSIE")) {
            // IE浏览器
            filename = URLEncoder.encode(filename, "utf-8");
            filename = filename.replace("+", " ");
        } /* else if (agent.contains("Firefox")) {
            // 火狐浏览器
            Base64.Encoder encoder = Base64.getEncoder();
            filename = "=?utf-8?B?" + encoder.encode(filename.getBytes("utf-8")) + "?=";
        } */ else {
            // 其它浏览器
            filename = URLEncoder.encode(filename, "utf-8");
        }
        return filename;
    }
}

网页

<h2>servlet下载文件</h2>>
    <a href="/web03_war_exploded/ServletDownload?filename=a.txt">txt</a></br>
    <a href="/web03_war_exploded/ServletDownload?filename=aaa.mp4">mp4</a></br>
    <a href="/web03_war_exploded/ServletDownload?filename=note.zip">zip</a></br>
    <a href="/web03_war_exploded/ServletDownload?filename=wo.mp3">mp3</a></br>
    <a href="/web03_war_exploded/ServletDownload?filename=bb.flv">flv</a></br>
    <a href="/web03_war_exploded/ServletDownload?filename=美女.txt">美女.txt</a></br>

五、路径写法:
1.绝对路径 /xx: 绝对路径URI http://xxx: 绝对路径URL
URL: 给客户端使用, 客户端访问服务器时必须使用的路径
例如: form action=“url”
a href=“url”
img src=“url”
link href=“url”
script src=“url”
重定向: url
URI: 给客户端使用可以用URI, 或者给服务器使用必须用URI
给服务器使用: 转发, 不需要使用虚拟目录
给客户端使用: 必须加虚拟目录

如果客户端通过URI来访问服务器, 会将URI拼接成URL

资源在同一个服务器中, 可以将URL简化成URI
http://localhost:8080/web03/form.html /web03/CheckImgServlet
不在同一个服务器中, 只能填完整的URL
http://www.baidu.com

2.相对路径
六、会话技术
一次会话: 客户端连接上服务器开始, 客户端退出结束
会产生多次请求

在会话期间, 存储数据
Cookie: 客户端技术, 不安全[客户端可以删除,查看], 减轻服务器压力
1.向客户端存储cookie
response.addCookie(cookie) - 本质是response.setHeader(“set-cookie”)
2.从客户端获取cookie
request.getCookies() - 本质是request.getHeader(“cookie”)

cookie默认的生命周期: new 开始, 到会话结束
    设置持久化时间
        cookie.setMaxAge(int 秒);
    设置时间=0, 立即删除
    设置时间=-1, 恢复默认
设置cookie的携带路径:
    cookie.setPath(): / -> 当前服务器所有资源
                      /web03 -> 当前应用所有资源
                      /web03/Demo08Servlet -> 单纯只有Demo08Servlet资源
                      默认: 当前目录
         set-cookie的资源路径: http://localhost:8080/web03/cookie/Demo07Servlet

小结: 只有当cookie的name和path一致的时候才会认为是同一个cookie
     Cookie c1 = new Cookie("name","zhangsan");

     Cookie c2 = new Cookie("name","zhangsan);
     c2.setPath("/");
//创建cookie对象
        Cookie cookie=new Cookie("name","zhangsan");
        //通过响应将数据传到客户端的Cookie。本质是设置响应头set-cookie。
        cookie.setMaxAge(60*5);//设置cookie生存周期,单位是秒

        //设置cookie的被携带路径
        cookie.setPath("/");//服务器的所有路径
//        cookie.setPath("/cookieone_war_exploded2/GetCookies");//应用GetCookies携带
//        cookie.setPath("/cookieone_war_exploded2/");//此虚拟目录下所有应用携带
        //如果不设置cookie的被携带路径,会表示再当前路径下携带。比如@WebServlet("/pubg/CookieDome1"),pubg为当前目录

        response.addCookie(cookie);
        response.getWriter().write("向客户端设置cookie成功");
        -----------------------------------------------------
        //服务器获取客户端cookie
        //通过请求获取客户端所有cookie
        Cookie[] cookies = request.getCookies();
        if(cookies!=null){
            String name = cookies[0].getName();
            String value = cookies[0].getValue();

            response.getWriter().write("success: " + name + "," + value);
        }else
        {
            response.getWriter().write("null");
        }