验证码的实现

     验证码的实现方式主要有两种,一种是手动模拟的形式,其基本的思想是将进行画图操作,并将其输出为图片格式,如jpeg等。而另外一种则是借助Validate工具包(jar包),由它来实现验证码的显示,只需要设置对应的参数(推荐)。

    1、手动模拟

package com.ms.servlet;

import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;

import java.awt.image.BufferedImage;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Random;

import javax.imageio.ImageIO;
import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

public class CheckcodeServlet2 extends HttpServlet {
	private int width;
	private int height;
	private int number; //显示多少个字符串
	private String codes;	
	
	
	//初始化
	@Override
	public void init(ServletConfig config) throws ServletException {
		width = Integer.parseInt(config.getInitParameter("width"));
		height = Integer.parseInt(config.getInitParameter("height"));
		number =Integer.parseInt( config.getInitParameter("number"));
		codes = config.getInitParameter("codes");
	}

	public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		
		
		//画图
		BufferedImage image = new BufferedImage(width,height,BufferedImage.TYPE_INT_RGB);
		Graphics g = image.createGraphics();
		
		//创建白色背景
		g.setColor(Color.white);
		g.fillRect(0, 0, width, height);
		
		//画黑边圈
		g.setColor(Color.black);
		g.drawRect(0, 0, width-1, height-1);
		
		Random random = new Random();
		
		//每个字符占据的宽度
		int x = (width - 1) / number;
		int y = height -4;
		
		//随机生成字符
		for(int i=0; i<number; i++){
			String code = String.valueOf( codes.charAt( random.nextInt(codes.length())) );
			int red = random.nextInt(255);
			int green = random.nextInt(255);
			int blue = random.nextInt(255);
			g.setColor(new Color(red,green,blue));
			
			Font font = new Font("Arial",Font.PLAIN,random(height/2,height));
			g.setFont(font);
			
			g.drawString(code, i * x + 1, y);
		}
		
		//随机生成一些点
		for(int i=0; i<50; i++){
			int red = random.nextInt(255);
			int green = random.nextInt(255);
			int blue = random.nextInt(255);
			g.setColor(new Color(red,green,blue));
			g.drawOval(random.nextInt(width), random.nextInt(height), 1, 1);
		}

		
		OutputStream out = response.getOutputStream();
		//将图片转换为JPEG类型
		response.setContentType("image/jpeg");
		ImageIO.write(image, "image/jpeg", out);
		out.flush();

	}
	
	/**
	 * 产生一个从min到max之间的随机数
	 * @param min
	 * @param max
	 * @return
	 */
	private int random(int min,int max){
		int m = new Random().nextInt(999999) % (max - min);
		return m + min;
	}



	public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		doGet(request,response);
	}
}

    2、Validate的使用

package com.ms.servlet;

import java.io.IOException;
import java.io.PrintWriter;

import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import cn.dsna.util.images.ValidateCode;

public class test01 extends HttpServlet {
	
	private int width;
	private int height;
	private int number; //显示多少个字符串
	private String codes;	
	
	
	//初始化
	@Override
	public void init(ServletConfig config) throws ServletException {
		width = Integer.parseInt(config.getInitParameter("width"));
		height = Integer.parseInt(config.getInitParameter("height"));
		number =Integer.parseInt( config.getInitParameter("number"));
	}
	
	public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
                /*第一个参数为宽,第二个参数为高,第三个参数为验证码内容的个数,第四个参数为线条的数量*/
		ValidateCode vc = new ValidateCode(width,height,number,5);
		request.getSession().setAttribute("scode", vc.getCode());
		vc.write(response.getOutputStream());
	}

	public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		doGet(request,response);
	}

}

    3、两者之间web.xml的配置(这里提供基于jar包实现的配置,关于手动方式的配置是一样的)

<servlet>
    <servlet-name>test</servlet-name>
    <servlet-class>com.ms.servlet.test01</servlet-class>
      	<init-param>
  		<param-name>width</param-name>
  		<param-value>50</param-value>
  	</init-param>
  	 	<init-param>
  		<param-name>height</param-name>
  		<param-value>20</param-value>
  	</init-param>
  	 	<init-param>
  		<param-name>number</param-name>
  		<param-value>4</param-value>
  	</init-param>
  </servlet>


  <servlet-mapping>
    <servlet-name>test</servlet-name>
    <url-pattern>/backend/images/checkcode.jpg</url-pattern>
  </servlet-mapping>
  

    4、jsp端的代码

