opencv 手写选择题阅卷 (一)表格设计与识别
(一)答题表格设计与识别
实际设计好的表格如下图

为了图像精确,表格和四角的标记都是由程序生成的,文字和数据是后期排版软件添加上去的.
图中四角的四个黑方块主要用来定位表格,然后就可以切割出每个单元格,最后去做字符识别.
具体步骤为:
1,灰度化并二值化;
2,查找轮廓,把找出四个定位标记;
3,透视变换,校正变形;
4,切割表格,分别识别每个表格;
实际操作中发现最关键的是表格一定要平整,变形对识别影响较大;
部分代码:
int table_recognition(IplImage* img,unsigned char * result)
{
//大图二值化
IplImage* bin_img = cvCloneImage(img);
image_threshold(bin_img);
//去噪
IplImage* tmp_img = cvCloneImage(bin_img);
cvErode(tmp_img, tmp_img, NULL, ); //腐蚀
cvDilate(tmp_img, tmp_img, NULL, ); //膨胀
//查找轮廓
CvSeq* contours;
CvMemStorage * storage = cvCreateMemStorage();
cvSetImageROI(tmp_img, cvRect(, , bin_img->width, bin_img->height));
cvFindContours(tmp_img, storage, &contours, , ));
];
];
];
CvPoint points[][];
;
// 检测每个轮廓
for (; contours; contours = contours->h_next)
{
//用指定精度逼近多边形曲线
CvSeq* result;
result = cvApproxPoly(contours, );
//不是四边形的不要
)
continue;
//不是凸多边形不要
if (!cvCheckContourConvexity(result))
continue;
//面积大小或小于指定值的排除
));
)
continue;
//解码每个轮廓标志,正确的保存下来**********************************************
CvPoint2D32f srcQuad[];
; i < ; i++){
CvPoint* pt = (CvPoint*)cvGetSeqElem(result, i);//取标记四边形的四个顶点
points[n][i] = *pt;
srcQuad[i].x = (float)pt->x;
srcQuad[i].y = (float)pt->y;
}
//透视变换取出marker
IplImage * mark_img = cvCreateImage(cvSize(,), , );
perspective(bin_img, mark_img, srcQuad);
//
int rt = marker_decode(mark_img, &ids[n], &rotates[n], &vals[n]);
)
continue;
//
n++;
)
break;
}
)//发现四个标记
;
//if (rotates[0] != rotates[1] || rotates[1] != rotates[2] || rotates[2] != rotates[3])//四个标记旋转一致
// return -1;
//marker 0123
] != && ids[] != && ids[] != && ids[] != )
;
] != && ids[] != && ids[] != && ids[] != )
;
] != && ids[] != && ids[] != && ids[] != )
;
] != && ids[] != && ids[] != && ids[] != )
;
//确定表格四个点
CvPoint2D32f pts[];
; i < ; i++)
{
int id = ids[i];
int rotate = rotates[i];
CvPoint pt;
){
pt = points[i][( + rotate)%];
}
){
pt = points[i][( + rotate) % ];
}
){
pt = points[i][( + rotate) % ];
}
){
pt = points[i][( + rotate) % ];
}
pts[id].x = pt.x;
pts[id].y = pt.y;
}
//CvPoint2D32f tmp_ptf = pts[1];
//pts[1] = pts[3];
//pts[3] = tmp_ptf;
IplImage * table_img = cvCreateImage(cvSize(*, *+), , );
perspective(img, table_img, pts);
//表格分割
;
IplImage* gird_img = cvCreateImage(cvSize(, ), , );
; j < ; j+=)
{
; i < ; i++)
{
cvSetImageROI(table_img, cvRect(+*i, +*j, , ));
cvCopy(table_img, gird_img);
#ifdef _WIN32
save_gird(gird_img, nt);
#endif
int rt = svm_recognition(gird_img);
result[nt] = rt;
nt++;
}
}
//cvNamedWindow("Image", CV_WINDOW_NORMAL);
//cvShowImage("Image", gird_img);
//cvWaitKey(0);
cvReleaseImage(&bin_img);
cvClearMemStorage(storage);
;
}
opencv 手写选择题阅卷 (一)表格设计与识别的更多相关文章
- opencv 手写选择题阅卷 (二)字符识别
opencv 手写选择题阅卷 (二)字符识别 选择题基本上只需要识别ABCD和空五个内容,理论上应该识别率比较高的,识别代码参考了网上搜索的代码,因为参考的网址比较多,现在也弄不清是参考何处的代码了, ...
- opencv 手写选择题阅卷 (四)Android端 手机应用开发
opencv 手写选择题阅卷 (四)Android 手机应用开发 在PC端把代码调通以后开始开发Android 手机应用,因为主要功能代码为C++代码,所以需要通过NDK编译,JAVA通过JNI方式调 ...
- opencv 手写选择题阅卷 (三)训练分类器
opencv 手写选择题阅卷 (三)训练分类器 1,分类器选择:SVM 本来一开始用的KNN分类器,但这个分类器目前没有实现保存训练数据的功能,所以选择了SVN分类器; 2,样本图像的预处理和特征提取 ...
- OpenCV手写数字字符识别(基于k近邻算法)
摘要 本程序主要参照论文,<基于OpenCV的脱机手写字符识别技术>实现了,对于手写阿拉伯数字的识别工作.识别工作分为三大步骤:预处理,特征提取,分类识别.预处理过程主要找到图像的ROI部 ...
- TF:TF分类问题之MNIST手写50000数据集实现87.4%准确率识别:SGD法+softmax法+cross_entropy法—Jason niu
import tensorflow as tf from tensorflow.examples.tutorials.mnist import input_data # number 1 to 10 ...
- AI应用开发实战 - 手写识别应用入门
AI应用开发实战 - 手写识别应用入门 手写体识别的应用已经非常流行了,如输入法,图片中的文字识别等.但对于大多数开发人员来说,如何实现这样的一个应用,还是会感觉无从下手.本文从简单的MNIST训练出 ...
- 背水一战 Windows 10 (62) - 控件(媒体类): InkCanvas 保存和加载, 手写识别
[源码下载] 背水一战 Windows 10 (62) - 控件(媒体类): InkCanvas 保存和加载, 手写识别 作者:webabcd 介绍背水一战 Windows 10 之 控件(媒体类) ...
- 5 TensorFlow入门笔记之RNN实现手写数字识别
------------------------------------ 写在开头:此文参照莫烦python教程(墙裂推荐!!!) ---------------------------------- ...
- 用tensorflow搭建RNN(LSTM)进行MNIST 手写数字辨识
用tensorflow搭建RNN(LSTM)进行MNIST 手写数字辨识 循环神经网络RNN相比传统的神经网络在处理序列化数据时更有优势,因为RNN能够将加入上(下)文信息进行考虑.一个简单的RNN如 ...
随机推荐
- C# Func<>托付
曾经我们为了可以调用一个方法.必须比照这种方法定义一个对应的delegate. 原先我们定义delegate // 托付声明 -- 定义一个签名: delegate double MathAction ...
- WIN8重见开始菜单
从win7进入win8感觉不适应,做为一个程序开发人员,更觉得不爽,因此想着如何恢复开始菜单,查了很多文章终于找到方法 首先,在桌面版Windows 8启动资源管理器,单击工具栏上的“查看”选项卡后, ...
- 最简单的视频编码器:编译(libx264,libx265,libvpx)
===================================================== 最简单的视频编码器系列文章列表: 最简单的视频编码器:编译 最简单的视频编码器:基于libx ...
- 和Timesten有个约会--Timesten技术专栏系列(一)
作者: 三十而立 时间:2009年10月03日 12:08:42 本文出自 “inthirties(三十而立)”博客,转载请务必注明作者和保留出处http://blog.csdn.net/inthir ...
- 假设用一个名为text的字符串向量存放文本文件的数据,其中的元素或者是一句话或者是一个用于表示段分隔的空字符串。将text中第一段全改为大写形式
#include<iostream> #include<string> #include<vector> using namespace std; int main ...
- 网口扫盲三:以太网芯片MAC和PHY的关系
转载:http://www.cnblogs.com/jason-lu/articles/3195473.html 问:如何实现单片以太网微控制器? 答:诀窍是将微控制器.以太网媒体接入控制器(MA ...
- PHP.3-DIV+CSS标准网页布局准备工作(上)
DIV+CSS标准网页布局准备工作(上) 概述 使用"DIV+CSS"对网站进行布局符合W3C标准,采用这种方式布局通常是为了说明与HTML表格定位方式的区别.因为现在的网站设计标 ...
- X-Sequence
Description Let {xi} be the infinite sequence of integers: 1) x0 = A; 2) xi = (alpha * xi-1^2 + beta ...
- PHP读书笔记(7)- 函数
自定义函数 PHP内置了超过1000个函数,因此函数使得PHP成为一门非常强大的语言.大多数时候我们使用系统的内置函数就可以满足需求,但是自定义函数通过将一组代码封装起来,使代码进行复用,程序结构与逻 ...
- iOS XMPP(2)自己创建客户端
一.目的以及效果: 用Xcode利用xmpp框架建立客户端,实现向服务器注册添加用户 密码,以及登陆,离线状态 工程的主要结构:新建singleview工程,用xib拖放两个输入框和两个按钮, 并在v ...