问题描述
CodeAction.javapackageAction;importjava.awt.Color;importjava.awt.Font;importjava.awt.Graphics;importjava.awt.image.BufferedImage;importjava.io.ByteArrayInputStream;importjava.io.ByteArrayOutputStream;importjava.util.Random;importjavax.imageio.ImageIO;importjavax.imageio.stream.ImageOutputStream;importcom.opensymphony.xwork2.ActionContext;importcom.opensymphony.xwork2.ActionSupport;publicclassCodeActionextendsActionSupport{/***http://465886163.iteye.com/blog/1189394*/privatestaticfinallongserialVersionUID=1L;privateStringcode1;privateByteArrayInputStreaminputStream;publicvoidsetInputStream(ByteArrayInputStreaminputStream){this.inputStream=inputStream;}publicByteArrayInputStreamgetInputStream(){returninputStream;}publicStringgetCode1(){returncode1;}publicvoidsetCode1(Stringcode1){this.code1=code1;}publicStringexecute()throwsException{//在内存中创建图象intwidth=85,height=20;BufferedImageimage=newBufferedImage(width,height,BufferedImage.TYPE_INT_RGB);//获取图形上下文Graphicsg=image.getGraphics();//生成随机类Randomrandom=newRandom();//设定背景色g.setColor(getRandColor(200,250));g.fillRect(0,0,width,height);//设定字体g.setFont(newFont("TimesNewRoman",Font.PLAIN,18));//随机产生155条干扰线,使图象中的认证码不易被其它程序探测到g.setColor(getRandColor(160,200));for(inti=0;i<155;i++){intx=random.nextInt(width);inty=random.nextInt(height);intxl=random.nextInt(12);intyl=random.nextInt(12);g.drawLine(x,y,x+xl,y+yl);}//取随机产生的认证码(6位数字)StringsRand="";for(inti=0;i<6;i++){Stringrand=String.valueOf(random.nextInt(10));sRand+=rand;//将认证码显示到图象中g.setColor(newColor(20+random.nextInt(110),20+random.nextInt(110),20+random.nextInt(110)));//调用函数出来的颜色相同,可能是因为种子太接近,所以只能直接生成g.drawString(rand,13*i+6,16);}code1=sRand;System.out.println(code1);//将认证码存入SESSIONActionContext.getContext().getApplication().put("rand",sRand);//图象生效g.dispose();ByteArrayOutputStreamoutput=newByteArrayOutputStream();ImageOutputStreamimageOut=ImageIO.createImageOutputStream(output);ImageIO.write(image,"JPEG",imageOut);imageOut.close();ByteArrayInputStreaminput=newByteArrayInputStream(output.toByteArray());this.setInputStream(input);returnSUCCESS;}/**给定范围获得随机颜色*/privateColorgetRandColor(intfc,intbc){Randomrandom=newRandom();if(fc>255)fc=255;if(bc>255)bc=255;intr=fc+random.nextInt(bc-fc);intg=fc+random.nextInt(bc-fc);intb=fc+random.nextInt(bc-fc);returnnewColor(r,g,b);}}jsp页面<body><s:textfieldname="code"label="请输入验证码:"></s:textfield><s:propertyvalue="#application.rand"/><imgsrc="rand.action"onclick="changeValidateCode(this)"/><s:debug></s:debug></body>配置文件<actionname="rand"class="Action.CodeAction"><resulttype="stream"><paramname="contentType">image/jpeg</param><paramname="inputName">inputStream</param></result></action>
解决方案
解决方案二:
没有试过这种做法,不过直接访问servlet获取验证码的例子:http://blog.csdn.net/w4bobo/article/details/8259560
解决方案三:
你现在是把生成的验证码字符串赋值给了code1,你再jsp页面上直接调用就可以了<s:propertyvalue="code1"/>这样就可以获得你的验证码字符串了
解决方案四:
楼主这个代码是有放入值栈的,楼主你多刷新几次这个jsp就会看到你<s:propertyvalue="#application.rand"/>每次显示的验证码都是上一次的验证码。这个是因为楼主的jsp页面的顺序而导致的<s:propertyvalue="#application.rand"/><imgsrc="rand.action"onclick="changeValidateCode(this)"/><s:debug></s:debug>你是先显示application.rand后请求rand.action所以你第一次打开这个jsp页面的时候你先取这个application.rand还没请求当然就没有赋值到值栈。你刷新一下就会发现显示的是你上一次的rand
解决方案五:
我把jsp代码改成下面的就能拿到值,但是还是上一次的验证啊,想问一下,怎么才能拿到当前的结果呢<imgsrc="rand.action"onclick="changeValidateCode(this)"/><s:propertyvalue="#application.rand"/>
解决方案六:
2楼的亲故,code1他没有进入值栈
解决方案七:
那你debug进你的Action类看下看页面一加载的时候进你的验证码那个CodeAction了吗。然后你看在你的CodeAction返回(return)以前那个code1的值是多少。action中的参数市放在ValueStack中的,只要你在Action中有get方法那么在页面上就是可以获取的。希望能够帮到你