1.图片校验码

<img  src="captcha.jpg"  />

web.xml配置

<servlet> 
    <servlet-name>CaptchaServlet</servlet-name>
     <servlet-class>anni.CaptchaServlet</servlet-class>
 </servlet> 
<servlet-mapping> 
    <servlet-name>CaptchaServlet</servlet-name>
     <url-pattern>/captcha.jpg</url-pattern>
 </servlet-mapping>

名叫/captcha.jpg的请求会交给CaptchaServlet处理

jsp和servlet并不是只能返回html格式的数据,实际上它们可以生成任意格式的数据,比如这里我们就用servlet生成了一个图片。

浏览器只是向服务器发送了一个请求,这个请求的地址是/captcha.jpg还是/index.jsp并没有什么区别,在服务器看来他们仅仅是一个字符串而已,接收到请求后服务器先去按照web.xml中的配置做映射,将请求交给对应的servlet处理,如果web.xml中没有对应这个请求的映射,才会去磁盘查找是否有这么一个文件,找到文件则输出到响应中传回客户端,如果找不到就返回经典的404(找不到访问资源)。

img只可能通过GET方式发送请求

public void doGet(HttpServletRequest request, HttpServletResponse response)     throws ServletException, IOException {  
    //设置页面不缓存 
    response.setHeader("Pragma", "No-cache");     response.setHeader("Cache-Control", "no-cache");     response.setDateHeader("Expires", 0);  
    // 在内存中创建图象 
    int width = 60, height = 20; 
    BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);  
    // 获取图形上下文 
    Graphics g = image.getGraphics();  
    //生成随机类 
    Random random = new Random();
    // 设定背景色 
    g.setColor(getRandColor(200, 250));     g.fillRect(0, 0, width, height);  
    //设定字体 
    g.setFont(new Font("Times New Roman", Font.PLAIN, 18));  
    //画边框 
    //g.setColor(new Color()); 
    //g.drawRect(0, 0, width - 1, height - 1); // 随机产生155条干扰线,使图象中的认证码不易被其它程序探测到     g.setColor(getRandColor(160, 200));     for (int i = 0; i < 155; i++) {         int x = random.nextInt(width);         int y = random.nextInt(height);         int xl = random.nextInt(12);         int yl = random.nextInt(12);         g.drawLine(x,y,x+xl,y+yl);     }  
    // 取随机产生的认证码(4位数字)     String sRand = ""; 
    for (int i = 0;i < 4; i++) { 
        String rand = String.valueOf(random.nextInt(10)); 
sRand += rand; 
        // 将认证码显示到图象中 
        // 调用函数出来的颜色相同,可能是因为种子太接近,所以只能直接生成 
        g.setColor(new Color(20 + random.nextInt(110), 20 + random.nextInt(110), 20 + random.nextInt(110)));         g.drawString(rand, 13 * i + 6, 16);     }  
    // 将认证码存入SESSION 
    request.getSession().setAttribute("captcha", sRand);  
    // 图象生效     g.dispose();  
    // 输出图象到页面 
    ImageIO.write(image, "JPEG", response.getOutputStream()); }

代码最先设置response(响应)中的头部配置,告诉浏览器不要缓存对/captcha.jpg的请求结果,这样才能保证每次刷新页面都看到最新生成的图片,要是设置了缓存很可能每次看到的都是最先请求看到的图片。

中间一大段代码实现动态生成图片的功能,我们先随机获得几个数字,然后写到BufferedImage中,最后就可以把图片数据写到response,因为图片是二进制数据,所以我们使用了response.getOutputStream()而不是response.getWriter()。

为了达到验证的功能,每次生成图片之后要记得讲随机得到的数字保存到session中,session中的变量可以跨越多个请求周期存在,等用户输入验证码提交后就能与session中的数据做比较了,这些是在CheckServlet中实现的。

public void doPost(HttpServletRequest request, HttpServletResponse response)     throws ServletException, IOException {  
    HttpSession session = request.getSession();  
    String requestCaptcha = request.getParameter("captcha"); 
    String sessionCaptcha = (String) session.getAttribute("captcha");  
    if (sessionCaptcha != null && sessionCaptcha.equals(requestCaptcha)) {         session.removeAttribute("captcha"); 
        request.getRequestDispatcher("/success.jsp").forward(request, response);     } else { 
        request.setAttribute("message", "验证码输入错误"); 
        request.getRequestDispatcher("/index.jsp").forward(request, response);     } }

