【学习笔记】Response

1.Response的运行流程

2.通过抓包工具抓取Http响应

通过response设置响应行:

设置响应行的状态码:setStatus(int sc)
设置响应头:
addHeader(String name,String value) , addIntHeader(String name,Intvalue) , addDataHeader(String name,Long value)
setHeader(String name,String value) , setIntHeader(String name,Int value) , setDateHeader(String name,Long value)
其中,add表示添加(添加多个可以全部显示),而set表示设置(只能显示一个)
设置响应体一:向response写入字符流:PrintWriter getWriter()方法可以获取字符流,通过write()方法把字符串写到response缓冲区,之后tom内核把response缓冲区的内容组装成http响应返回给客户端。
中文乱码问题:原因:response缓冲区的编码格式是ISO8859-1,此码表没有中文,可通过response的setCharacterEncoding()设置response的编码。但还会乱码,因为浏览器默认编码是GBK,需要通过response.setHeader("Content-Type", "text/html;charset=utf-8")告知客户端使用何种编码格式进行解析。
还有一种更加简洁的方法:response.setContentType("text/html;charset=utf-8"),这个方法可以同时指定response和浏览器解析的编码,所以在实际开发中使用这个方法。

字节和字符:字节(byte)是计算存储容量的单位,字符是计算机中使用的文字和符号,字符串。

设置响应体二:向Response写入字节流:ServletOutputStream getOutputStream()方法获得字节流,通过字节流的write(byte[] bytes)可以将图片视频文本等文件写入response缓冲区,再由tomcat内核组装成http响应返回给浏览器。
注意:response.getOutputStream()和response.getWriter()只能同时存在一个。
写入图片示例:

ServletOutputStream out = response.getOutputStream();
		//获得服务器上的图
		String path = this.getServletContext().getRealPath("download/a.jpg");
		InputStream in=new FileInputStream(path);//fileInputStream 是InputStream 的实现类  
		int len=0;
		byte[] buffer=new byte[1024];
		while((len=in.read(buffer))>0) {
			out.write(buffer, 0, len);
		}
		out.close();
		in.close();

案例:文件下载(中文乱码处理):

响应头的filename属性要传给客户端,客户端需要有对应的解码方式,否则会乱码

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<body>
<a href="/Test002/download?filename=美.jpg">美.jpg</a><br>
<a href="/Test002/download?filename=a.flv">a.flv</a><br>
<a href="/Test002/download?filename=a.mp3">a.mp3</a><br>
</body>
</html>
package com.xing.download;

import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URLEncoder;

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

import sun.misc.BASE64Encoder;

@WebServlet("/download")
public class download extends HttpServlet {
	private static final long serialVersionUID = 1L;
	protected void doGet(HttpServletRequest request, HttpServletResponse response)
			throws ServletException, IOException {

		String filename = request.getParameter("filename");//此处显示中文正常,浏览器能识别
		//filename属性是下载的文件名称,但传入中文会乱码,解决方法:
		//filename属性设置方式一:
//		String fname=URLEncoder.encode(filename,"UTF-8");
		// 告知浏览器文件打开方式是下载,filename属性是下载的文件名称,但此处传入的是中文,会乱码。解决:filename="+fname+";"+"filename*=utf-8''"+fname
//		response.setHeader("Content-Disposition", "attachment;filename="+fname+";"+"filename*='utf-8'"+fname);
		
		
		//filename属性设置方式二:针对不同浏览器设置不同解码方式
		//获得请求头中的User-Agent
		String agent = request.getHeader("User-Agent");
		//根据不同浏览器进行不同的编码
		String filenameEncoder = "";
		if (agent.contains("MSIE")) {
			// IE浏览器
			filenameEncoder = URLEncoder.encode(filename, "utf-8");
			filenameEncoder = filenameEncoder.replace("+", " ");
		} else if (agent.contains("Firefox")) {
			// 火狐浏览器
			BASE64Encoder base64Encoder = new BASE64Encoder();
			filenameEncoder = "=?utf-8?B?"
					+ base64Encoder.encode(filename.getBytes("utf-8")) + "?=";
		} else {
			// 其它浏览器
			filenameEncoder = URLEncoder.encode(filename, "utf-8");				
		}
        //告知浏览器文件打开方式是下载,filename属性是下载的文件名称,传入中文浏览器不能解析
		response.setHeader("Content-Disposition", "attachment;filename="+filenameEncoder);
		
		
		ServletOutputStream out = response.getOutputStream();
		String path = this.getServletContext().getRealPath("download/" + filename);
		InputStream in = new FileInputStream(path);
		byte[] buffer = new byte[1024];
		int len;
		while ((len = in.read(buffer)) > 0) {
			out.write(buffer, 0, len);
		}
		in.close();

	}

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

}

