页面中验证码的生成总结(struts2)

下面是结合用Struts2 来生成的例子

感谢原文作者的贡献,这个是原文地址:http://mxl86.iteye.com/blog/300128

 

 

1、login.jsp页面程序

<script type="text/javascript"> 
function changeValidateCode(obj) { 
//获取当前的时间作为参数,无具体意义 
var timenow = new Date().getTime(); 
//每次请求需要一个不同的参数,否则可能会返回同样的验证码 
//这和浏览器的缓存机制有关系,也可以把页面设置为不缓存,这样就不用这个参数了。 
obj.src="rand.action?d="+timenow; 
} 
</script>

在表单中添加下面这句话:
<s:text name="random"></s:text>:<s:textfield name="rand" size="5"></s:textfield><img src="rand.action" title="点击图片刷新验证码"/>

 

 

2、RandomNumUtil.java 生成验证码的类文件

import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.image.BufferedImage;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.util.Random;
import javax.imageio.ImageIO;
import javax.imageio.stream.ImageOutputStream;
public class RandomNumUtil { 
private ByteArrayInputStream image;//图像 
private String str;//验证码 

private RandomNumUtil(){ 
init();//初始化属性 
} 
/* 
* 取得RandomNumUtil实例 
*/ 
public static RandomNumUtil Instance(){ 
return new RandomNumUtil(); 
} 
/* 
* 取得验证码图片 
*/ 
public ByteArrayInputStream getImage(){ 
return this.image; 
} 
/* 
* 取得图片的验证码 
*/ 
public String getString(){ 
return this.str; 
} 

private void init() { 
// 在内存中创建图象 
int width=85, 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)); 
// 随机产生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); 
} 
// 取随机产生的认证码(6位数字) 
String sRand=""; 
for (int i=0;i<6;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); 
}
//赋值验证码
this.str=sRand; 

//图象生效 
g.dispose(); 
ByteArrayInputStream input=null; 
ByteArrayOutputStream output = new ByteArrayOutputStream(); 
try{ 
ImageOutputStream imageOut = ImageIO.createImageOutputStream(output); 
ImageIO.write(image, "JPEG", imageOut); 
imageOut.close(); 
input = new ByteArrayInputStream(output.toByteArray()); 
}catch(Exception e){ 
System.out.println("验证码图片产生出现错误:"+e.toString()); 
} 

this.image=input;/* 赋值图像 */ 
} 
/* 
* 给定范围获得随机颜色 
*/ 
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); 
}
}

 

3、RandomAction.java  生成验证码的action程序

import java.io.ByteArrayInputStream;
import com.mxl.util.RandomNumUtil;
import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ActionSupport;
public class RandomAction extends ActionSupport{ 
private ByteArrayInputStream inputStream; 
public String execute() throws Exception{ 
RandomNumUtil rdnu=RandomNumUtil.Instance(); 
this.setInputStream(rdnu.getImage());//取得带有随机字符串的图片 
ActionContext.getContext().getSession().put("random", rdnu.getString());//取得随机字符串放入HttpSession 
return SUCCESS; 
} 
public void setInputStream(ByteArrayInputStream inputStream) { 
this.inputStream = inputStream; 
} 
public ByteArrayInputStream getInputStream() { 
return inputStream; 
}
}

 

4、LoginAction.java 验证验证码的action

private String rand; //表单中的rand
public String getRand() {
return rand;
}
public void setRand(String rand) {
this.rand = rand;
}
//从session中取出RandomAction.java 中生成的验证码random
String arandom=(String)(ActionContext.getContext().getSession().get("random"));

//下面就是将session中保存验证码字符串与客户输入的验证码字符串对比了
if(arandom.equals(this.getRand())) {
ActionContext.getContext().getSession().put("user", this.getUsername());
return SUCCESS;
}
else {
return ERROR;
}

 

5、配置struts.xml文件

