
Web服务器收到客户端的http请求,会针对每一次请求,分别创建一个用于代表请求的request对象、和代表响应的response对象。
getWrite(); 字符输出流 getOutputStream(); 字节输出流
两者不能同时使用

案例:向客户端输出中文
@Override
	protected void doGet(HttpServletRequest req, HttpServletResponse resp) 
			throws ServletException, IOException {
		// 设置响应类型和字符编码为UTF-8, 这样支持汉字显示
//		resp.setContentType("text/html");
//		resp.setCharacterEncoding("UTF-8");
		
		//设置客户编码类型
		resp.setHeader("Content-Type", "text/html;charset=utf-8");
		//不设置编译类型,默认编码发送数据 ISO-8859-1(没有中国二字编码),此时会发生乱码
		resp.getWriter().write("hello你好");
	}常见的状态码参照:HTTP状态码详解
设置响应状态码

Override
	protected void doGet(HttpServletRequest req, HttpServletResponse resp) 
			throws ServletException, IOException {
		String filename = "Java基础考试题卷.docx";
		
		//下载文件
		String path = "C:/Users//Desktop/upload/" + filename;
		FileInputStream fis = new FileInputStream(path);
		
		//设置请求头,文件名需要UTF-8编码
		filename = URLEncoder.encode(filename, "UTF-8");
		resp.setHeader("Content-disposition", "attachment;filename=" + filename);
		byte[] bs = new byte[1024];
		ServletOutputStream sos = resp.getOutputStream();
		int len = 0;
		while((len = fis.read(bs)) != -1){
			sos.write(bs, 0, len);
		}
		
		fis.close();
	}实际开发直接调包
输出随机图片(CAPTCHA图像):Completely Automated Public Turing Test to Tell Computers and Humans Apart (全自动区分计算机和人类的测试)
相关主要类(JDK 查看API)
Sevlet
@Override
	protected void doGet(HttpServletRequest req, HttpServletResponse resp) 
			throws ServletException, IOException {
		int WIDTH=120;
		int HEIGHT=30;
		//创建画布
		BufferedImage image = new BufferedImage(WIDTH, HEIGHT, BufferedImage.TYPE_INT_RGB);
	
		//创建画笔
		Graphics g = image.getGraphics();
		
		//设置画笔颜色
		g.setColor(Color.YELLOW);
		
		//画背景
		g.fillRect(0, 0, WIDTH, HEIGHT);
		
		//设置干扰线颜色
		g.setColor(Color.BLUE);
		
		//加干扰线
		for(int i=0;i<6;i++){
			int xStart = new Random().nextInt(WIDTH);
			int yStart = new Random().nextInt(HEIGHT);
			int xEnd = new Random().nextInt(WIDTH);
			int yEnd = new Random().nextInt(HEIGHT);
			g.drawLine(xStart, yStart, xEnd, yEnd);
		}
		
		int x = 5;
		//画随机数
		for(int i = 0; i< 4;i++){
			g.drawString(new Random().nextInt(9) + "", x+i*30, 20);
		}
		
		//输出图片 设置响应格式
		resp.setContentType("image/jpeg");
		ImageIO.write(image, "JPEG",resp.getOutputStream());
	}html

效果

如何关联静态html和class 1.Html中的src写 /项目名/对应类的@WebServlet注解地址 2.对应类只需要加上路由注解 3.访问的时候为/项目名/html名字.html


ValidateCode(width,height,码数,干扰数)
请求重定向指:一个web资源收到客户端请求后,通知客户端去访问另外一个web资源,这称之为请求重定向。
重定向特点:地址栏会变,并发送2次请求,增加服务器负担
实现方式
response.sendRedirect(request.getContextPath()+”/转发注解”)实现原理: 302/307状态码和location头即可实现重定向
转发特点:地址栏不会变,客户端发送一次请求,状态码依旧200
request.getRequestDispatcher(“/转发注解”).forward(request,response)@Override
	protected void doGet(HttpServletRequest req, HttpServletResponse response) 
			throws ServletException, IOException {
		//浏览器刷新时就不会有缓存
		response.addHeader("Pragma", "no-cache"); 
		response.setHeader("Cache-Control", "no-cache"); 
		response.setHeader("Expires", "0");
		ValidateCode code = new ValidateCode(WIDTH, HEIGHT, 4, 6);
		
		code.write(response.getOutputStream());
	}protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		// TODO Auto-generated method stub
		response.setContentType("text/html;charset=UTF-8");
		//response.setHeader("Refresh", "1");//每隔一秒刷新一次
		
		response.setHeader("Refresh", "5;URL=index.html");//3秒后转到另一页面
		response.getWriter().write("注册成功!3秒后会自动跳转,苦没有中转点击<a href='index.html'>这里</a>");
	}HttpServletRequest对象代表客户端的请求,当客户端通过HTTP协议访问服务器时,HTTP请求头中的所有信息都封装在这个对象中,开发人员通过这个对象的方法,可以获得客户这些信息。


