验证码2(较详细含有文字,干扰线)

package cn.itcast.image;


import java.awt.BasicStroke;
import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Random;


import javax.imageio.ImageIO;


public class VerifyCode {
private int w = 70;
private int h = 35;
  private Random r = new Random();
  // {"宋体", "华文楷体", "黑体", "华文新魏", "华文隶书", "微软雅黑", "楷体_GB2312"}
private String[] fontNames  = {"宋体", "华文楷体", "黑体", "微软雅黑", "楷体_GB2312"};
// 可选字符
private String codes  = "23456789abcdefghjkmnopqrstuvwxyzABCDEFGHJKMNPQRSTUVWXYZ";
// 背景色
private Color bgColor  = new Color(255, 255, 255);
// 验证码上的文本
private String text ;

// 生成随机的颜色
private Color randomColor () {
int red = r.nextInt(150);
int green = r.nextInt(150);
int blue = r.nextInt(150);
return new Color(red, green, blue);
}

// 生成随机的字体
private Font randomFont () {
int index = r.nextInt(fontNames.length);
String fontName = fontNames[index];//生成随机的字体名称
int style = r.nextInt(4);//生成随机的样式, 0(无样式), 1(粗体), 2(斜体), 3(粗体+斜体)
int size = r.nextInt(5) + 24; //生成随机字号, 24 ~ 28
return new Font(fontName, style, size);
}

// 画干扰线
private void drawLine (BufferedImage image) {
int num  = 3;//一共画3条
Graphics2D g2 = (Graphics2D)image.getGraphics();
for(int i = 0; i < num; i++) {//生成两个点的坐标,即4个值
int x1 = r.nextInt(w);
int y1 = r.nextInt(h);
int x2 = r.nextInt(w);
int y2 = r.nextInt(h); 
g2.setStroke(new BasicStroke(1.5F)); 
g2.setColor(Color.BLUE); //干扰线是蓝色
g2.drawLine(x1, y1, x2, y2);//画线
}
}

// 随机生成一个字符
private char randomChar () {
int index = r.nextInt(codes.length());
return codes.charAt(index);
}

// 创建BufferedImage
private BufferedImage createImage () {
BufferedImage image = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB); 
Graphics2D g2 = (Graphics2D)image.getGraphics(); 
g2.setColor(this.bgColor);
g2.fillRect(0, 0, w, h);
  return image;
}

// 调用这个方法得到验证码
public BufferedImage getImage () {
BufferedImage image = createImage();//创建图片缓冲区 
Graphics2D g2 = (Graphics2D)image.getGraphics();//得到绘制环境
StringBuilder sb = new StringBuilder();//用来装载生成的验证码文本
// 向图片中画4个字符
for(int i = 0; i < 4; i++)  {//循环四次,每次生成一个字符
String s = randomChar() + "";//随机生成一个字母 
sb.append(s); //把字母添加到sb中
float x = i * 1.0F * w / 4; //设置当前字符的x轴坐标
g2.setFont(randomFont()); //设置随机字体
g2.setColor(randomColor()); //设置随机颜色
g2.drawString(s, x, h-5); //画图
}
this.text = sb.toString(); //把生成的字符串赋给了this.text
drawLine(image); //添加干扰线
return image;
}

// 返回验证码图片上的文本
public String getText () {
return text;
}

// 保存图片到指定的输出流
public static void output (BufferedImage image, OutputStream out) 
throws IOException {
ImageIO.write(image, "JPEG", out);
}

}




测试代码

package image;


import java.awt.image.BufferedImage;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;




