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如 ...
随机推荐
- 创建MySQL从库
我们知道Oracle有DataGuard实时备份数据.能够做主备切换,而MySQL也有自己的一套备库方案.称之为主从复制. 搭建MySQL从库是为了实时同步主库数据,同一时候也能够分担主库的读压力.对 ...
- ios开发——实用技术篇&Pist转模型详细介绍
Pist转模型详细介绍 关于Plist转模型在iOS开发中是非常常见的,每开一一个项目或者实现一个功能都要用到它,所以今天就给大家讲讲Plist怎么转成模型数据, 前提:必须有一个Plist文件或者通 ...
- php生成CSV格式(转)
参考网址: php对csv文件的常用操作集合 http://blog.xhbin.com/archives/748 1,下载CSV格式文档 唯一需要特别注意的是编码. <? include_on ...
- Centos自动登录系统并自动打开VNC Server
系统自动登录 修改配置文件 sudo vim /etc/gdm/custom.conf 增加配置 [daemon] AutomaticLogin=spark AutomaticLoginEnable= ...
- IIS7程序发布后 之 报图表处理程序配置 [c:\TempImageFiles\] 中的临时目录无效
把.net4.0的ASP.NET网站布置在IIS7上,原本开发时一切ok,图形都能够出来,但是一旦部署到iis上,再访问的话, 错误问题:图表处理程序配置 [c:\TempImageFiles\] 中 ...
- 关于 未能加载文件或程序集“MySql.Web, Version=6.7.4.0, Culture=neutral, PublicKeyToken=c5687fc88969c44d”或它的某一个依赖项。系统找不到指定的文件。
我这个项目是MVC4的,有两个版本,第一个版本直接运行没什么问题,但是跑第二个版本的时候就给我提示这个错误.好吧,百度果然是万能的.下边是解决方案. 1.找到 C:\Windows\Microsoft ...
- Maven学习小结(六 setting.xml详解[转])
当Maven运行过程中的各种配置,例如pom.xml,不想绑定到一个固定的project或者要分配给用户时,我们使用settings.xml中的settings元素来确定这些配置.这包含了本地仓库位置 ...
- BeanDefinitionRegistry extends AliasRegistry
// 用该Registry注册一个新定义的bean,但是新的bean必须支持父的定义和子的定义void registerBeanDefinition(String beanName, BeanDefi ...
- Android(java)学习笔记106-2:反射机制
1.反射机制: JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为 ...
- Android(java)学习笔记95:Android原理揭秘系列之View、ViewGroup
作过Android 应用开发的朋友都知道,Android的UI界面都是由View和ViewGroup及其派生类组合而成的.其中,View是所有UI组件的基类,而ViewGroup是容纳这些组件的容器, ...