如何用java实现验证码?
2020/09/01 11:08 分类: 技术交流 浏览:4
验证码的出现在一定程度上降低了程序被攻击的风险,也使得我们的软件安全性得到有效提高。Java发展至今,网上也出现了各种各样的验证码:有图像验证码,字符验证码,还有用于计算的验证码等等。接下来为大家介绍几个原始验证码的写法。效果如下:
|
|
|
第一款是基本的验证码,就背景颜色和字体颜色随机的。第二款是每个字符都是不同的颜色。第三款是每个字符都有不同的倾斜角度。
验证码的实现原理:前端页面发送请求【使用<img>标签】到后端,后端使用io流写出一张图片,前端页面展示出来。前端代码如下:
验证码:<input name="code" style="width: 145px;vertical-align:middle"/> <img alt="" src="code" style="vertical-align:middle" onclick="this.src='code?r='+Math.random()"> |
接下来就是后台代码实现了,后台画一张图片实际上就跟美术生写生一样,先有一个画板,然后在画板上铺上一张宣纸,在使用画笔在宣纸上进行图像的绘制。
第一部分:获取画板对象,并设置绘画区域:
//=================================创建画板(缓存图片)=================================// //1.创建画板(缓存图片)(参数:宽度,高度,图像类型 -- 表示一个图像,该图像具有整数像素的 RGB 颜色) BufferedImage image = new BufferedImage(100, 30, BufferedImage.TYPE_INT_BGR);
//2.根据缓冲图片获取一只画笔,默认颜色为白色 Graphics g = image.getGraphics(); //3.创建随机数对象,用于获取随机颜色 Random r = new Random();
//==================================画背景(随机颜色)==================================// //4.先给笔设置颜色,才能设置画的区域,范围在0~255 //g.setColor(Color.gray);//设置固定颜色 g.setColor(new Color(r.nextInt(256), r.nextInt(256), r.nextInt(256)));
//5.在画板上设置画的区域(参数:0,0表示起点的x和y坐标,100表示宽度,30表示高度) g.fillRect(0, 0, 100, 30); |
第二部分:画字符串(随机字符串,随机颜色,随机字体):
//6.获取随机字符串 String string = getString(4); //7.为笔设置随机颜色,少了这一步的话,就会与背景颜色一样 g.setColor(new Color(r.nextInt(256), r.nextInt(256), r.nextInt(256))); //8.为该字符串设置随机的字体 g.setFont(getFont()); //9.将字符串画出(参数:画的字符串,10,25指的是画字符串的x坐标和字符串基线坐标y) g.drawString(string, 10, 25); |
第三部分:画干扰线和干扰点
//==========================画干扰线(随机位置)===========================// for(int i = 0 ; i < 4 ; i++){ g.setColor(new Color(r.nextInt(256), r.nextInt(256), r.nextInt(256))); //参数:第一个点的坐标(x,y) 第二个点的坐标(x,y) //将笔强转成Graphics2D类型 Graphics2D g1 = (Graphics2D)g; //然后设置笔为原始宽度的1.5倍,Stroke:画笔 g1.setStroke(new BasicStroke(1.5f)); g.drawLine(r.nextInt(100), r.nextInt(30), r.nextInt(100), r.nextInt(30)); }
//==========================画干扰点(随机位置)===========================// for(int i = 0 ; i < 20 ; i++){ g.setColor(new Color(r.nextInt(256), r.nextInt(256), r.nextInt(256))); //参数:圆心坐标(x,y) 横轴长度 纵轴长度,Oval椭圆 g.drawOval(r.nextInt(100), r.nextInt(30), 2, 2); } |
第四部分:画出图片,释放资源
//============================将缓存图片写出去=============================// //x.设置响应的类型 resp.setContentType("image/jpeg"); //y.创建一个字节流(图片是二进制文件,只能通过字节流写出) ServletOutputStream os = resp.getOutputStream(); //z.写出图片 ImageIO.write(image, "jpeg", os);
//关闭流释放资源 os.close(); |
其他:自定义的方法:
//获取随机字符串 public String getString(int num){ //这里不写0,O,1,l,2,z,6,b,U,V,v,u,9,q是因为用于不好区分,提高用户体验度 String words = "acdefghjkmnprstwxy34578ACEFGHJKLMNPQRSTWXY"; String result = ""; Random r = new Random(); for(int i = 0;i < num;i++){ result += words.charAt(r.nextInt(words.length())) + " "; } return result; } //随机字体 public Font getFont(){ Font[] fonts = new Font[5]; Random r = new Random(); fonts[0] = new Font("微软雅黑", Font.ITALIC, 24); fonts[1] = new Font("新宋体", Font.PLAIN, 24); fonts[2] = new Font("Microsoft YaHei UI", Font.PLAIN, 24); fonts[3] = new Font("仿宋", Font.PLAIN, 24); fonts[4] = new Font("Cambria", Font.BOLD, 24); return fonts[r.nextInt(fonts.length)]; } //随机颜色 public Color getRandomColor() { Random ran = new Random(); Color color = new Color(ran.nextInt(256), ran.nextInt(256), ran.nextInt(256)); return color; } |
实现不同字符的不同颜色:
StringBuilder string = new StringBuilder(); int left = 10; for(int i = 0 ; i < 4 ; i++){ g.setColor(getRandomColor()); g.setFont(getFont()); String words = getString(1); string.append(words); g.drawString(words, left, 25); left += 20; } |
实现不同字符的不同旋转角度:
StringBuilder string = new StringBuilder();//接收验证码的字符串 int left = 10;//设置画字符串的原始x坐标 int x = 15;//设置字符串旋转的x坐标 Random ran = new Random(); for(int i = 0 ; i < 4 ; i++){ g.setColor(getRandomColor());//随机颜色 g.setFont(getFont());//随机字体 String words = getString(1);//获取随机字符 string.append(words);//拼接到字符串string Graphics2D g2 = (Graphics2D)g;//转换成Graphics2D对象,它才有rotate方法 double th = ran.nextInt(100)/100.0;//随机旋转角度 g2.rotate(th, x, 25);//旋转一个字符 g2.drawString(words, left, 25);//画字符串 g2.rotate(-th, x, 25);//调回原始角度 left += 20;//继续画下一个字符 x += 20;//继续旋转下一个字符 } |
赞 0