public class demo2 {
  public static void main(String[] args) throws FileNotFoundException, IOException{
 
VerifyCode verifyCode = new VerifyCode();
BufferedImage bi = verifyCode.getImage(); 
System.out.println(verifyCode.getText()); 

VerifyCode.output(bi, new FileOutputStream("F:/b.jpg"));
}
  }
 



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
验证码识别是一项比较复杂的任务,需要用到图像处理、机器学习等知识,但是我们可以通过一些简单的技巧来实现一个基本的验证码识别程序。以下是一个基于Python的验证码识别程序的实现过程: 1. 导入必要的库 我们需要用到PIL库来处理图像,以及sklearn库来进行机器学习。 ```python from PIL import Image import numpy as np from sklearn.neural_network import MLPClassifier ``` 2. 预处理图像 我们需要将图像转换为灰度图,并且将其二值化。这可以通过以下代码实现: ```python def preprocess_image(image_path): # Open image and convert to grayscale image = Image.open(image_path).convert('L') # Apply threshold to get black and white image threshold_value = 100 image = image.point(lambda x: 0 if x < threshold_value else 255) return image ``` 3. 分割字符 我们需要将验证码中的每个字符分割出来,以便进行单独的识别。这可以通过以下代码实现: ```python def segment_image(image): # Find contours contours = find_contours(image, 0) # Get bounding rectangles for each contour rectangles = [cv2.boundingRect(cnt) for cnt in contours] # Sort rectangles from left to right rectangles = sorted(rectangles, key=lambda x: x[0]) # Crop and save each character image characters = [] for rect in rectangles: x, y, w, h = rect character_image = image[y:y+h, x:x+w] characters.append(character_image) return characters ``` 4. 特征提取 我们需要将每个字符转换为一个向量,以便进行机器学习。这可以通过以下代码实现: ```python def extract_features(character): # Resize image to 10x10 pixels resized_character = cv2.resize(character, (10, 10)) # Flatten image to a 1D array flattened_character = resized_character.flatten() # Normalize pixel values to be between 0 and 1 normalized_character = flattened_character / 255.0 return normalized_character ``` 5. 训练模型 我们需要使用机器学习算法训练一个模型来识别验证码中的字符。这可以通过以下代码实现: ```python def train_model(X, y): # Create MLP classifier clf = MLPClassifier(hidden_layer_sizes=(100,), max_iter=500) # Train classifier on training data clf.fit(X, y) return clf ``` 6. 识别验证码 现在我们已经有了一个训练好的模型,我们可以将每个字符提取特征并将其输入到模型中进行预测。这可以通过以下代码实现: ```python def recognize_captcha(image_path, model): # Preprocess image image = preprocess_image(image_path) # Segment characters characters = segment_image(image) # Extract features for each character features = [extract_features(char) for char in characters] # Predict labels for each character labels = model.predict(features) # Convert labels to string captcha_text = ''.join(labels) return captcha_text ``` 完整的代码如下: ```python from PIL import Image import numpy as np from sklearn.neural_network import MLPClassifier import cv2 def preprocess_image(image_path): # Open image and convert to grayscale image = Image.open(image_path).convert('L') # Apply threshold to get black and white image threshold_value = 100 image = image.point(lambda x: 0 if x < threshold_value else 255) return image def segment_image(image): # Find contours contours = find_contours(image, 0) # Get bounding rectangles for each contour rectangles = [cv2.boundingRect(cnt) for cnt in contours] # Sort rectangles from left to right rectangles = sorted(rectangles, key=lambda x: x[0]) # Crop and save each character image characters = [] for rect in rectangles: x, y, w, h = rect character_image = image[y:y+h, x:x+w] characters.append(character_image) return characters def extract_features(character): # Resize image to 10x10 pixels resized_character = cv2.resize(character, (10, 10)) # Flatten image to a 1D array flattened_character = resized_character.flatten() # Normalize pixel values to be between 0 and 1 normalized_character = flattened_character / 255.0 return normalized_character def train_model(X, y): # Create MLP classifier clf = MLPClassifier(hidden_layer_sizes=(100,), max_iter=500) # Train classifier on training data clf.fit(X, y) return clf def recognize_captcha(image_path, model): # Preprocess image image = preprocess_image(image_path) # Segment characters characters = segment_image(image) # Extract features for each character features = [extract_features(char) for char in characters] # Predict labels for each character labels = model.predict(features) # Convert labels to string captcha_text = ''.join(labels) return captcha_text # Load training data X = np.load('X.npy') y = np.load('y.npy') # Train model model = train_model(X, y) # Recognize captcha captcha_text = recognize_captcha('captcha.png', model) print(captcha_text) ``` 在运行代码之前,你需要准备一些验证码图像和对应的标签,将它们转换为特征向量和标签向量,然后将其保存为`X.npy`和`y.npy`文件。这可以通过以下代码实现: ```python # Load captcha images and corresponding labels captcha_images = [] captcha_labels = [] for i in range(1, 501): image_path = 'captcha{}.png'.format(i) captcha_text = image_path.split('.')[0][-6:] captcha_image = preprocess_image(image_path) captcha_images.append(captcha_image) captcha_labels.append(captcha_text) # Convert captcha images to feature vectors X = np.array([extract_features(image) for image in captcha_images]) # Convert captcha labels to label vectors y = np.array(captcha_labels) # Save training data to file np.save('X.npy', X) np.save('y.npy', y) ``` 这个程序可能无法完全识别所有的验证码,但是它可以作为一个基本的验证码识别程序的起点。你可以通过改进特征提取、机器学习算法等来提高识别的准确率。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值