java生成图片验证码返回base64图片信息

一般在生成验证码,有两种形式:

前端js生成,并在前端验证通过,提交表单到后台。安全性较低,很容易被识别
后端生成返回给前端展示,提交用户验证码到后台验证通过。安全性较前端方式高,因为整个验证过程在后台。
其实验证码也没有绝对的安全,只能说后端的处理方式比前端好一些而已,所以这个示例是利用java后台生成二维码图片,并返回给前端base64格式,展示给用户,后台将二维码真值保存在session,用户提交输入的验证码和保存的真值做比较,然后返回前台是否验证成功。

1.创建一个验证码生成工具类。此类为静态类,
import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.Serializable;
import java.util.Random;

import javax.imageio.ImageIO;

import org.apache.commons.lang3.RandomUtils;

import com.alibaba.druid.util.Base64;

/**
*
* @ClassName: ValidateCodeUtil
* @Description: 验证码生成工具类
* @author chenhx
* @date 2017年11月14日 上午11:00:07
*/
public class ValidateCodeUtil {
private static Validate validate = null; //验证码类,用于最后返回此对象,包含验证码图片base64和真值
private static Random random = new Random(); //随机类,用于生成随机参数
private static String randString = “0123456789abcdefghijkmnpqrtyABCDEFGHIJLMNQRTY”;//随机生成字符串的取值范围

private static int width = 80;     //图片宽度  
private static int height = 34;    //图片高度  
private static int StringNum = 4;  //字符的数量  
private static int lineSize = 40;  //干扰线数量  


//将构造函数私有化 禁止new创建
private ValidateCodeUtil() {
    super();
}

/** 
 * 获取随机字符,并返回字符的String格式 
 * @param index (指定位置) 
 * @return 
 */  
private static String getRandomChar(int index) {  
    //获取指定位置index的字符,并转换成字符串表示形式  
    return String.valueOf(randString.charAt(index));  
}  
/**
 * 获取随机指定区间的随机数 
 * @param min (指定最小数) 
 * @param max (指定最大数) 
 * @return 
 */
private static int getRandomNum(int min,int max) { 
    return RandomUtils.nextInt(min, max);
}  

/** 
 * 获得字体 
 * @return 
 */  
private static Font getFont() {  
    return new Font("Fixedsys", Font.CENTER_BASELINE, 25);  //名称、样式、磅值  
}  

/** 
 * 获得颜色 
 * @param fc 
 * @param bc 
 * @return 
 */  
private static Color getRandColor(int frontColor, int backColor) {  
    if(frontColor > 255)  
        frontColor = 255;  
    if(backColor > 255)  
        backColor = 255;  

    int red = frontColor + random.nextInt(backColor - frontColor - 16);  
    int green = frontColor + random.nextInt(backColor - frontColor -14);  
    int blue = frontColor + random.nextInt(backColor - frontColor -18);  
    return new Color(red, green, blue);  
}  

/** 
 * 绘制字符串,返回绘制的字符串 
 * @param g 
 * @param randomString 
 * @param i 
 * @return 
 */  
private static String drawString(Graphics g, String randomString, int i) { 
    Graphics2D g2d = (Graphics2D) g;
    g2d.setFont(getFont());   //设置字体  
    g2d.setColor(new Color(random.nextFloat(), random.nextFloat(), random.nextFloat()));//设置颜色  
    String randChar = String.valueOf(getRandomChar(random.nextInt(randString.length())));  
    randomString += randChar;   //组装  
    int rot = getRandomNum(5,10);
    g2d.translate(random.nextInt(3), random.nextInt(3)); 
    g2d.rotate(rot * Math.PI / 180);
    g2d.drawString(randChar, 13*i, 20);   
    g2d.rotate(-rot * Math.PI / 180);
    return randomString;  
}  

/** 
 * 绘制干扰线 
 * @param g 
 */  
private static void drawLine(Graphics g) {  
    //起点(x,y)  偏移量x1、y1  
    int x = random.nextInt(width);  
    int y = random.nextInt(height);  
    int xl = random.nextInt(13);  
    int yl = random.nextInt(15);  
    g.setColor(new Color(random.nextFloat(), random.nextFloat(), random.nextFloat()));
    g.drawLine(x, y, x + xl, y + yl);  
}  

/**
 * 
 * @MethodName: getRandomCode
 * @Description: 生成Base64图片验证码
 * @param key
 * @return String 返回base64
 */
public static Validate getRandomCode() {  
    validate = validate==null?new Validate():validate;

    // BufferedImage类是具有缓冲区的Image类,Image类是用于描述图像信息的类  
    BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_BGR);  
    Graphics g = image.getGraphics();// 获得BufferedImage对象的Graphics对象  
    g.fillRect(0, 0, width, height);//填充矩形  
    g.setFont(new Font("Times New Roman", Font.ROMAN_BASELINE, 18));//设置字体  
    g.setColor(getRandColor(110, 133));//设置颜色  
    //绘制干扰线  
    for(int i = 0; i <= lineSize; i++) {  
        drawLine(g);  
    }  
    //绘制字符  
    String randomString = "";  
    for(int i = 1; i <= StringNum; i++) {  
        randomString = drawString(g, randomString, i); 
        validate.setValue(randomString);
    }  

    g.dispose();//释放绘图资源  
    ByteArrayOutputStream bs = null;
    try {  
        bs = new ByteArrayOutputStream();
        ImageIO.write(image, "png", bs);//将绘制得图片输出到流  
        String imgsrc = Base64.byteArrayToBase64(bs.toByteArray());
        validate.setBase64Str(imgsrc);
    } catch (Exception e) {  
        e.printStackTrace();  
    } finally {  
        try {
            bs.close();
        } catch (IOException e) {
            e.printStackTrace();
        }finally{
            bs = null;
        }
    }     
    return validate;
}  

/**
 * 
 * @ClassName: Validate
 * @Description: 验证码类
 * @author chenhx
 * @date 2017年11月14日 上午11:35:34
 */
public static class Validate implements Serializable{
    private static final long serialVersionUID = 1L;
    private String Base64Str;       //Base64 值
    private String value;           //验证码值

    public String getBase64Str() {
        return Base64Str;
    }
    public void setBase64Str(String base64Str) {
        Base64Str = base64Str;
    }
    public String getValue() {
        return value;
    }
    public void setValue(String value) {
        this.value = value;
    }
}

}

2.在java接口中使用:
Validate v = ValidateCodeUtil.getRandomCode(); //直接调用静态方法,返回验证码对象
if(v!=null){
session.setAttribute(“validate”, MD5Util.md5(v.getValue().toLowerCase())); //将验证码值保存session
result.setResult(v.getBase64Str());
result.setResultCode(0);
result.setMsg(“刷新验证码成功”);
}

3.前端接收返回的base64,然后展示在页面上
点击刷新

具体的ajax请求什么的,我就不详细写了,反正关键信息就是这些。最后效果是这样

ValidateCodeUtil类中使用了其他的jar包,org.apache.commons.lang3.RandomUtils和com.alibaba.druid.util.Base64。

  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值