Servlet的重定向与转发的区别(附做验证码的例子)
学习Javaweb,不能不清楚servlet中的重定向和转发的区别,虽然两种方式都可以实现页面跳转,但是混用可能会发生盗链问题。
先来总结,再来详细看看区别在哪:
重定向的特点:
1. 地址栏会发生变化,显示的是转向的其他站点的地址
2. 重定向可以访问其他网站的资源
3. 重定向对浏览器做的是两次请求,并且不能使用request对象来存储和共享数据
4. 重定向是客户端行为
转发的特点:
1. 转发中地址栏路径不会发生变化,不会显示出转向的其他站点的地址
2. 转发只能访问当前服务器下的资源
3. 转发对浏览器做的是一次请求,可以使用request对象来共享数据
4. 转发是服务器行为
接下来看看解释:
1.重定向访问过程结束后,浏览器地址栏中显示的URL地址会发生改变,由初始的URL地址变成另外一个站点的URL;请求转发过程结束后,浏览器地址栏URL仍然指向开始的页面。
2.值得注意的是,我们可以利用response.sendRedirect() 方法重定向到当前应用程序中的其他资源,也可以重定向到同一个站点上的其他应用程序中的资源,还可以使用重定向到其他站点的资源。虽然转发速度更快(重定向需要经过客户端,而转发不需要。),还能保持request内的对象,但是它只能将请求转发给当前WEB的应用程序,不能访问其他站点资源,如果用转发来访问其他站点资源,容易造成盗链问题,很可能访问不到其他站点资源。
3.这里其实是很容易理解的,因为在重定向中,它会向服务器报告状态码302,然后服务器会告诉客户端要转向哪个地址,客户端再自己去请求转向的地址,因此,状态栏中会显示转向站点的地址,所以说,客户端对浏览器至少进行了两次的访问请求。
而在转发中,是服务器自己代替客户端去请求其他站点资源,然后返回给客户端。客户端并不知道服务器帮他去访问了,因此状态栏中地址不会显示出转向的地址,还是保留原来的地址。
举个例子,就好比如你去村委会叫他们帮你办件事,但是村委会一看这事他们帮不了忙,叫你自己去镇政府找人帮忙,然后你就拿着村委会给的地址去镇政府,找到人解决了事情。然后你来看看,你是不是一共去了两个地方,访问了两个站点?这就很好地解释了为什么重定向对浏览器做的是两次请求了。
再来看看转发,你去村委会叫他们帮你办件事,村委会一看这事他们帮不了忙,但是他们叫你在这里等着,他自己去打电话给镇政府叫人来帮忙,然后事情解决了。在这个过程中,你不知道村委会去叫人来帮忙了,你还以为是村委会帮你解决的,这也是为什么说转发容易造成盗链的原因了。我们来看,在这个过程中你只去过村委会,只访问一个站点,因此转发对浏览器做的是一次请求。
4.重定向经过客户端,需要客户端自己另外去访问其他站点,而转发只是在服务器内完成了。
实现代码:
//转发,转发到other.html
request.getRequestDispatcher("other.html").forward(request, response);
//重定向,重定向到new.html
response.sendRedirect("new.html");
附做最基础最低级验证码的代码:
html代码:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<script>
/*
点击刷新验证码
步骤:
1.给超链接和图片绑定单击事件
2.重新设置图片的src属性值
*/
window.onload = function(){
//1.获取图片对象
var img = document.getElementById("checkCode");
//2.绑定单击事件
img.onclick = function(){
//加时间戳
var date = new Date().getTime();
img.src = "/day15/checkCodeServlet?"+date;
}
}
</script>
</head>
<body>
<img id="checkCode" src="/day15/checkCodeServlet" />
<a id="change" href="">看不清换一张?</a>
</body>
</html>
Java代码:
package Response_Study;
/**
做验证码
*/
import javax.imageio.ImageIO;
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 java.awt.*;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.util.Random;
@WebServlet("/checkCodeServlet")
public class CheckCodeServlet extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
int width = 100;
int height = 50;
//1.创建一对象,在内存中图片(验证码图片对象)
BufferedImage image = new BufferedImage(width,height,BufferedImage.TYPE_INT_RGB);
//2.美化图片
//第一步: 填充背景色
Graphics g = image.getGraphics();//画笔对象
g.setColor(Color.yellow);//设置画笔颜色
g.fillRect(0,0,width,height);
//第二步:1.画边框
g.setColor(Color.BLUE);
g.drawRect(0,0,width - 1,height - 1);
String str = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghigklmnopqrstuvwxyz0123456789";
//第二步:2.生成验证码随机角标
Random ran = new Random();
//第三步:写验证码
for (int i = 1; i <= 4; i++) {
int index = ran.nextInt(str.length());
//获取字符
char ch = str.charAt(index);//随机字符
//验证码关键步骤
g.drawString(ch+"",width/5*i,height/2);
}
//第四步:画干扰线
g.setColor(Color.RED);
//随机生成干扰线坐标点
for (int i = 0; i < 10; i++) {
int x1 = ran.nextInt(width);
int x2 = ran.nextInt(width);
int y1 = ran.nextInt(height);
int y2 = ran.nextInt(height);
g.drawLine(x1,y1,x2,y2);
}
//3.将图片输出到页面展示
ImageIO.write(image,"jpg",response.getOutputStream());
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request,response);
}
}
效果:
总结:在javaweb中servlet是比较重要的一个点,我们要牢牢掌握,虽然目前只是学习了一些皮毛,但是多个皮毛汇起来就是一件毛衣了,暖不暖和就不知道了,起码还可以穿吧。。