目录
摘要
本文介绍windows系统下springboot+opencv进行对图片中的人脸进行提取,适合入门学习
参考:springboot,opencv
一、配置项目
1. 创建Springboot应用
2. 下载opencv开发资源包
下载windows下opencv开发包4.5.1点击下载,下载完成后打开exe提取资源到本地
3. 拷贝jar包,修改pom配置
在springboot项目resources目录下新建lib包,将上一步提取资源目录下的\build\java\opencv-451.jar复制到lib包下
修改pom文件,将该jar包添加到项目依赖中
<dependency> <groupId>org.opencv</groupId> <artifactId>opencv</artifactId> <version>4.5.1</version> <scope>system</scope> <systemPath>${project.basedir}/src/main/resources/lib/opencv-451.jar</systemPath> </dependency>
添加动态库
在项目根目录新建文件夹dll,将opencv资源目录下的\build\java x64,x86复制到dll下添加opencv分类器
在项目根目录新建文件夹haarcascades,将opencv资源目录下\sources\data\haarcascades中的xml文件复制到该目录
注: 也可以使用其他分类器,如haarcascades_cuda、lbpcascades下的,效果对比参考此处最终的项目结构如下:
pom完整文件
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.4.2</version> <relativePath/> <!-- lookup parent from repository --> </parent> <groupId>com.demo</groupId> <artifactId>face</artifactId> <version>0.0.1-SNAPSHOT</version> <name>face</name> <description>Demo project for Spring Boot Face recognize by opencv</description> <properties> <maven.compiler.source>8</maven.compiler.source> <maven.compiler.target>8</maven.compiler.target> </properties> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> <dependency> <groupId>org.opencv</groupId> <artifactId>opencv</artifactId> <version>4.5.1</version> <scope>system</scope> <systemPath>${project.basedir}/src/main/resources/lib/opencv-451.jar</systemPath> </dependency> </dependencies> <build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> </plugin> </plugins> </build> </project>
二、代码编写
1. 人脸处理工具类FaceUtils.java源码
package com.demo.face;
import org.opencv.core.*;
import org.opencv.imgproc.Imgproc;
import org.opencv.objdetect.CascadeClassifier;
import java.io.File;
import java.util.Comparator;
/**
* //TODO 描述
*
* @author: Ray.ma
* @Data: 2021-01-28 11:01
* @Version: 1.0.0
*/
public class FaceUtils {
static {
String dir = System.getProperty("user.dir").replace("/", "\\");
String filePath = "\\dll\\" + (System.getProperty("java.vm.name").contains("64") ? "x64" : "x86") + "\\opencv_java451.dll";
String path = dir.concat(filePath);
File file = new File(path);
if (!file.exists()) {
throw new RuntimeException("找不到动态库".concat(path));
}
System.load(path);
System.out.println(path.concat(" load success"));
}
/**
* 检测人脸数量
*
* @param base64
* @return
*/
public static int checkFaceNo(String base64) {
String path = System.getProperty("user.dir").concat("/haarcascades/haarcascade_frontalface_alt.xml");
CascadeClassifier faceDetector = new CascadeClassifier(path);
MatOfRect faceDetections = new MatOfRect();
faceDetector.detectMultiScale(StreamUtils.base642Mat(base64), faceDetections);
return faceDetections.toArray().length;
}
public static String extractMaxFace(String base64) {
String path = System.getProperty("user.dir").concat("/haarcascades/haarcascade_frontalface_alt.xml");
CascadeClassifier faceDetector = new CascadeClassifier(path);
MatOfRect faceDetections = new MatOfRect();
Mat mat = StreamUtils.base642Mat(base64);
faceDetector.detectMultiScale(mat, faceDetections);
if (faceDetections.toArray().length > 0) {
Rect rect = faceDetections.toList().parallelStream().max(Comparator.comparingInt(o -> o.height * o.width)).get();
Mat face = imageCut(mat, rect.x, rect.y, rect.width, rect.height);
return StreamUtils.catToBase64(face);
}
return null;
}
/**
* @param image 原始图像
* @param posX x坐标
* @param posY y坐标
* @param width 宽度
* @param height 高度
* @return
*/
private static Mat imageCut(Mat image, int posX, int posY, int width, int height) {
// 截取的区域:参数,坐标X,坐标Y,截图宽度,截图长度
Rect rect = new Rect(posX, posY, width, height);
// 两句效果一样
Mat sub = image.submat(rect); // Mat sub = new Mat(image,rect);
Mat mat = new Mat();
Size size = new Size(width, height);
Imgproc.resize(sub, mat, size);// 将人脸进行截图并保存
return mat;
}
public static String markFace(String base64Images) {
String path = System.getProperty("user.dir").concat("/haarcascades/haarcascade_frontalface_alt.xml");
CascadeClassifier faceDetector = new CascadeClassifier(path);
MatOfRect faceDetections = new MatOfRect();
Mat mat = StreamUtils.base642Mat(base64Images);
faceDetector.detectMultiScale(mat, faceDetections);
if (faceDetections.toArray().length > 0) {
for (Rect rect : faceDetections.toList()) {
Imgproc.rectangle(mat, new Point(rect.x, rect.y), new Point(rect.x + rect.width, rect.y + rect.height), new Scalar(0, 255, 0), 3);
}
}
return StreamUtils.catToBase64(mat);
}
}
用到的StreamUtils.java源码
package com.demo.face;
import org.opencv.core.CvType;
import org.opencv.core.Mat;
import org.opencv.core.MatOfByte;
import org.opencv.imgcodecs.Imgcodecs;
import org.springframework.util.Base64Utils;
import sun.misc.BASE64Decoder;
import javax.imageio.ImageIO;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.awt.image.DataBufferByte;
import java.io.ByteArrayInputStream;
import java.io.IOException;
/**
* //TODO 描述
*
* @author: Ray.ma
* @Data: 2021-01-28 11:26
* @Version: 1.0.0
*/
public class StreamUtils {
/**
* 装换回编码
*
* @param correctMat
* @return
*/
public static String catToBase64(Mat correctMat) {
return bufferToBase64(toByteArray(correctMat));
}
/**
* 转换成base64编码
*
* @param buffer
* @return
*/
public static String bufferToBase64(byte[] buffer) {
return Base64Utils.encodeToString(buffer);
}
/**
* base64编码转换成字节数组
*
* @param base64Str
* @return
*/
public static byte[] base64ToByteArray(String base64Str) {
return Base64Utils.decodeFromString(base64Str);
}
/**
* base64 编码转换为 BufferedImage
*
* @param base64
* @return
*/
public static BufferedImage base64ToBufferedImage(String base64) {
BASE64Decoder Base64 = new BASE64Decoder();
try {
byte[] bytes1 = Base64.decodeBuffer(base64);
ByteArrayInputStream bais = new ByteArrayInputStream(bytes1);
return ImageIO.read(bais);
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
/**
* mat转换成bufferedImage
*
* @param matrix
* @return
*/
public static byte[] toByteArray(Mat matrix) {
MatOfByte mob = new MatOfByte();
Imgcodecs.imencode(".jpg", matrix, mob);
return mob.toArray();
}
/**
* mat转换成bufferedImage
*
* @param matrix
* @return
*/
public static BufferedImage toBufferedImage(Mat matrix) throws IOException {
byte[] buffer = toByteArray(matrix);
ByteArrayInputStream bais = new ByteArrayInputStream(buffer);
return ImageIO.read(bais);
}
/**
* base64转Mat
*
* @param base64
* @return
* @throws IOException
*/
public static Mat base642Mat(String base64) {
return bufImg2Mat(base64ToBufferedImage(base64), BufferedImage.TYPE_3BYTE_BGR, CvType.CV_8UC3);
}
/**
* BufferedImage转换成Mat
*
* @param original 要转换的BufferedImage
* @param imgType bufferedImage的类型 如 BufferedImage.TYPE_3BYTE_BGR
* @param matType 转换成mat的type 如 CvType.CV_8UC3
*/
public static Mat bufImg2Mat(BufferedImage original, int imgType, int matType) {
if (original == null) {
throw new IllegalArgumentException("original == null");
}
// Don't convert if it already has correct type
if (original.getType() != imgType) {
// Create a buffered image
BufferedImage image = new BufferedImage(original.getWidth(), original.getHeight(), imgType);
// Draw the image onto the new buffer
Graphics2D g = image.createGraphics();
try {
g.setComposite(AlphaComposite.Src);
g.drawImage(original, 0, 0, null);
original = image;
} catch (Exception e) {
e.printStackTrace();
} finally {
g.dispose();
}
}
byte[] pixels = ((DataBufferByte) original.getRaster().getDataBuffer()).getData();
Mat mat = Mat.eye(original.getHeight(), original.getWidth(), matType);
mat.put(0, 0, pixels);
return mat;
}
}
2. 测试controller源码
package com.demo.face;
import org.springframework.util.StringUtils;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
/**
* //TODO 描述
*
* @author: Ray.ma
* @Data: 2021-01-28 11:30
* @Version: 1.0.0
*/
@RestController
@RequestMapping
public class FaceController {
/**
* 获取图片中人脸个数
*
* @param base64Images base64图片,不包含逗号之前的部分
* @return
*/
@PostMapping("checkFaceNumber")
public int checkFaceNumber(String base64Images) {
return FaceUtils.checkFaceNo(base64Images);
}
/**
* 提取最大脸
*
* @param response
* @param base64Images
* @throws IOException
*/
@PostMapping("extractMaxFace")
public void extractMaxFace(HttpServletResponse response, String base64Images) throws IOException {
int indexOf = base64Images.indexOf(",");
if (indexOf > 0) {
base64Images = base64Images.substring(indexOf + 1);
}
response.setContentType("application/octet-stream");
PrintWriter writer = response.getWriter();
base64Images = FaceUtils.extractMaxFace(base64Images);
if (!StringUtils.hasLength(base64Images)) {
throw new RuntimeException("没有人脸");
}
writer.write("data:image/png;base64,".concat(base64Images));
writer.flush();
}
/**
* 标记脸
*
* @param response
* @param base64Images
* @throws IOException
*/
@PostMapping("markFace")
public void markFace(HttpServletResponse response, String base64Images) throws IOException {
int indexOf = base64Images.indexOf(",");
if (indexOf > 0) {
base64Images = base64Images.substring(indexOf + 1);
}
response.setContentType("application/octet-stream");
PrintWriter writer = response.getWriter();
base64Images = FaceUtils.markFace(base64Images);
writer.write("data:image/png;base64,".concat(base64Images));
writer.flush();
}
}
三、最终效果
标记脸的效果如下:
四、其他
绘制文字,绘制线
public static void main(String[] args) throws IOException {
Mat mat = Imgcodecs.imread("d:\\face\\1.jpg");
String path = System.getProperty("user.dir").concat("/haarcascades/haarcascade_frontalface_alt.xml");
CascadeClassifier faceDetector = new CascadeClassifier(path);
MatOfRect faceDetections = new MatOfRect();
faceDetector.detectMultiScale(mat, faceDetections);
if (faceDetections.toArray().length > 0) {
for (Rect rect : faceDetections.toList()) {
//绘制长方形
// Imgproc.rectangle(mat, new Point(rect.x, rect.y), new Point(rect.x + rect.width, rect.y + rect.height), new Scalar(0, 255, 0), 3);
//绘制文字
// Imgproc.putText(mat,"Face", new Point(rect.x, rect.y), 0,0.5,new Scalar(0, 255, 0),3);
//绘制线
Imgproc.line(mat, new Point(rect.x, rect.y), new Point(rect.x, rect.y + 20), new Scalar(0, 255, 0), 3);
Imgproc.line(mat, new Point(rect.x, rect.y), new Point(rect.x + 20, rect.y), new Scalar(0, 255, 0), 3);
Imgproc.line(mat, new Point(rect.x + rect.width, rect.y), new Point(rect.x + rect.width - 20, rect.y), new Scalar(0, 255, 0), 3);
Imgproc.line(mat, new Point(rect.x + rect.width, rect.y), new Point(rect.x + rect.width, rect.y + 20), new Scalar(0, 255, 0), 3);
Imgproc.line(mat, new Point(rect.x, rect.y + rect.height), new Point(rect.x, rect.y + rect.height - 20), new Scalar(0, 255, 0), 3);
Imgproc.line(mat, new Point(rect.x, rect.y + rect.height), new Point(rect.x + 20, rect.y + rect.height), new Scalar(0, 255, 0), 3);
Imgproc.line(mat, new Point(rect.x + rect.width, rect.y + rect.height), new Point(rect.x + rect.width, rect.y + rect.height - 20), new Scalar(0, 255, 0), 3);
Imgproc.line(mat, new Point(rect.x + rect.width, rect.y + rect.height), new Point(rect.x + rect.width - 20, rect.y + rect.height), new Scalar(0, 255, 0), 3);
}
}
Imgcodecs.imwrite("D:\\face\\1-new.jpg", mat);
}
下载源码 链接:https://pan.baidu.com/s/14Lfc_5dnuRdYup_67ub9uA 提取码: rj5r