乱码心得:使用getWriter().write向response写入字符流乱码,使用response.setContentType("text/html;charset=utf-8")解决(一般出现在页面中文乱码);使用setHeader()把中文写入响应头,浏览器不能解析,乱码(下载名称中文乱码),需要针对不同浏览器设置解析格式。

3.重定向流程图(sendRedirect())

response.sendRedirect("login.html");

案例:验证码:

package cn.itcast.estore.web.servlet;

import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
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 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);
	}

}
一唱一和
一呼百应
一干二净
一举两得
一落千丈
一模一样
一暴十寒
一日千里
一五一十
一心一意
两面三刀
三长两短
三番五次
三三两两
三头六臂
三心二意
三言两语
四分五裂
四面八方
四通八达
四平八稳
五光十色
五湖四海
五花八门
五颜六色
六神无主
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>

<script type="text/javascript">
	function changeCode(obj){
		obj.src="/Test002/code?time="+new Date().getTime();
	}

</script>
</head>
<body>
<div style="width=100px;height=100px;">
	验证码:<input type="text" name="username"><img onclick="changeCode(this)" src="/Test002/code"><br/>
</div>
</body>
</html>

 

补充:服务器端WEB-INF/classes文件夹相当于项目工程的src文件夹,类是编译过的。读取WEB-INF/classes里的文件,可通过ServletContext对象或者类加载器。

	@Override
	protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
		//servletContext读取WEB-INF下的web.xml
		ServletContext servletContext = this.getServletContext();
		String driver = servletContext.getInitParameter("driver");
		System.out.println(driver);
		//类加载器读取读取WEB-INF下的txt
		String path = getNum.class.getClassLoader().getResource("../new_words.txt").getPath();//路径从classes里的com开始,注意txt文件位置
		System.out.println(path);
	}

读取包下的txt路径改为:getResource("com/xing/getWeb/new_words.txt")

转载于:https://my.oschina.net/u/3943244/blog/1928280

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我可以和你讨论Servlet中的Filter和Listener。 在Servlet中,Filter和Listener是两个重要的组件,它们可以用于在Servlet容器中拦截请求和响应,以及处理Servlet容器中的事件。 Filter是一种拦截器,它可以在请求被Servlet处理之前或响应被发送回客户端之前拦截请求和响应,并对它们进行修改或处理。Filter可以用于对请求进行身份验证、授权、日志记录、压缩、加密等。 在Servlet中,Filter是通过实现javax.servlet.Filter接口来创建的。Filter接口有三个方法:init()、doFilter()和destroy()。其中,init()方法用于初始化Filter,doFilter()方法用于对请求和响应进行处理,destroy()方法用于销毁Filter。 下面是一个简单的Filter示例,它可以对请求进行日志记录: ```java public class LogFilter implements Filter { public void init(FilterConfig config) throws ServletException { // 初始化Filter } public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { // 记录请求日志 System.out.println("Request received from " + request.getRemoteAddr()); // 继续处理请求 chain.doFilter(request, response); // 记录响应日志 System.out.println("Response sent to " + request.getRemoteAddr()); } public void destroy() { // 销毁Filter } } ``` Listener是一种监听器,它可以监听Servlet容器中的事件,例如Servlet的创建和销毁、Session的创建和销毁、ServletContext的创建和销毁等。Listener可以用于在Servlet容器中实现一些全局的业务逻辑,例如初始化数据库连接池、加载配置文件等。 在Servlet中,Listener是通过实现javax.servlet.ServletContextListener接口来创建的。ServletContextListener接口有两个方法:contextInitialized()和contextDestroyed()。其中,contextInitialized()方法在ServletContext被创建时调用,可以用于初始化全局资源;contextDestroyed()方法在ServletContext被销毁时调用,可以用于释放全局资源。 下面是一个简单的Listener示例,它可以在ServletContext被创建时初始化数据库连接池: ```java public class ConnectionPoolListener implements ServletContextListener { public void contextInitialized(ServletContextEvent event) { // 初始化数据库连接池 ConnectionPool pool = new ConnectionPool(); event.getServletContext().setAttribute("connectionPool", pool); } public void contextDestroyed(ServletContextEvent event) { // 释放数据库连接池 ConnectionPool pool = (ConnectionPool) event.getServletContext().getAttribute("connectionPool"); pool.release(); } } ``` 以上就是Servlet中Filter和Listener的简单介绍,希望能对你有所帮助。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值