Map查询效率非常高,通常用上图方法来遍历Map(entrySet返回Entry,然后通过entry查询
目标:掌握通过getParameter和getParameterValues获取请求参数 掌握POST和GET请求解决乱码的方案 掌握表单提交通过都用POST方法
<form action="/项目名/注解内的地址" method="post">
用户名:<input type="text" name="username"><br>
密---码:<input type="password" name="password"><br>
性---别:<input type="radio" name="gender" value="男">男
	<input type="radio" name="gender" value="女">女<br>
爱---好:<input type="checkbox" name="hobby" value="写代码">写代码
<input type="checkbox" name="hobby" value="看书">看书
<input type="checkbox" name="hobby" value="听课">听课<br>
备---注:<textarea rows="5" cols="20" name="remark"></textarea><br>
<input type="submit" value="提交">
</form>
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws 	ServletException, IOException {
		//接收表单请求参数
	request.setCharacterEncoding("UTF-8");
	String username = request.getParameter("username");
	String password = request.getParameter("password");
	String gender = request.getParameter("gender");
	String remark = request.getParameter("remark");
	String[] hobby = request.getParameterValues("hobby");
	
	System.out.println(username);
	System.out.println(password);
	System.out.println(gender);
	System.out.println(remark);
	System.out.println(Arrays.toString(hobby));	
}Model

Servlet
public class Test extends HttpServlet{
	@Override
	protected void doPost(HttpServletRequest request, HttpServletResponse resp) throws ServletException, IOException {
		//post请求时,中文会乱码,解决方法setCharacterEncoding,只针对post请求有效
		request.setCharacterEncoding("UTF-8");
		
		//创建用户对象
		User user = new User();
		
		//获取请求参数
		/**
		 * key			value
		 * username		{"shunxu"};
		 * password     {"123"};
		 * hobby		{"study","reading"}
		 */
		Map<String, String[]> map = request.getParameterMap();
		try {
			for(Entry<String, String[]> entry : map.entrySet()){
				String pname = entry.getKey();
				//通过name创建对象
				if(pname.equals("username")){
					user.setUsername(entry.getValue()[0]);
				}else if(pname.equals("hobby")){
					user.setHobby(entry.getValue());
				}
			}
		} catch (Exception e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		
		System.out.println(user);
	}
}导入
如果不知道如何选择版本,比较靠谱的方式是去Maven Repository搜索包名,查看最多人用的版本

调包使用

BeanUtils自动将request.getParameterMap()中获取到的Map装载入user中。

请求转发指一个web资源收到客户端请求后,通知服务器去调用另外一个web资源进行处理。 request对象提供了一个getRequestDispatcher方法,该方法返回一个RequestDispatcher对象,调用这个对象的forward方法可以实现请求转发。 request对象同时也是一个域对象(context对象),开发人员通过request对象在实现转发时,把数据通过request对象带给其它web资源处理。

重定向机制的运作流程
HttpServeltResponse的sendRedirect(String location)用于重定向
@WebServlet("/test")
public class Test extends httpServlet{
	private static final long serialVersionUID =1L;
	protected void doGet(HttpServletRequest request,HttpServeltResponse response)
			throws Exception{
		String cp = request.getContextPath();
		System.out.println(cp);   //打印上下文路径
		response.sendRedirect(cp+"/login.jsp"); //跳转到jsp页面,需要加上contextPath
}域对象:session,request,page等(详情关注后续章节) 域:区间、范围
转发存放在Request中的变量不会失效

顾名思义先翻译:
假定第一次请求的为servlet1,处理转发的为servlet2
include()方法将请求转发给servlet2,servle2对该请求做出了的响应并入到原来的servlet1响应对象中,原来的servlet1还可以继续输出响应信息(servlet1和servlet2按顺序依次响应数据,就像是一个页面一样)。
forward方法将请求转发给其他的servlet2,servlet2负责对请求做出响应,而原先的servlet1的执行则终止(servlet1无法响应任何数据)。
sendRedict()在浏览器请求servlet1之后,重新告诉浏览器将请求重新定位到servlet。

具体结果读者自行尝试