<!-- safecode.setAttribute('src','images/checkcode.jpg?'+Math.random()); -->
<script type="text/javascript">
        function changecode(){
		var safecode = document.getElementById("safecode");
		safecode.src="images/checkcode.jpg?"+Math.random(); //在线进行验证码的点击更新
	}
</script>

 <img src="images/checkcode.jpg" id="safecode" οnclick="changecode()"/>

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 动态验证码是一种常用的验证码形式,可以有效地防止机器恶意攻击。在 Java 中,我们可以通过以下步骤实现动态验证码: 1. 生成随机字符或数字,可以使用 Random 类或 Math.random() 方法。 2. 将生成的字符或数字绘制到图片上,可以使用 Java 的 Graphics2D 类来实现。 3. 将生成的验证码图片输出到客户端,可以使用 Java 的 ImageIO 类将图片输出为 PNG、JPEG 等格式。 下面是一个简单的动态验证码生成示例代码: ```java import java.awt.Color; import java.awt.Font; import java.awt.Graphics2D; import java.awt.image.BufferedImage; import java.io.IOException; import java.util.Random; import javax.imageio.ImageIO; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class CaptchaServlet extends HttpServlet { private static final long serialVersionUID = 1L; private static final String CHARACTERS = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"; private static final int WIDTH = 120; private static final int HEIGHT = 40; private static final int FONT_SIZE = 20; private static final int CHAR_COUNT = 4; protected void doGet(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/png"); // 创建验证码图片 BufferedImage captchaImage = new BufferedImage(WIDTH, HEIGHT, BufferedImage.TYPE_INT_RGB); Graphics2D g = captchaImage.createGraphics(); // 随机生成验证码 String captchaText = ""; Random random = new Random(); for (int i = 0; i < CHAR_COUNT; i++) { char c = CHARACTERS.charAt(random.nextInt(CHARACTERS.length())); captchaText += c; } // 设置背景色 g.setColor(Color.WHITE); g.fillRect(0, 0, WIDTH, HEIGHT); // 绘制字符 g.setColor(Color.BLACK); g.setFont(new Font("Arial", Font.BOLD, FONT_SIZE)); for (int i = 0; i < captchaText.length(); i++) { char c = captchaText.charAt(i); int x = (i * WIDTH / CHAR_COUNT) + (WIDTH / CHAR_COUNT - FONT_SIZE) / 2; int y = HEIGHT / 2 + FONT_SIZE / 2; g.drawString(String.valueOf(c), x, y); } // 绘制干扰线 g.setColor(Color.GRAY); for (int i = 0; i < 10; i++) { int x1 = random.nextInt(WIDTH); int y1 = random.nextInt(HEIGHT); int x2 = random.nextInt(WIDTH); int y2 = random.nextInt(HEIGHT); g.drawLine(x1, y1, x2, y2); } // 输出图片到客户端 ImageIO.write(captchaImage, "png", response.getOutputStream()); // 保存验证码到 Session request.getSession().setAttribute("captcha", captchaText); } } ``` 在上面的代码中,我们使用了 Servlet 技术,生成了一个 4 个字符的验证码图片,并将验证码保存到 Session 中,以便后续校验。你可以将上述代码复制到一个 Java 文件中,然后将其部署到一个 Web 服务器上,在浏览器中访问该 Servlet 的 URL,就可以看到生成的动态验证码了。 ### 回答2: 动态验证码是一种用于验证用户身份的安全措施,通常用于登录、注册等操作。在 Java 中,我们可以通过以下步骤实现动态验证码: 1. 生成验证码:利用随机数生成器生成一个包含数字和字母的随机字符串作为验证码,可以使用 Random 类或者 SecureRandom 类来生成随机数。 2. 将生成的验证码存储到 Session 或者数据库中,关联到用户信息。 3. 发送验证码给用户:通过短信、邮件或者其他途径将生成的验证码发送给用户。 4. 用户输入验证码:在用户登录或者注册界面,让用户输入收到的验证码。 5. 验证用户输入:将用户输入的验证码与 Session 或者数据库中存储的验证码进行比较,判断是否一致。 6. 判断验证结果:如果验证码一致,说明用户输入正确,可以继续进行后续操作;如果验证码不一致,说明用户输入错误。 总结:以上是基本的动态验证码实现过程,可以根据具体的需求和场景进行一些额外的优化和安全措施,比如设置验证码有效时间、加密存储等。在 Java 中,我们可以通过使用随机数生成器和与 Session 或者数据库交互,来实现动态验证码功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值