2.图片验证码 JSP版本

<%@ page contentType="image/jpeg" import="java.awt.*,java.awt.image.*,java.util.*,javax.imageio.*" %> <%! 
// 给定范围获得随机颜色 
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); } %> <% 
//设置页面不缓存 
response.setHeader("Pragma", "No-cache"); response.setHeader("Cache-Control", "no-cache"); response.setDateHeader("Expires", 0);  
// 在内存中创建图象 
int width = 60, height = 20; 
BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);  
// 获取图形上下文 
Graphics g = image.getGraphics();  
//生成随机类 
Random random = new Random();   // 设定背景色 
g.setColor(getRandColor(200,250)); g.fillRect(0, 0, width, height);  
//设定字体 
g.setFont(new Font("Times New Roman", Font.PLAIN, 18));  
//画边框 
//g.setColor(new Color()); 
//g.drawRect(0,0,width-1,height-1); 
  
// 随机产生155条干扰线,使图象中的认证码不易被其它程序探测到 
g.setColor(getRandColor(160, 200)); for (int i = 0; i < 155; i++) {     int x = random.nextInt(width);     int y = random.nextInt(height);     int xl = random.nextInt(12);     int yl = random.nextInt(12);     g.drawLine(x,y,x+xl,y+yl); }  
// 取随机产生的认证码(4位数字) 
String sRand = ""; 
for (int i = 0;i < 4; i++) { 
    String rand = String.valueOf(random.nextInt(10));     sRand += rand; 
    // 将认证码显示到图象中 
    // 调用函数出来的颜色相同,可能是因为种子太接近,所以只能直接生成 
    g.setColor(new Color(20 + random.nextInt(110), 20 + random.nextInt(110), 20 + random.nextInt(110)));     g.drawString(rand, 13 * i + 6, 16); }  
// 将认证码存入SESSION 
session.setAttribute("captcha", sRand);  
// 图象生效 
g.dispose();  
// 输出图象到页面 
ImageIO.write(image, "JPEG", response.getOutputStream());  
out.clear(); 
out = pageContext.pushBody(); 

在jsp中使用response.getOutputStream()很可能引起一个问题。

getOutputStream() has already been called for this response

在tomcat5下jsp中出现此错误,一般都是在jsp中使用了输出流(如输出图片验证码,文件下载等)后没有妥善处理好。

具体的原因就是在tomcat中,jsp转换成servlet之后在函数_jspService(HttpServletRequest request, 
HttpServletResponse response)的最后有一段这样的代码

finally { 
    if (_jspxFactory != null) _jspxFactory.releasePageContext(_jspx_page_context); } 

这里是在释放在jsp中使用的对象,会调用response.getWriter(),因为这个方法是和
response.getOutputStream()相冲突的,所以会出现以上这个异常。 
然后当然是要提出解决的办法,其实挺简单的(并不是和某些朋友说的那样--将jsp内的所有空格和回车符号所有都删除掉),在使用完输出流以后调用以下两行代码即可:

out.clear(); 
out = pageContext.pushBody();

