1.安装:
配置linux或windows环境
[Linux下配置OpenCV](http://note.youdao.com/noteshare?id=5de54af1ef6fef8352b8f3d3a9356845&sub=3D706CF274274B68B3BA2C9C42254747)
[Windows下配置OpenCV](http://note.youdao.com/noteshare?id=e0df335c7bba4d7633874375539c228a&sub=1015DEB3B7C847C28D0BBB87F21EDB59)
核心包:将下载的exe文件运行,取出其中的opencv.jar包
2.核心工具类:
package com.acts.opencv.common.utils;
import java.awt.image.BufferedImage;
import java.awt.image.DataBufferByte;
import java.io.IOException;
import java.util.Date;
import java.util.Vector;
import org.opencv.core.Mat;
import org.opencv.core.MatOfByte;
import org.opencv.core.MatOfPoint;
import org.opencv.core.Point;
import org.opencv.core.Size;
import org.opencv.highgui.Highgui;
import org.opencv.imgproc.Imgproc;
public class OpenCVUtil {
public static BufferedImage covertMat2Buffer(Mat mat) throws IOException {
long time1 = new Date().getTime();
// Mat 转byte数组
BufferedImage originalB = toBufferedImage(mat);
long time3 = new Date().getTime();
System.out.println("保存读取方法2转=" + (time3 - time1));
return originalB;
// ImageIO.write(originalB, "jpg", new File("D:\\test\\testImge\\ws2.jpg"));
}
public static byte[] covertMat2Byte(Mat mat) throws IOException {
long time1 = new Date().getTime();
// Mat 转byte数组
byte[] return_buff = new byte[(int) (mat.total() * mat.channels())];
Mat mat1 = new Mat();
mat1.get(0, 0, return_buff);
long time3 = new Date().getTime();
System.out.println(mat.total() * mat.channels());
System.out.println("保存读取方法2转=" + (time3 - time1));
return return_buff;
}
public static byte[] covertMat2Byte1(Mat mat) throws IOException {
long time1 = new Date().getTime();
MatOfByte mob = new MatOfByte();
Highgui.imencode(".jpg", mat, mob);
long time3 = new Date().getTime();
// System.out.println(mat.total() * mat.channels());
System.out.println("Mat转byte[] 耗时=" + (time3 - time1));
return mob.toArray();
}
public static BufferedImage toBufferedImage(Mat m) {
int type = BufferedImage.TYPE_BYTE_GRAY;
if (m.channels() > 1) {
type = BufferedImage.TYPE_3BYTE_BGR;
}
int bufferSize = m.channels() * m.cols() * m.rows();
byte[] b = new byte[bufferSize];
m.get(0, 0, b); // get all the pixels
BufferedImage image = new BufferedImage(m.cols(), m.rows(), type);
final byte[] targetPixels = ((DataBufferByte) image.getRaster().getDataBuffer()).getData();
System.arraycopy(b, 0, targetPixels, 0, b.length);
return image;
}
/**
* 腐蚀膨胀是针对于白色区域来说的,腐蚀即腐蚀白色区域
* 腐蚀算法(黑色区域变大)
* @param source
* @return
*/
public static Mat eroding(Mat source) {
return eroding(source, 1);
}
public static Mat eroding(Mat source, double erosion_size) {
Mat resultMat = new Mat(source.rows(), source.cols(), source.type());
Mat element = Imgproc.getStructuringElement(Imgproc.MORPH_RECT, new Size(2 * erosion_size + 1,
2 * erosion_size + 1));
Imgproc.erode(source, resultMat, element);
return resultMat;
}
/**
* 腐蚀膨胀是针对于白色区域来说的,膨胀是膨胀白色区域
* 膨胀算法(白色区域变大)
* @param source
* @return
*/
public static Mat dilation(Mat source) {
return dilation(source, 1);
}
/**
* 腐蚀膨胀是针对于白色区域来说的,膨胀是膨胀白色区域
* @Author 王嵩
* @param source
* @param dilationSize 膨胀因子2*x+1 里的x
* @return Mat
* @Date 2018年2月5日
* 更新日志
* 2018年2月5日 王嵩 首次创建
*
*/
public static Mat dilation(Mat source, double dilation_size) {
Mat resultMat = new Mat(source.rows(), source.cols(), source.type());
Mat element = Imgproc.getStructuringElement(Imgproc.MORPH_RECT, new Size(2 * dilation_size + 1,
2 * dilation_size + 1));
Imgproc.dilate(source, resultMat, element);
return resultMat;
}
/**
* 轮廓识别,使用最外轮廓发抽取轮廓RETR_EXTERNAL,轮廓识别方法为CHAIN_APPROX_SIMPLE
* @param source 传入进来的图片Mat对象
* @return 返回轮廓结果集
*/
public static Vector<MatOfPoint> findContours(Mat source) {
Mat rs = new Mat();
/**
* 定义轮廓抽取模式
*RETR_EXTERNAL:只检索最外面的轮廓;
*RETR_LIST:检索所有的轮廓,并将其放入list中;
*RETR_CCOMP:检索所有的轮廓,并将他们组织为两层:顶层是各部分的外部边界,第二层是空洞的边界;
*RETR_TREE:检索所有的轮廓,并重构嵌套轮廓的整个层次。
*/
int mode = Imgproc.RETR_EXTERNAL;
// int mode = Imgproc.RETR_TREE;
/**
* 定义轮廓识别方法
* 边缘近似方法(除了RETR_RUNS使用内置的近似,其他模式均使用此设定的近似算法)。可取值如下:
*CV_CHAIN_CODE:以Freeman链码的方式输出轮廓,所有其他方法输出多边形(顶点的序列)。
*CHAIN_APPROX_NONE:将所有的连码点,转换成点。
*CHAIN_APPROX_SIMPLE:压缩水平的、垂直的和斜的部分,也就是,函数只保留他们的终点部分。
*CHAIN_APPROX_TC89_L1,CV_CHAIN_APPROX_TC89_KCOS:使用the flavors of Teh-Chin chain近似算法的一种。
*LINK_RUNS:通过连接水平段的1,使用完全不同的边缘提取算法。使用CV_RETR_LIST检索模式能使用此方法。
*/
int method = Imgproc.CHAIN_APPROX_SIMPLE;
Vector<MatOfPoint> contours = new Vector<MatOfPoint>();
Imgproc.findContours(source, contours, rs, mode, method, new Point());
return contours;
}
}
3.答题卡识别实现
controller
package com.acts.opencv.base;
import java.io.File;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.TreeMap;
import java.util.Vector;
import javax.servlet.http.HttpServletResponse;
import org.apache.commons.lang3.StringUtils;
import org.opencv.core.Core;
import org.opencv.core.CvType;
import org.opencv.core.Mat;
import org.opencv.core.MatOfPoint;
import org.opencv.core.Point;
import org.opencv.core.Rect;
import org.opencv.core.Scalar;
import org.opencv.core.Size;
import org.opencv.highgui.Highgui;
import org.opencv.imgproc.Imgproc;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import com.acts.opencv.common.utils.Constants;
import com.acts.opencv.common.utils.RectComp;
import com.acts.opencv.common.web.BaseController;
@Controller
@RequestMapping(value = "card")
public class CardController extends BaseController {
private static final Logger logger = LoggerFactory.getLogger(CardController.class);
/**
* 答题卡识别
* step1 高斯模糊
* 创建者 Songer
* 创建时间 2018年3月22日
*/
@RequestMapping(value = "step1")
public void step1(HttpServletResponse response, String imagefile, Integer ksize) {
System.loadLibrary(Core.NATIVE_LIBRARY_NAME);
logger.info("\n 高斯模糊");
String sourcePath = Constants.PATH + imagefile;
logger.info("url==============" + sourcePath);
// 加载为灰度图显示
Mat source = Highgui.imread(sourcePath, Highgui.CV_LOAD_IMAGE_GRAYSCALE);
Mat destination = new Mat(source.rows(), source.cols(), source.type());
Imgproc.GaussianBlur(source, destination, new Size(2 * ksize + 1, 2 * ksize + 1), 0, 0);
String destPath = Constants.PATH + Constants.DEST_IMAGE_PATH + "card1.png";
File dstfile = new File(destPath);
if (StringUtils.isNotBlank(destPath) && dstfile.isFile() && dstfile.exists()) {
dstfile.delete();
logger.info("删除图片:" + destPath);
}
Highgui.imwrite(destPath, destination);
logger.info("生成目标图片==============" + destPath);
renderString(response, Constants.DEST_IMAGE_PATH + "card1.png");
}
/**
* 答题卡识别
* step2 二值化,反向二值化
* 创建者 Songer
* 创建时间 2018年3月22日
*/
@RequestMapping(value = "step2")
public void step2(HttpServletResponse response, String imagefile, Double thresh) {
System.loadLibrary(Core.NATIVE_LIBRARY_NAME);
logger.info("\n 二值化处理");
// 灰度化
// Imgproc.cvtColor(source, destination, Highgui.CV_LOAD_IMAGE_GRAYSCALE);
String sourcePath = Constants.PATH + imagefile;
logger.info("url==============" + sourcePath);
// 加载为灰度图显示
Mat source = Highgui.imread(sourcePath, Highgui.CV_LOAD_IMAGE_GRAYSCALE);
Mat destination = new Mat(source.rows(), source.cols(), source.type());
Imgproc.threshold(source, destination, thresh, 255, Imgproc.THRESH_BINARY_INV);
String destPath = Constants.PATH + Constants.DEST_IMAGE_PATH + "card2.png";
File dstfile = new File(destPath);
if (StringUtils.isNotBlank(destPath) && dstfile.isFile() && dstfile.exists()) {
dstfile.delete();
logger.info("删除图片:" + destPath);
}
Highgui.imwrite(destPath, destination);
logger.info("生成目标图片==============" + destPath);
renderString(response, Constants.DEST_IMAGE_PATH + "card2.png");
}
/**
* 答题卡识别
* step3 膨胀腐蚀闭运算(针对反向二值图是开运算)
* 创建者 Songer
* 创建时间 2018年3月22日
*/
@RequestMapping(value = "step3")
public void step3(HttpServletResponse response, String imagefile, Integer ksize) {
System.loadLibrary(Core.NATIVE_LIBRARY_NAME);
logger.info("\n 开运算");
// 灰度化
// Imgproc.cvtColor(source, destination, Highgui.CV_LOAD_IMAGE_GRAYSCALE);
String sourcePath = Constants.PATH + imagefile;
logger.info("url==============" + sourcePath);
// 加载为灰度图显示
Mat source = Highgui.imread(sourcePath, Highgui.CV_LOAD_IMAGE_GRAYSCALE);
Mat destination = new Mat(source.rows(), source.cols(), source.type());
Mat element = Imgproc.getStructuringElement(Imgproc.MORPH_RECT, new Size(2 * ksize + 1, 2 * ksize + 1));
Imgproc.morphologyEx(source, destination, Imgproc.MORPH_OPEN, element);
String destPath = Constants.PATH + Constants.DEST_IMAGE_PATH + "card3.png";
File dstfile = new File(destPath);
if (StringUtils.isNotBlank(destPath) && dstfile.isFile() && dstfile.exists()) {
dstfile.delete();
logger.info("删除图片:" + destPath);
}
Highgui.imwrite(destPath, destination);
logger.info("生成目标图片==============" + destPath);
renderString(response, Constants.DEST_IMAGE_PATH + "card3.png");
}
/**
* 答题卡识别
* step4 轮廓识别
* 创建者 Songer
* 创建时间 2018年3月22日
*/
@RequestMapping(value = "step4")
public void step4(HttpServletResponse response, String imagefile) {
System.loadLibrary(Core.NATIVE_LIBRARY_NAME);
logger.info("\n 轮廓识别");
// 灰度化
// Imgproc.cvtColor(source, destination, Highgui.CV_LOAD_IMAGE_GRAYSCALE);
String sourcePath = Constants.PATH + imagefile;
logger.info("url==============" + sourcePath);
// 加载为灰度图显示
Mat source = Highgui.imread(sourcePath, Highgui.CV_LOAD_IMAGE_GRAYSCALE);
Highgui.imwrite("D:\\test\\abc\\source.png", source);
//此处固定写死,取每一行选项,切割后进行轮廓识别
Mat ch1 = source.submat(new Rect(170, 52, 294, 32));
Mat ch2 = source.submat(new Rect(170, 104, 294, 32));
Mat ch3 = source.submat(new Rect(170, 156, 294, 32));
Mat ch4 = source.submat(new Rect(170, 208, 294, 32));
Mat ch5 = source.submat(new Rect(170, 260, 294, 32));
Mat ch6 = source.submat(new Rect(706, 50, 294, 32));
Mat ch7 = source.submat(new Rect(706, 104, 294, 32));
Mat ch8 = source.submat(new Rect(706, 156, 294, 32));
Mat ch9 = source.submat(new Rect(706, 208, 294, 32));
Mat ch10 = source.submat(new Rect(706, 260, 294, 32));
Mat ch11 = source.submat(new Rect(1237, 50, 294, 32));
Mat ch12 = source.submat(new Rect(1237, 104, 294, 32));
Mat ch13 = source.submat(new Rect(1237, 156, 294, 32));
Mat ch14 = source.submat(new Rect(1237, 208, 294, 32));
Mat ch15 = source.submat(new Rect(1237, 260, 294, 32));
Mat ch16 = source.submat(new Rect(1766, 50, 294, 32));
Mat ch17 = source.submat(new Rect(1766, 104, 294, 32));
Mat ch18 = source.submat(new Rect(1766, 156, 294, 32));
Mat ch19 = source.submat(new Rect(1766, 208, 294, 32));
Mat ch20 = source.submat(new Rect(1766, 260, 294, 32));
Mat ch21 = source.submat(new Rect(170, 358, 294, 32));
Mat ch22 = source.submat(new Rect(170, 410, 294, 32));
Mat ch23 = source.submat(new Rect(170, 462, 294, 32));
Mat ch24 = source.submat(new Rect(170, 514, 294, 32));
Mat ch25 = source.submat(new Rect(170, 566, 294, 32));
List<Mat> chlist = new ArrayList<Mat>();
chlist.add(ch1);
chlist.add(ch2);
chlist.add(ch3);
chlist.add(ch4);
chlist.add(ch5);
chlist.add(ch6);
chlist.add(ch7);
chlist.add(ch8);
chlist.add(ch9);
chlist.add(ch10);
chlist.add(ch11);
chlist.add(ch12);
chlist.add(ch13);
chlist.add(ch14);
chlist.add(ch15);
chlist.add(ch16);
chlist.add(ch17);
chlist.add(ch18);
chlist.add(ch19);
chlist.add(ch20);
chlist.add(ch21);
chlist.add(ch22);
chlist.add(ch23);
chlist.add(ch24);
chlist.add(ch25);
Mat hierarchy = new Mat();
java.util.TreeMap<Integer,String> listenAnswer = new TreeMap<Integer,String>();
for (int no=0;no<chlist.size();no++) {
Vector<MatOfPoint> contours = new Vector<MatOfPoint>();
Mat ch = chlist.get(no);
Imgproc.findContours(ch, contours, hierarchy, Imgproc.RETR_TREE, Imgproc.CHAIN_APPROX_SIMPLE, new Point());
Vector<RectComp> rectCompList = new Vector<RectComp>();
for(int i = 0;i<contours.size();i++){
MatOfPoint mop= contours.get(i);
// 获取轮廓外矩,即使用最小矩形将轮廓包裹
Rect rm = Imgproc.boundingRect(mop);
RectComp rc = new RectComp(rm);
rectCompList.add(rc);
}
// System.out.println(no+"size="+rectCompList.size());
Collections.sort(rectCompList);
// for(int t = 0;t<rectCompList.size();t++){
// RectComp rect = rectCompList.get(t);
// System.out.println(rect.getRm().area() + "--------" + rect.getRm().x);
// if (rect.getRm().area() < 300) {// 小于300的pass,完美填图的话是≈1500
// continue;
// }
// if (rect.getRm().x < 68) {
// listenAnswer.put(Integer.valueOf(no), "A");
// } else if ((rect.getRm().x > 68) && (rect.getRm().x < 148)) {
// listenAnswer.put(Integer.valueOf(no), "B");
// } else if ((rect.getRm().x > 148) && (rect.getRm().x < 228)) {
// listenAnswer.put(Integer.valueOf(no), "C");
// } else if (rect.getRm().x > 228) {
// listenAnswer.put(Integer.valueOf(no), "D");
// }
// }
// 因为已经按面积排序了,所以取第一个面积最大的轮廓即可
RectComp rect = rectCompList.get(0);
System.out.println(rect.getRm().area() + "--------" + rect.getRm().x);
if (rect.getRm().area() > 300) {// 小于300的pass,说明未填写,完美填图的话是≈1500
if (rect.getRm().x < 68) {
listenAnswer.put(Integer.valueOf(no), "A");
} else if ((rect.getRm().x > 68) && (rect.getRm().x < 148)) {
listenAnswer.put(Integer.valueOf(no), "B");
} else if ((rect.getRm().x > 148) && (rect.getRm().x < 228)) {
listenAnswer.put(Integer.valueOf(no), "C");
} else if (rect.getRm().x > 228) {
listenAnswer.put(Integer.valueOf(no), "D");
}
} else {
listenAnswer.put(Integer.valueOf(no), "未填写");
}
Mat result = new Mat(ch.size(), CvType.CV_8U, new Scalar(255));
Imgproc.drawContours(result, contours, -1, new Scalar(0, 255, 0), 2);
String destPath = Constants.PATH + Constants.DEST_IMAGE_PATH + "ch" + (no + 1) + ".png";
File dstfile = new File(destPath);
if (StringUtils.isNotBlank(destPath) && dstfile.isFile() && dstfile.exists()) {
dstfile.delete();
logger.info("删除图片:" + destPath);
}
Highgui.imwrite(destPath, result);
logger.info("生成目标图片==============" + result);
}
String resultValue = "最终结果:试题编号-答案<br> ";
for (Integer key : listenAnswer.keySet()) {
resultValue += "【" + (key + 1) + ":" + listenAnswer.get(key) + "】";
if ((key + 1) % 5 == 0) {
resultValue += "<br>";
}
}
renderString(response, resultValue);
}
}
package com.acts.opencv.base;
import java.text.DecimalFormat;
import java.text.NumberFormat;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.Date;
import java.util.List;
import javax.servlet.http.HttpServletResponse;
import org.opencv.core.Core;
import org.opencv.core.CvType;
import org.opencv.core.Mat;
import org.opencv.core.MatOfInt;
import org.opencv.core.MatOfPoint;
import org.opencv.core.MatOfPoint2f;
import org.opencv.core.Point;
import org.opencv.core.Rect;
import org.opencv.core.Scalar;
import org.opencv.core.Size;
import org.opencv.highgui.Highgui;
import org.opencv.imgproc.Imgproc;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import com.acts.opencv.common.utils.Constants;
import com.acts.opencv.common.web.BaseController;
@Controller
@RequestMapping(value = "card2")
public class Card2Controller extends BaseController {
private static final Logger logger = LoggerFactory.getLogger(Card2Controller.class);
/**
* 答题卡识别
* @Author Songer
* @param response
* @param imagefile void
* @Date 2018年10月22日
* 更新日志
* 2018年10月22日 Songer 首次创建
*
*/
@RequestMapping(value = "cardMarking")
public void cardMarking(HttpServletResponse response, String imagefile,Integer picno) {
try {
System.loadLibrary(Core.NATIVE_LIBRARY_NAME);
long t1 = new Date().getTime();
//String sourceimage1 = "D:\\test\\abc\\card\\test4.jpg";
String sourceimage = Constants.PATH + imagefile;
//表格检测,获取到表格内容,是查找识别区域的部分,返回的是校正之后的图像
Mat mat = markingArea(sourceimage);
String destPath = Constants.PATH + Constants.DEST_IMAGE_PATH + "cardResult_3.png";
Highgui.imwrite(destPath, mat);
// Highgui.imwrite("D:\\test\\abc\\card\\card111.png", mat);
//具体的答题卡识别过程,主要就是答案识别部分了
String result = cardResult(mat);
renderString(response, result);
long t2 = new Date().getTime();
logger.info("===耗时"+(t2-t1));
} catch (Exception e) {
e.printStackTrace();
logger.error("答题卡识别异常!", e);
}
}
/**
* 此方法主要是通过边缘检测凸包,查找识别区域。即客观题的框
* @Author Songer
* @Date 2018年9月21日
* 更新日志
* 2018年9月21日 Songer 首次创建
*
*/
public static Mat markingArea(String path){
Mat source = Highgui.imread(path, Highgui.CV_LOAD_IMAGE_COLOR);
Mat img = new Mat();;
Mat result= source.clone();
// 彩色转灰度
Imgproc.cvtColor(source, img, Imgproc.COLOR_BGR2GRAY);
//此处图像预处理,可以使用方式1也可以使用方式2都可以,自己也可以测试下2种方式的差异
// //方式1:通过高斯滤波然后边缘检测膨胀来链接边缘,将轮廓连通便于轮廓识别
// // 高斯滤波,降噪
// Imgproc.GaussianBlur(img, img, new Size(3,3), 2, 2);
// // Canny边缘检测
// Imgproc.Canny(img, img, 20, 60, 3, false);
// // 膨胀,连接边缘
// Imgproc.dilate(img, img, new Mat(), new Point(-1,-1), 3, 1, new Scalar(1));
//
//方式2:使用形态学梯度算法,此算法来保留物体的边缘轮廓很有效果
Mat element = Imgproc.getStructuringElement(Imgproc.MORPH_RECT, new Size(5,5));
Imgproc.morphologyEx(img, img, Imgproc.MORPH_GRADIENT, element);
//图像二值化,使用的是OTSU二值化,阈值为170,也可以使用自适用二值化
// Imgproc.adaptiveThreshold(img,img, 255, Imgproc.ADAPTIVE_THRESH_MEAN_C, Imgproc.THRESH_BINARY_INV, 51, 10);
Imgproc.threshold(img,img, 170, 255, Imgproc.THRESH_BINARY|Imgproc.THRESH_OTSU);
String destPath = Constants.PATH + Constants.DEST_IMAGE_PATH + "cardResult_1.png";
Highgui.imwrite(destPath, img);
// Highgui.imwrite("D:\\test\\abc\\card\\card1.png", img);
List<MatOfPoint> contours = new ArrayList<>();
Mat hierarchy = new Mat();
//轮廓查找,主要就是找最外表格框
Imgproc.findContours(img, contours, hierarchy, Imgproc.RETR_EXTERNAL, Imgproc.CHAIN_APPROX_SIMPLE);
// 找出轮廓对应凸包的四边形拟合
List<MatOfPoint> squares = new ArrayList<>();
List<MatOfPoint> hulls = new ArrayList<>();
MatOfInt hull = new MatOfInt();
MatOfPoint2f approx = new MatOfPoint2f();
approx.convertTo(approx, CvType.CV_32F);
for (MatOfPoint contour: contours) {
// 边框的凸包
Imgproc.convexHull(contour, hull);
// 用凸包计算出新的轮廓点
Point[] contourPoints = contour.toArray();
int[] indices = hull.toArray();
List<Point> newPoints = new ArrayList<>();
for (int index : indices) {
newPoints.add(contourPoints[index]);
}
MatOfPoint2f contourHull = new MatOfPoint2f();
contourHull.fromList(newPoints);
// 多边形拟合凸包边框(此时的拟合的精度较低)
Imgproc.approxPolyDP(contourHull, approx, Imgproc.arcLength(contourHull, true)*0.02, true);
// 筛选出面积大于某一阈值的,且四边形的各个角度都接近直角的凸四边形
MatOfPoint approxf1 = new MatOfPoint();
approx.convertTo(approxf1, CvType.CV_32S);
//此处是筛选表格框,面积大于40000
if (approx.rows() == 4 && Math.abs(Imgproc.contourArea(approx)) > 40000 &&
Imgproc.isContourConvex(approxf1)) {
double maxCosine = 0;
for (int j = 2; j < 5; j++) {
double cosine = Math.abs(getAngle(approxf1.toArray()[j%4], approxf1.toArray()[j-2], approxf1.toArray()[j-1]));
maxCosine = Math.max(maxCosine, cosine);
}
// 考虑到图片倾斜等情况,角度大概72度
if (maxCosine < 0.3) {
MatOfPoint tmp = new MatOfPoint();
contourHull.convertTo(tmp, CvType.CV_32S);
squares.add(approxf1);
hulls.add(tmp);
}
}
}
// 找出外接矩形最大的四边形
int index = findLargestSquare(squares);
MatOfPoint largest_square = squares.get(index);
if (largest_square.rows() == 0 || largest_square.cols() == 0)
return result;
// 找到这个最大的四边形对应的凸边框,再次进行多边形拟合,此次精度较高,拟合的结果可能是大于4条边的多边形
MatOfPoint contourHull = hulls.get(index);
MatOfPoint2f tmp = new MatOfPoint2f();
contourHull.convertTo(tmp, CvType.CV_32F);
Imgproc.approxPolyDP(tmp, approx, 3, true);
List<Point> newPointList = new ArrayList<>();
double maxL = Imgproc.arcLength(approx, true) * 0.02;
// 找到高精度拟合时得到的顶点中 距离小于低精度拟合得到的四个顶点maxL的顶点,排除部分顶点的干扰
for (Point p : approx.toArray()) {
if (!(getSpacePointToPoint(p, largest_square.toList().get(0)) > maxL &&
getSpacePointToPoint(p, largest_square.toList().get(1)) > maxL &&
getSpacePointToPoint(p, largest_square.toList().get(2)) > maxL &&
getSpacePointToPoint(p, largest_square.toList().get(3)) > maxL)) {
newPointList.add(p);
}
}
// 找到剩余顶点连线中,边长大于 2 * maxL的四条边作为四边形物体的四条边
List<double[]> lines = new ArrayList<>();
for (int i = 0; i < newPointList.size(); i++) {
Point p1 = newPointList.get(i);
Point p2 = newPointList.get((i+1) % newPointList.size());
if (getSpacePointToPoint(p1, p2) > 2 * maxL) {
lines.add(new double[]{p1.x, p1.y, p2.x, p2.y});
logger.info("p1x:"+p1.x+" p1y:"+p1.y+" p2x:"+p2.x+" p2y:"+p2.y);
//画出4条边线,真正识别过程中这些都是可以注释掉的,只是为了方便观察
Core.line(source, new Point(p1.x, p1.y), new Point( p2.x, p2.y), new Scalar(255, 0, 0),4);
}
}
destPath = Constants.PATH + Constants.DEST_IMAGE_PATH + "cardResult_2.png";
Highgui.imwrite(destPath, source);
// Highgui.imwrite("D:\\test\\abc\\card\\card2.png", source);
// 计算出这四条边中 相邻两条边的交点,即物体的四个顶点
List<Point> corners = new ArrayList<>();
for (int i = 0; i < lines.size(); i++) {
Point corner = computeIntersect(lines.get(i),lines.get((i+1) % lines.size()));
corners.add(corner);
}
// 对顶点顺时针排序
sortCorners(corners);
// 计算目标图像的尺寸
Point p0 = corners.get(0);
Point p1 = corners.get(1);
Point p2 = corners.get(2);
Point p3 = corners.get(3);
logger.info(" "+p0.x+" "+p0.y);
logger.info(" "+p1.x+" "+p1.y);
logger.info(" "+p2.x+" "+p2.y);
logger.info(" "+p3.x+" "+p3.y);
double space0 = getSpacePointToPoint(p0, p1);
double space1 = getSpacePointToPoint(p1, p2);
double space2 = getSpacePointToPoint(p2, p3);
double space3 = getSpacePointToPoint(p3, p0);
// 使用最宽和最长的边作为进行图像矫正目标图像的长宽
double imgWidth = space1 > space3 ? space1 : space3;
double imgHeight = space0 > space2 ? space0 : space2;
logger.info("imgWidth:"+imgWidth+" imgHeight:"+imgHeight);
// 如果提取出的图片宽小于高,则旋转90度,因为示例中的矩形框是宽>高的,如果宽小于高应该是图片旋转了
if (imgWidth > imgHeight) {
logger.info("----in");
double temp = imgWidth;
imgWidth = imgHeight;
imgHeight = temp;
Point tempPoint = p0.clone();
p0 = p1.clone();
p1 = p2.clone();
p2 = p3.clone();
p3 = tempPoint.clone();
}
// Mat quad = Mat.zeros((int)imgHeight * 2, (int)imgWidth * 2, CvType.CV_8UC3);
Mat quad = Mat.zeros((int)imgHeight, (int)imgWidth, CvType.CV_8UC3);
MatOfPoint2f cornerMat = new MatOfPoint2f(p0, p1, p2, p3);
// MatOfPoint2f quadMat = new MatOfPoint2f(new Point(imgWidth*0.4, imgHeight*1.6),
// new Point(imgWidth*0.4, imgHeight*0.4),
// new Point(imgWidth*1.6, imgHeight*0.4),
// new Point(imgWidth*1.6, imgHeight*1.6));
//quadMat目标图像的点设置,以之前取出的最长的长宽作为新图像的长宽,创建一个图层
MatOfPoint2f quadMat = new MatOfPoint2f(new Point(0, 0),
new Point(imgWidth, 0),
new Point(imgWidth, imgHeight),
new Point(0, imgHeight));
// 提取图像,使用warpPerspective做图像的透视变换
Mat transmtx = Imgproc.getPerspectiveTransform(cornerMat, quadMat);
Imgproc.warpPerspective(result, quad, transmtx, quad.size());
return quad;
}
// 根据三个点计算中间那个点的夹角 pt1 pt0 pt2
private static double getAngle(Point pt1, Point pt2, Point pt0){
double dx1 = pt1.x - pt0.x;
double dy1 = pt1.y - pt0.y;
double dx2 = pt2.x - pt0.x;
double dy2 = pt2.y - pt0.y;
return (dx1*dx2 + dy1*dy2)/Math.sqrt((dx1*dx1 + dy1*dy1)*(dx2*dx2 + dy2*dy2) + 1e-10);
}
// 找到最大的正方形轮廓
private static int findLargestSquare(List<MatOfPoint> squares) {
if (squares.size() == 0)
return -1;
int max_width = 0;
int max_height = 0;
int max_square_idx = 0;
int currentIndex = 0;
for (MatOfPoint square : squares) {
Rect rectangle = Imgproc.boundingRect(square);
if (rectangle.width >= max_width && rectangle.height >= max_height) {
max_width = rectangle.width;
max_height = rectangle.height;
max_square_idx = currentIndex;
}
currentIndex++;
}
return max_square_idx;
}
// 点到点的距离
private static double getSpacePointToPoint(Point p1, Point p2) {
double a = p1.x - p2.x;
double b = p1.y - p2.y;
return Math.sqrt(a * a + b * b);
}
// 两直线的交点
private static Point computeIntersect(double[] a, double[] b) {
if (a.length != 4 || b.length != 4)
throw new ClassFormatError();
double x1 = a[0], y1 = a[1], x2 = a[2], y2 = a[3], x3 = b[0], y3 = b[1], x4 = b[2], y4 = b[3];
double d = ((x1 - x2) * (y3 - y4)) - ((y1 - y2) * (x3 - x4));
if (d != 0) {
Point pt = new Point();
pt.x = ((x1 * y2 - y1 * x2) * (x3 - x4) - (x1 - x2) * (x3 * y4 - y3 * x4)) / d;
pt.y = ((x1 * y2 - y1 * x2) * (y3 - y4) - (y1 - y2) * (x3 * y4 - y3 * x4)) / d;
return pt;
}
else
return new Point(-1, -1);
}
// 对多个点按顺时针排序
private static void sortCorners(List<Point> corners) {
if (corners.size() == 0) return;
Point p1 = corners.get(0);
int index = 0;
for (int i = 1; i < corners.size(); i++) {
Point point = corners.get(i);
if (p1.x > point.x) {
p1 = point;
index = i;
}
}
corners.set(index, corners.get(0));
corners.set(0, p1);
Point lp = corners.get(0);
for (int i = 1; i < corners.size(); i++) {
for (int j = i + 1; j < corners.size(); j++) {
Point point1 = corners.get(i);
Point point2 = corners.get(j);
if ((point1.y-lp.y*1.0)/(point1.x-lp.x)>(point2.y-lp.y*1.0)/(point2.x-lp.x)) {
Point temp = point1.clone();
corners.set(i, corners.get(j));
corners.set(j, temp);
}
}
}
}
/**
* 答题卡识别,增加注释
* 已得到矫正后的图像后,进行后续答案识别算法过程
* @Author Songer
* @param mat
* @return String
* @Date 2018年12月19日
* 更新日志
* 2018年12月19日 王嵩 首次创建
*
*/
public String cardResult(Mat mat){
//设置剪切的边距,目的是裁剪表格边框,防止边框影响轮廓查找,这里设置为20像素
int cutsize = 20;
Mat img_cut = mat.submat(cutsize,mat.rows()-cutsize,cutsize,mat.cols()-cutsize);new Mat();
Mat img_gray = img_cut.clone();
//图像灰度化
Imgproc.cvtColor(img_cut, img_gray, Imgproc.COLOR_BGR2GRAY);
//图像二值化,注意是反向二值化以及OTSU算法
Imgproc.threshold(img_gray,img_gray, 170, 255, Imgproc.THRESH_BINARY_INV|Imgproc.THRESH_OTSU);
Mat temp = img_gray.clone();
//此处使用的是方式2,形态学梯度算法保留填图选项的边框
// //方式1:通过高斯滤波然后边缘检测膨胀来链接边缘,将轮廓连通便于轮廓识别
// // 高斯滤波,降噪
// Imgproc.GaussianBlur(temp, temp, new Size(3,3), 2, 2);
// // Canny边缘检测
// Imgproc.Canny(temp, temp, 20, 60, 3, false);
// // 膨胀,连接边缘
// Imgproc.dilate(temp, temp, new Mat(), new Point(-1,-1), 3, 1, new Scalar(1));
//方式2:使用形态学梯度算法,此算法来保留物体的边缘轮廓很有效果
Mat element = Imgproc.getStructuringElement(Imgproc.MORPH_RECT, new Size(5,5));
Imgproc.morphologyEx(temp, temp, Imgproc.MORPH_GRADIENT, element);
Imgproc.threshold(temp, temp, 170, 255, Imgproc.THRESH_BINARY|Imgproc.THRESH_OTSU);
String destPath = Constants.PATH + Constants.DEST_IMAGE_PATH + "cardResult_4.png";
Highgui.imwrite(destPath, temp);
// Highgui.imwrite("D:\\test\\abc\\card\\card3.png", temp);
//按比例截取,此处是根据原答题卡的各列的比例截取成4列,学号那列还要横向分隔一下,上半部分是学号下半部分是答题卡
Mat cut1 = temp.submat(0,temp.rows(),0,(int)(0.275*temp.cols()));
Mat cut_gray1 = img_gray.submat(0,img_gray.rows(),0,(int)(0.275*img_gray.cols()));
Mat cut2 = temp.submat(0,temp.rows(),(int)(0.275*temp.cols()),(int)(0.518*temp.cols()));
Mat cut_gray2 = img_gray.submat(0,img_gray.rows(),(int)(0.275*img_gray.cols()),(int)(0.518*img_gray.cols()));
Mat cut3 = temp.submat(0,temp.rows(),(int)(0.518*temp.cols()),(int)(0.743*temp.cols()));
Mat cut_gray3 = img_gray.submat(0,img_gray.rows(),(int)(0.518*img_gray.cols()),(int)(0.743*img_gray.cols()));
Mat cut4 = temp.submat((int)(0.387*temp.rows()),temp.rows(),(int)(0.743*temp.cols()),temp.cols());
Mat cut_gray4 = img_gray.submat((int)(0.387*img_gray.rows()),img_gray.rows(),(int)(0.743*img_gray.cols()),img_gray.cols());
//学号
Mat cut5 = temp.submat(0,(int)(0.387*temp.rows()),(int)(0.743*temp.cols()),temp.cols());
Mat cut_gray5 = img_gray.submat(0,(int)(0.387*img_gray.rows()),(int)(0.743*img_gray.cols()),img_gray.cols());
// Highgui.imwrite("D:\\test\\abc\\card\\card_cut1.png", cut1);
// Highgui.imwrite("D:\\test\\abc\\card\\card_cut1_gary.png", cut_gray1);
// Highgui.imwrite("D:\\test\\abc\\card\\card_cut2.png", cut2);
// Highgui.imwrite("D:\\test\\abc\\card\\card_cut3.png", cut3);
// Highgui.imwrite("D:\\test\\abc\\card\\card_cut4.png", cut4);
// Highgui.imwrite("D:\\test\\abc\\card\\card_cut5.png", cut5);
List<String> resultList = new ArrayList<String>();
//按列处理
List<String> list1 = processByCol(cut1,cut_gray1,img_cut,5);
List<String> list2 = processByCol(cut2,cut_gray2,img_cut,5);
List<String> list3 = processByCol(cut3,cut_gray3,img_cut,4);
List<String> list4 = processByCol(cut4,cut_gray4,img_cut,4);
//学号单独处理
List<String> list5 = processByCol(cut5,cut_gray5,img_cut,4);
resultList.addAll(list1);
resultList.addAll(list2);
resultList.addAll(list3);
resultList.addAll(list4);
String studentNo = getStudentNo(list5);
logger.info("学生学号为:"+studentNo);
StringBuffer result = new StringBuffer("学生学号为:"+studentNo);
for (int i = 0; i < resultList.size(); i++) {
result.append(resultList.get(i));
logger.info(resultList.get(i));
}
return result.toString();
}
/**
* 根据列单独处理
* @Author Songer
* @param cut1 传入的答案列,一般1-20一列
* @param cut_gary 传入的答案列未处理过
* @param temp 表格范围mat
* @param toIndex 列答案数,即每几个答案一组
* @Date 2018年9月20日
* 更新日志
* 2018年9月20日 Songer 首次创建
*
*/
private static List<String> processByCol(Mat cut1,Mat cut_gray,Mat temp,int answerCols) {
List<String> result = new ArrayList<String>();
List<MatOfPoint> contours = new ArrayList<MatOfPoint>();
List<MatOfPoint> answerList = new ArrayList<MatOfPoint>();
Mat hierarchy = new Mat();
//进行轮廓查找,注意因为该答题卡特征是闭合填图区域,所以用这种方式,如果是非闭合填涂区域,则不适用
Imgproc.findContours(cut1.clone(), contours, hierarchy, Imgproc.RETR_EXTERNAL, Imgproc.CHAIN_APPROX_SIMPLE);
// logger.info(contours.size());
// logger.info("-----w------"+(temp.width()*80/2693-20));
// logger.info("-----h------"+(temp.height()*80/2764-20));
for(int i = 0;i<contours.size();i++){
MatOfPoint mop= contours.get(i);
//每个轮廓给出外包矩形,便于操作
Rect rect = Imgproc.boundingRect(mop);
// logger.info("-----------"+rect.width+" "+rect.height);
//一个填图区域大概占整个表格的w:80/2733 h:0/2804,,所以排除掉太小的轮廓和过大的轮廓
// //绘制每个轮廓图
// Imgproc.drawContours(cut1, contours, i, new Scalar(170), 2);
//此处是为了排除杂点,较小或较大的轮廓都是非填图选项,可以排除,可以按实际情况灵活变动限制条件,最好输出出轮廓便于观察
if(rect.width>(temp.width()*80/2693-20) && rect.height>(temp.height()*80/2764-20) && rect.width<temp.width()*0.05 && rect.height<temp.height()*0.05){
// if(rect.width>50&&rect.height>50&&rect.area()>2500&&rect.area()<10000){
// Core.rectangle(img_cut, new Point(rect.x, rect.y), new Point(rect.x + rect.width, rect.y
// + rect.height), new Scalar(0, 255, 0), 2);
answerList.add(mop);
}
}
Collections.sort(answerList, new Comparator<MatOfPoint>() {
//按照y坐标升序排列
@Override
public int compare(MatOfPoint o1, MatOfPoint o2) {
Rect rect1 = Imgproc.boundingRect(o1);
Rect rect2 = Imgproc.boundingRect(o2);
if(rect1.y<rect2.y){
return -1;
}else if(rect1.y>rect2.y){
return 1;
}else{
return 0;
}
}
});
//每4/5个一组组成新list,并按x坐标排序
//该方式依赖于预处理后的图片没有干扰轮廓。如果图像处理不佳,干扰项没排除,那么此处的分组就会错乱。
//提供一个优化思路:对于闭合填涂区域的可以使用水平垂直投影进行坐标定位点的判定。
int queno = 1;
int totoSize = answerList.size();
for (int i = 0; i < totoSize; i+=answerCols) {
int toIndex = i+answerCols;
if(toIndex>totoSize){
toIndex = totoSize-1;
}
List<MatOfPoint> newList = answerList.subList(i,toIndex);
Collections.sort(newList, new Comparator<MatOfPoint>() {
//按照x坐标升序排列
@Override
public int compare(MatOfPoint o1, MatOfPoint o2) {
Rect rect1 = Imgproc.boundingRect(o1);
Rect rect2 = Imgproc.boundingRect(o2);
if(rect1.x<rect2.x){
return -1;
}else if(rect1.x>rect2.x){
return 1;
}else{
return 0;
}
}
});
String resultChoose = "";
for (int j = 0; j < newList.size(); j++) {
Imgproc.drawContours(cut_gray, newList, j, new Scalar(170), 2);
//掩模提取出轮廓
Mat mask = Mat.zeros(cut_gray.size(), CvType.CV_8UC1);
//绘制轮廓便于观察
Imgproc.drawContours(mask, newList, j, new Scalar(255), -1);
Mat dst = new Mat();
Core.bitwise_and(cut_gray, mask, dst);
//获取填涂百分比,填涂区域的二值化后取出非0点/掩模的轮廓面积
double p100 = Core.countNonZero(dst) * 100 / Core.countNonZero(mask);
String anno = index2ColName(j);
//认为非0像素超过80%的算是填涂
if(p100>80){
resultChoose += anno;
logger.info(p100+" 第"+queno+"行:选项("+anno+")填涂");
}else{
logger.info(p100+" 第"+queno+"行:选项("+anno+")未填涂");
}
// Highgui.imwrite("D:\\test\\abc\\card\\card_x"+i+j+".png", dst);
if(i==0&&j==0){//输出一下第一个掩模
String destPath = Constants.PATH + Constants.DEST_IMAGE_PATH + "cardResult_5.png";
Highgui.imwrite(destPath, dst);
}
}
result.add(resultChoose);
queno++;
}
// Highgui.imwrite("D:\\test\\abc\\card\\card4.png", cut1);
return result;
}
//编号转答案0-A 1-B
public static String index2ColName(int index){
if (index < 0) {
return null;
}
int num = 65;// A的Unicode码
String colName = "";
do {
if (colName.length() > 0) {
index--;
}
int remainder = index % 26;
colName = ((char) (remainder + num)) + colName;
index = (int) ((index - remainder) / 26);
} while (index > 0);
return colName;
}
/**根据表元的列名转换为列号
* @param colName 列名, 从A开始
* @return A1->0; B1->1...AA1->26
*/
public static int colName2Index(String colName) {
int index = -1;
int num = 65;// A的Unicode码
int length = colName.length();
for (int i = 0; i < length; i++) {
char c = colName.charAt(i);
if (Character.isDigit(c)) break;// 确定指定的char值是否为数字
index = (index + 1) * 26 + (int)c - num;
}
return index;
}
/**
* 根据返回的结果集转换为学号
* 因为学号部分的处理跟答案一样是一横行进行处理的,同时返回的是ABCDE等选项结果
* 转换公式为:学生学号=遍历list的index值*(A=1000,B=100,C=10,D=1)相加
* @Author Songer
* @param resultList
* @return String
* @Date 2018年9月20日
* 更新日志
* 2018年9月20日 Songer 首次创建
*
*/
public static String getStudentNo(List<String> resultList){
int studentNo = 0;
for (int i = 0; i < resultList.size(); i++) {
String result = resultList.get(i);
if (result.contains("A")) {
studentNo += 1000 * i;
}
if (result.contains("B")) {
studentNo += 100 * i;
}
if (result.contains("C")) {
studentNo += 10 * i;
}
if (result.contains("D")) {
studentNo += i;
}
}
NumberFormat formatter = new DecimalFormat("0000");
String number = formatter.format(studentNo);
return number;
}
}
/**
* Copyright © 2016-2020 公众学业 All rights reserved.
*/
package com.acts.opencv.common.web;
import java.beans.PropertyEditorSupport;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletResponse;
import org.apache.commons.lang3.StringEscapeUtils;
import org.springframework.http.MediaType;
import org.springframework.ui.Model;
import org.springframework.web.bind.WebDataBinder;
import org.springframework.web.bind.annotation.InitBinder;
import org.springframework.web.servlet.mvc.support.RedirectAttributes;
import com.acts.opencv.common.mapper.JsonMapper;
import com.acts.opencv.common.utils.Constants;
/**
* 控制器支持类
* 创建者 Songer
* 创建时间 2016年7月21日
*
*/
public abstract class BaseController{
/**
* 添加Model消息
* @param message
*/
protected void addMessage(Model model, String... messages) {
StringBuilder sb = new StringBuilder();
for (String message : messages){
sb.append(message).append(messages.length>1?"<br/>":"");
}
model.addAttribute("message", sb.toString());
}
/**
* 添加Flash消息
* @param message
*/
protected void addMessage(RedirectAttributes redirectAttributes, String... messages) {
StringBuilder sb = new StringBuilder();
for (String message : messages){
sb.append(message).append(messages.length>1?"<br/>":"");
}
redirectAttributes.addFlashAttribute("message", sb.toString());
}
/**
* 客户端返回JSON字符串
* @param response
* @param string
* @return
*/
protected void renderString(HttpServletResponse response, Object object) {
try {
response.reset();
response.setContentType(MediaType.APPLICATION_JSON_UTF8_VALUE);
response.setHeader("Cache-Control", "no-cache, must-revalidate");
PrintWriter writer = response.getWriter();
writer.write(JsonMapper.toJsonString(object));
writer.flush();
writer.close();
} catch (IOException e) {
e.printStackTrace();
}
}
/**
* 客户端返回图片类型
* @param response
* @param object void
* @throws IOException
* @Date 2018年3月13日
* 更新日志
* 2018年3月13日 Songer 首次创建
*
*/
protected void renderImage(HttpServletResponse response, byte[] object) {
try {
response.reset();
response.setContentType("image/*");
ServletOutputStream output = response.getOutputStream();
output.flush();
output.write(object);
output.close();
// ServletOutputStream output = response.getOutputStream();
// FileInputStream fis = new FileInputStream("E:\\tomcat7\\webapps\\java_opencv\\statics\\distimage\\lena.png");
// byte[] buffer = new byte[1024];
// int i = -1;
// while ((i = fis.read(buffer)) != -1) {
// output.write(buffer, 0, i);
// }
// output.flush();
// output.close();
// fis.close();
} catch (IOException e) {
// 如果是ClientAbortException异常,可以不用管,原因是页面参数变化太快,response请求被中断
try {
// response.reset();
PrintWriter writer = response.getWriter();
response.setContentType("text/html;charset=utf-8");
writer.write("无法打开图片!");
writer.close();
} catch (IOException e1) {
e1.printStackTrace();
}
e.printStackTrace();
}
}
/**
* 客户端返回字符串
* @param response
* @param string
* @return
*/
protected void renderString(HttpServletResponse response) {
renderString(response, Constants.SUCCESS);
}
/**
* 初始化数据绑定
* 1. 将所有传递进来的String进行HTML编码,防止XSS攻击
* 2. 将字段中Date类型转换为String类型
*/
@InitBinder
protected void initBinder(WebDataBinder binder) {
// String类型转换,将所有传递进来的String进行HTML编码,防止XSS攻击
binder.registerCustomEditor(String.class, new PropertyEditorSupport() {
@Override
public void setAsText(String text) {
setValue(text == null ? null : StringEscapeUtils.escapeHtml4(text.trim()));
}
@Override
public String getAsText() {
Object value = getValue();
return value != null ? value.toString() : "";
}
});
}
}
4.其他工具类
package com.acts.opencv.common.utils;
import org.springframework.web.context.ContextLoader;
/**
* 常量 创建者 Songer 创建时间 2018年3月09日
*
*/
public class Constants {
public static final String CURRENT_USER = "UserInfo";
public static final String WECHAT_USER = "weChatUserInfo";
public static final String REFERENCE_CODE = "referenceCode";
public static final String SUCCESS = "success";
public static final String ERROR = "error";
public static final String SF_FILE_SEPARATOR = System.getProperty("file.separator");// 文件分隔符
public static final String SF_LINE_SEPARATOR = System.getProperty("line.separator");// 行分隔符
public static final String SF_PATH_SEPARATOR = System.getProperty("path.separator");// 路径分隔符
public static final String PATH = ContextLoader.getCurrentWebApplicationContext().getServletContext().getRealPath("/");
/**
* 文件
*/
public static final String SOURCE_IMAGE_PATH = Constants.SF_FILE_SEPARATOR + "statics"
+ Constants.SF_FILE_SEPARATOR + "sourceimage" + Constants.SF_FILE_SEPARATOR;// 图片原地址
public static final String DEST_IMAGE_PATH = Constants.SF_FILE_SEPARATOR + "statics" + Constants.SF_FILE_SEPARATOR
+ "destimage" + Constants.SF_FILE_SEPARATOR;// 图片生成地址
/**
* 返回参数规范
*/
/** 区分类型 1 -- 无错误,Code重复 */
public static final String CODE_DUPLICATE = "1";
/** 区分类型 2 -- 无错误,名称重复 */
public static final String NAME_DUPLICATE = "2";
/** 区分类型 3 -- 数量超出 */
public static final String NUMBER_OVER = "3";
/** 区分类型 0 -- 无错误,程序正常执行 */
public static final String NO_ERROR = "0";
/** 区分类型 -1 -- 无错误,返回结果为空 */
public static final String NULL_POINTER = "-1";
/** 区分类型 -2 -- 错误,参数不正确 */
public static final String INCORRECT_PARAMETER = "-2";
/** 区分类型 -3 -- 错误,程序执行错误 */
public static final String PROGRAM_EXECUTION_ERROR = "-3";
/** 区分类型 -5 -- 错误,数据已删除 */
public static final String DATA_DELETED = "-5";
/** 区分类型 -6 -- 错误,参数不一致(验证码) */
public static final String DATA_NOT_SAME = "-6";
/**json文件缺失 */
public static final String NO_JSON_FILE = "-7";
/**
* 分页中可能用到的常量
*/
public static final Integer PAGE_SIZE=10;//一页共有十条内容
}
package com.acts.opencv.common.utils;
import org.opencv.core.Rect;
public class RectComp implements Comparable<Object> {
private Rect rm;
public Rect getRm() {
return rm;
}
public void setRm(Rect rm) {
this.rm = rm;
}
public RectComp() {
super();
}
public RectComp(Rect rm) {
super();
this.rm = rm;
}
// @Override
// public int compareTo(Object object) {
// if(this == object){
// return 0;
// } else if (object != null && object instanceof RectComp) {
// RectComp rect = (RectComp) object;
// if (rm.x <= rect.rm.x) {
// return -1;
// }else{
// return 1;
// }
// }else{
// return -1;
// }
// }
@Override
// 按面积排序,最大的放第一个
public int compareTo(Object object) {
if(this == object){
return 0;
} else if (object != null && object instanceof RectComp) {
RectComp rect = (RectComp) object;
if (rm.area() >= rect.rm.area()) {
return -1;
} else {
return 1;
}
} else {
return -1;
}
}
}