<!-- Random验证码 -->
<action name="rand" class="com.mxl.rand.RandomAction">   
       <result type="stream">   
            <param name="contentType">image/jpeg</param>   
            <param name="inputName">inputStream</param>   
       </result>
   </action>

 

说明:

如果想修改验证码生成的个数,需要修改以下几个地方:

 

第一点:

int width=85, height=20;



第二点:

for (int i=0;i<6;i++)



数字6,修改成你想生成的位数就可以了~

 

 

 

下面是用servlet来生成验证码的例子

感谢原文作者的贡献,该文章对验证码生成的各种方法做了个总结,很不错了。这个是原文地址:

http://blog.csdn.net/wirror800/archive/2009/02/16/3897115.aspx

servlet文件

 

public class AuthImage extends HttpServlet   
{   
  
    private static final String CONTENT_TYPE = "text/html; charset=gb2312";   
    //设置字母的大小,大小   
    private Font mFont = new Font("Times New Roman", Font.PLAIN, 17);   
    public void init() throws ServletException   
    {   
        super.init();   
    }   
    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);   
    }   
  
    public void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException   
    {   
        response.setHeader("Pragma","No-cache");   
        response.setHeader("Cache-Control","no-cache");   
        response.setDateHeader("Expires", 0);   
        //表明生成的响应是图片   
        response.setContentType("image/jpeg");   
           
        int width=100, height=18;   
        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(1, 1, width-1, height-1);   
        g.setColor(new Color(102,102,102));   
        g.drawRect(0, 0, width-1, height-1);   
        g.setFont(mFont);   
  
        g.setColor(getRandColor(160,200));   
  
        //画随机线   
        for (int i=0;i<155;i++)   
        {   
            int x = random.nextInt(width - 1);   
            int y = random.nextInt(height - 1);   
            int xl = random.nextInt(6) + 1;   
            int yl = random.nextInt(12) + 1;   
            g.drawLine(x,y,x + xl,y + yl);   
        }   
  
        //从另一方向画随机线   
        for (int i = 0;i < 70;i++)   
        {   
            int x = random.nextInt(width - 1);   
            int y = random.nextInt(height - 1);   
            int xl = random.nextInt(12) + 1;   
            int yl = random.nextInt(6) + 1;   
            g.drawLine(x,y,x - xl,y - yl);   
        }   
  
        //生成随机数,并将随机数字转换为字母   
        String sRand="";   
        for (int i=0;i<6;i++)   
        {   
            int itmp = random.nextInt(26) + 65;   
            char ctmp = (char)itmp;   
            sRand += String.valueOf(ctmp);   
            g.setColor(new Color(20+random.nextInt(110),20+random.nextInt(110),20+random.nextInt(110)));   
            g.drawString(String.valueOf(ctmp),15*i+10,16);   
        }   
  
        HttpSession session = request.getSession(true);   
        session.setAttribute("rand",sRand);   
        g.dispose();   
        ImageIO.write(image, "JPEG", response.getOutputStream());   
    }   
    public void destroy()   
    {   
    }   
}  

 

web.xml

 

 

<servlet>   
    <servlet-name>image</servlet-name>   
    <servlet-class>org.test.web.AuthImage</servlet-class>   
</servlet>   
  
<servlet-mapping>   
    <servlet-name>image</servlet-name>   
 <url-pattern>/authImage</url-pattern>   
</servlet-mapping>  

 

html页面和上面struts2 例子的页面是类似的,只是把图像生成的路径改成servlet就行了。。

例如:<img src="authImage"/>

 

总结下:

  总的来说验证码的生成也并不是很困难,方法都是类似的。如果项目集成了struts2的话,用servlet类的方法来生成验证码的话,会有点麻烦,因为struts2的拦截器会把servlet类给拦截了,解决办法我还没想到。。 

如果要生成中文的验证码的话,可参考下面的文章:http://gx80.iteye.com/blog/148851

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值