《机器学习》KNN算法实现手写数字识别

目录

一、项目介绍

二、数据集介绍

三、需要解决的问题

四、代码实际展示

代码展示

实验结果

五、使用自己的数据进行测试

代码展示

结果展示

六、总结


一、项目介绍

通过对一张2000*1000像素写满0-9手写数字的图片进行处理。分割出训练集和测试集使用KNN算法进行训练并且预测测试集的标签。

二、数据集介绍

数据集是一张2000*1000像素的图片,有50行100列手写数字,其中每个手写数字占20*20的像素。将图片对半分割,左侧50列作为训练集,右侧50列作为测试集。当模型建立好后也可以自己传入手写数字图片进行预测。

三、需要解决的问题

1、如何分割训练集以及测试集并将每个手写数字分割出来?

答:可以借助numpy库对图片进行分割

2、如何把得到的训练集以及测试集转换成能传入模型的数值?

答:可以将组成图像的二维数组展开为一行,这样就变成能传入模型的列表数据

3、如何给数据添加标签?

答:可以自己创建与样本数量相同的0-9的数字放在列表中作为标签

4、如何得到模型的正确率?

答:可以将模型输出的结果与测试集标签进行对比,查看相同结果占比

5、如何使用其他图片来验证模型?

答:可以用与样本大小相同的灰度图传入数据进行预测

四、代码实际展示

代码展示

python 复制代码
import numpy as np
import cv2
from networkx.classes import neighbors
from sklearn.neighbors import KNeighborsClassifier

img_gray = cv2.imread(r'./data/num.png',0)

# 将每张照片分割出来
# 列表推导式
cells = [np.hsplit(row,100) for row in np.vsplit(img_gray,50)]
# 列表推导式展开
# cells = []
# for row in np.vsplit(img_gray,50):
#     cells.append(np.hsplit(row, 100))

# 将cells列表转换为数组
x = np.array(cells)
# 将前50列分割为训练集
train = x[:,:50]
# 将后50列分割为测试集
test = x[:,50:100]

# 将每张图像展开为一行400列
train_new = train.reshape(-1,400).astype(np.float32)
test_new = test.reshape(-1,400).astype(np.float32)

# 创建标签
k = np.arange(0,10)
labels = np.repeat(k,250)
# 创建训练集标签
train_labels = labels[:,np.newaxis]
# 创建测试标签
test_labels = labels[:,np.newaxis]

# 创建KNN模型(为cv2库自带)
knn = cv2.ml.KNearest_create()
# 训练模型
knn.train(train_new,cv2.ml.ROW_SAMPLE,train_labels)
# 用测试集测试模型
ret,result,neigh,dist = knn.findNearest(test_new,k=3)
# 计算模型的正确率
matches = result == test_labels
correct = np.count_nonzero(matches)
accuracy = correct*100.0/result.size
print('使用cv2准确率为:',accuracy)

# 使用sklearn库自带的KNN库进行试验
knn1 = KNeighborsClassifier(n_neighbors=3)
knn1.fit(train_new,train_labels)
result1 = knn1.predict(test_new)
result1 = result1[:,np.newaxis]
matches = result1 == test_labels
correct = np.count_nonzero(matches)
accuracy = correct*100.0/result1.size
print('使用sklearn准确率为:',accuracy)

实验结果

可见该模型的正确率还挺客观,但远远达不到预期结果。

五、使用自己的数据进行测试

以下是本人自己写的几张20*20像素的手写数字图片

代码展示

python 复制代码
import numpy as np
import cv2
from sklearn.neighbors import KNeighborsClassifier

img_gray = cv2.imread(r'./data/num.png',0)
# 读取自己的数据
img_2 = cv2.imread(r'./data/2.png',0).reshape(-1,400).astype(np.float32)
img_3 = cv2.imread(r'./data/3.png',0).reshape(-1,400).astype(np.float32)
img_5 = cv2.imread(r'./data/5.png',0).reshape(-1,400).astype(np.float32)
img_8 = cv2.imread(r'./data/8.png',0).reshape(-1,400).astype(np.float32)

cells = [np.hsplit(row,100) for row in np.vsplit(img_gray,50)]
x = np.array(cells)
train = x[:,:50]
train = train.reshape(-1,400).astype(np.float32)

# 将数据纵向合并
test = np.vstack((img_2,img_3,img_5,img_8))

# 训练集标签
k = np.arange(0,10)
labels = np.repeat(k,250)
train_labels = labels[:,np.newaxis]
# 设置验证集标签
test_labels = [[2], [3], [5], [8]]

# 创建KNN模型
Knn = KNeighborsClassifier(n_neighbors=3)
# 训练模型
Knn.fit(train,train_labels)
# 进行预测
result = Knn.predict(test)
result = result[:,np.newaxis]
# 计算正确率
matches = result == test_labels
correct = np.count_nonzero(matches)
accuracy = correct*100.0/result.size
print('验证自写2、3、5、8正确率为:',accuracy)

结果展示

可见模型将数字8预测成了3

六、总结

KNN算法是机器学习的入门算法,所以对初学者来说能够熟练运用对以后的学习有很大意义。虽然模型的正确率不高但对训练模型的学习过程很重要。

相关推荐
Zfox_2 分钟前
DeepSeek R1本地化部署 Ollama + Chatbox 打造最强 AI 工具
人工智能·ai·大模型教程·deepseek
XY_墨莲伊14 分钟前
【算法设计与分析】实验5:贪心算法—装载及背包问题
c语言·数据结构·c++·算法·贪心算法·排序算法
CodeLinghu14 分钟前
Agentic Automation:基于Agent的企业认知架构重构与数字化转型跃迁---我的AI经典战例
人工智能·重构·架构
银行数字化转型导师坚鹏16 分钟前
数字化转型导师坚鹏:AI大模型DEEPSEEK重构人工智能格局的里程碑
人工智能·ai·重构·deepseek
Happy_Traveller39 分钟前
三角形的最大周长(976)
数据结构·算法·leetcode
水蓝烟雨39 分钟前
[HOT 100] 2824. 统计和小于目标的下标对数目
算法·hot 100
KuaCpp1 小时前
搜索与图论复习2最短路
c++·算法·图论
X.AI6661 小时前
【大模型LLM面试合集】大语言模型架构_MHA_MQA_GQA
人工智能·语言模型·自然语言处理
智识世界Intelligence1 小时前
DeepSeek的崛起与全球科技市场的震荡
人工智能
弥树子1 小时前
使用 PyTorch 实现逻辑回归并评估模型性能
人工智能·pytorch·逻辑回归