JSP入门 导出文件的更多相关文章

  1. Jsp入门学习笔记

    #Jsp入门 一.JSP基础语法 1.JSP指令: page inlcude taglib 2.JSP注释: a.html注释: <!-- abcdefghijklmn --> b.jsp ...

  2. JSP入门:介绍什么是JSP和Servlet(转)

    转自:http://developer.51cto.com/art/200907/134506.htm JSP入门:什么是jsp? JSP(Java Server Pages)是由Sun Micros ...

  3. JSP入门必读

    JSP基础知识:转自老师上课梳理的笔记,希望对大家有所帮助.有什么不妥当的地方还望大家批评指正. 特别适用于JSP入门的人员使用.1.JSP [1] 简介1.1 HTML    HTML擅长显示一个静 ...

  4. 九、JSP入门(1)

    JSP入门 1 JSP概述 1.1 什么是JSP JSP(Java Server Pages)是JavaWeb服务器端的动态资源.它与html页面的作用是相同的,显示数据和获取数据. 1.2 JSP的 ...

  5. day11(jsp入门&Cookie&HttpSession&一次性图片校验码)

    day11 JSP入门   1 JSP概述 1.1 什么是JSP JSP(Java Server Pages)是JavaWeb服务器端的动态资源.它与html页面的作用是相同的,显示数据和获取数据. ...

  6. JSP入门之自定义标签

    第二部分简单讲解:主要讲解el表达式,核心标签库.本章主要讲解:自定义标签库:404页面,505页面,错误页面配置方法 全部代码下载:链接 1.JSP自定义标签: 自定义标签是用户定义的JSP语言元素 ...

  7. Jsp入门实战上

    前面讲了servlet入门实践现在开始介绍jsp入门实践,开发环境的搭建请参考我前面的tomcat的文章,jsp入门教程分为上下两部分,第一部分简单讲解:jsp语法的规范,以及三大编译指令,七个动作指 ...

  8. Cookie&&Session&&jsp入门

    会话技术 会话:一次会话中包含多次请求和响应. 一次会话:浏览器第一次给服务器资源发送请求,会话建立,直到有一方断开为止 功能:在一次会话的范围内的多次请求间,共享数据 方式: 客户端会话技术:Coo ...

  9. JSP入门

    JSP简介 所谓JSP就是在网页文件中嵌入Java代码或JSP定义的一些标记.JSP是建立在Servlet上的,在执行时JSP容器会先将JSP文件转换成Servlet文件以及class 文件,然后再执 ...

随机推荐

  1. thinkjs—控制器方法名不能大写

    async updateInfoAction(){ ... } 上面的接口如果通过ajax访问,就会报404的错误.原因似乎在于访问updateInfo时,会自动转化成小写,而小写的updateinf ...

  2. [知了堂学习笔记]_纯JS制作《飞机大战》游戏_第1讲(实现思路与游戏界面的实现)

    整体效果展示: 一.实现思路 如图,这是我完成该项目的一个逻辑图,也是一个功能模块完成的顺序图. 游戏界面的完成 英雄飞机对象实现,在实现发射子弹方法过程中,又引出了子弹对象并实现.在此时,英雄飞机能 ...

  3. Python学习笔记3

    __slots__ 如果我们想要限制class的属性怎么办?比如,只允许对Student实例添加name和age属性. 为了达到限制的目的,Python允许在定义class的时候,定义一个特殊的__s ...

  4. [2013-06-05]bat脚本设置DNS

    有时候需要切换本机dns,将网络环境转至测试环境 @echo off netsh interface ip set dns name="本地连接" source=static ad ...

  5. oracle导库

    cmd窗口直接输入导库命令即可,不需要进入sqlplus C:\Documents and Settings\Administrator> imp username/pass@orcl file ...

  6. .net分布式压力测试工具(Beetle.DT)

    肯定有人会问为什么会写这样一个开源工具?和现有的有什么差别?不过对于一个程序员来说写东西还真不需要理由的:),主要原因是工作有点闲(开玩笑),不过说实话一个程员怎可能会停止写代码呢(作为一个奔4的程序 ...

  7. spring mvc:注解@ModelAttribute妙用

    在Spring mvc中,注解@ModelAttribute是一个非常常用的注解,其功能主要在两方面: 运用在参数上,会将客户端传递过来的参数按名称注入到指定对象中,并且会将这个对象自动加入Model ...

  8. Tomcat Cluster负载均衡

    author:JevonWei 版权声明:原创作品 Tomcat Cluster负载均衡 环境 tomcatA 172.16.253.108 tomcatB 172.16.253.105 代理服务器 ...

  9. [js高手之路]深入浅出webpack教程系列7-( babel-loader,css-loader,style-loader)的用法

    什么是loader呢,官方解释为文件的预处理器,通俗点说webpack在处理静态资源的时候,需要加载各种loader,比如,html文件,要用html-loader, css文件要用css-loade ...

  10. vue 父子组件传参

    父向子组件传参 例子:App.vue为父,引入componetA组件之后,则可以在template中使用标签(注意驼峰写法要改成componet-a写法,因为html对大小写不敏感,component ...