Java+OpenCV實(shí)現(xiàn)圖片中的人臉識別
經(jīng)過前三個教程,我們可以知道了OpenCV的基本使用了。
今天,我們就要講OpenCV中認(rèn)出,這是一個人臉是怎么做的。
MatOfRect.detectMultiScale函數(shù)
OpenCV用的是detectMultiScale來認(rèn)出這是一個臉的。記得,這只是認(rèn)出這是一個臉,而不是這個臉是誰。
這個臉是誰我們會逐步展開,前面勿求夯實(shí)基礎(chǔ)。
detectMultiScale需要兩個參數(shù)(Mat src, MatOfRect objDetections);
- 第一個函數(shù),是傳入的圖片,帶有人臉的圖片;
- 第二個函數(shù),是把所有的這個圖片里的人臉得到并輸出到MatOfRect對象里;
實(shí)現(xiàn)代碼
ImageViewer.java
再上一遍
package org.mk.opencv;
import org.mk.opencv.util.OpenCVUtil;
import org.opencv.core.Mat;
import javax.swing.*;
import java.awt.*;
public class ImageViewer {
private JLabel imageView;
private Mat image;
private String windowName;
private JFrame frame = null;
public ImageViewer() {
frame = createJFrame(windowName, 800, 600);
}
public ImageViewer(Mat image) {
this.image = image;
}
/**
* @param image 要顯示的mat
* @param windowName 窗口標(biāo)題
*/
public ImageViewer(Mat image, String windowName) {
frame = createJFrame(windowName, 1024, 768);
this.image = image;
this.windowName = windowName;
}
public void setTitle(String windowName) {
this.windowName = windowName;
}
public void setImage(Mat image) {
this.image = image;
}
/**
* 圖片顯示
*/
public void imshow() {
setSystemLookAndFeel();
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);// 用戶點(diǎn)擊窗口關(guān)閉
if (image != null) {
Image loadedImage = OpenCVUtil.matToImage(image);
// JFrame frame = createJFrame(windowName, image.width(), image.height());
imageView.setIcon(new ImageIcon(loadedImage));
frame.pack();
// frame.setLocationRelativeTo(null);
// frame.setVisible(true);
// frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);// 用戶點(diǎn)擊窗口關(guān)閉
}
}
private void setSystemLookAndFeel() {
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (UnsupportedLookAndFeelException e) {
e.printStackTrace();
}
}
private JFrame createJFrame(String windowName, int width, int height) {
JFrame frame = new JFrame(windowName);
imageView = new JLabel();
final JScrollPane imageScrollPane = new JScrollPane(imageView);
imageScrollPane.setPreferredSize(new Dimension(width, height));
frame.add(imageScrollPane, BorderLayout.CENTER);
frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
return frame;
}
}
DetectFace.java
這個是主類。
老三樣:
1.加載opencv_java343.dll;
2.加載人臉分揀器;
3.創(chuàng)建Mat對象;
然后我們開始把臉識別出來:
1.使用detectMultiScale把傳入的Mat對象中含有臉的那些全部識別出來;
2.識別出來后我們可以使用for (Rect rect : objDetections.toArray())把所有的臉枚舉出來;
3.使用Imgproc.rectangle在每個識別出來的臉上用“綠”色把它們一個個框出來;
4.使用ImageViewer的.imgShow顯示標(biāo)識出來的臉;
package org.mk.opencv;
import org.opencv.core.Core;
import org.opencv.core.Mat;
import org.opencv.core.MatOfRect;
import org.opencv.core.Point;
import org.opencv.core.Rect;
import org.opencv.core.Scalar;
import org.opencv.imgcodecs.Imgcodecs;
import org.opencv.imgproc.Imgproc;
import org.opencv.objdetect.CascadeClassifier;
public class DetectFace {
public static void main(String[] args) {
System.loadLibrary(Core.NATIVE_LIBRARY_NAME);
//Mat src = Imgcodecs.imread("/Users/chrishu123126.com/opt/img/detect-face-4.jpg");
Mat src = Imgcodecs.imread("D:\\opencv-demo\\green-arrow.jpg");
if (src.empty()) {
System.out.println("圖片路徑不正確");
return;
}
Mat dst = dobj(src);
ImageViewer imageViewer = new ImageViewer(dst, "識臉");
imageViewer.imshow();
}
private static Mat dobj(Mat src) {
Mat dst = src.clone();
CascadeClassifier objDetector = new CascadeClassifier(
"D:\\opencvinstall\\build\\install\\etc\\lbpcascades\\lbpcascade_frontalface.xml");
MatOfRect objDetections = new MatOfRect();
objDetector.detectMultiScale(dst, objDetections);
if (objDetections.toArray().length <= 0) {
return src;
}
for (Rect rect : objDetections.toArray()) {
Imgproc.rectangle(dst, new Point(rect.x, rect.y), new Point(rect.x + rect.width, rect.y + rect.width),
new Scalar(0, 255, 0), 1); //new Scalar(0, 255, 0), 1)綠 //new Scalar(0, 0, 255), 1)紅 //new Scalar(255, 0, 0), 1)藍(lán)
}
return dst;
}
}
把識別出來的臉存成文件
我們現(xiàn)在把識別出來的5張臉存成5個jpg圖片。
制作一個寫盤函數(shù),很簡單。
private static void outputFace(String outputDir, Mat face) {
long millSecs = System.currentTimeMillis();
int temp = (int) (Math.random() * 10000);
StringBuffer outputImgName = new StringBuffer();
outputImgName.append(outputDir).append("/").append(millSecs).append(temp).append(".jpg");
if (face != null) {
Imgcodecs.imwrite(outputImgName.toString(), face);
logger.info(">>>>>>write image into->" + outputDir);
}
}
然后我們在我們的原來的代碼中加入這個函數(shù)
package org.mk.opencv;
import org.apache.log4j.Logger;
import org.mk.opencv.face.FaceRecogFromFiles;
import org.opencv.core.Core;
import org.opencv.core.Mat;
import org.opencv.core.MatOfRect;
import org.opencv.core.Point;
import org.opencv.core.Rect;
import org.opencv.core.Scalar;
import org.opencv.imgcodecs.Imgcodecs;
import org.opencv.imgproc.Imgproc;
import org.opencv.objdetect.CascadeClassifier;
public class DetectFace {
private static Logger logger = Logger.getLogger(DetectFace.class);
private final static String faceOutPutDir = "d://opencv-demo/face";
public static void main(String[] args) {
System.loadLibrary(Core.NATIVE_LIBRARY_NAME);
// Mat src =
// Imgcodecs.imread("/Users/chrishu123126.com/opt/img/detect-face-4.jpg");
Mat src = Imgcodecs.imread("D:\\opencv-demo\\green-arrow.jpg");
if (src.empty()) {
System.out.println("圖片路徑不正確");
return;
}
Mat dst = dobj(src);
ImageViewer imageViewer = new ImageViewer(dst, "識臉");
imageViewer.imshow();
}
private static Mat dobj(Mat src) {
Mat dst = src.clone();
CascadeClassifier objDetector = new CascadeClassifier(
"D:\\opencvinstall\\build\\install\\etc\\lbpcascades\\lbpcascade_frontalface.xml");
MatOfRect objDetections = new MatOfRect();
objDetector.detectMultiScale(dst, objDetections);
if (objDetections.toArray().length <= 0) {
return src;
}
for (Rect rect : objDetections.toArray()) {
Imgproc.rectangle(dst, new Point(rect.x, rect.y), new Point(rect.x + rect.width, rect.y + rect.width),
new Scalar(0, 255, 0), 1); // new Scalar(0, 255, 0), 1)綠 //new Scalar(0, 0, 255), 1)紅 //new
// Scalar(255, 0, 0), 1)藍(lán)
outputFace(faceOutPutDir, src.submat(rect));
}
return dst;
}
private static void outputFace(String outputDir, Mat face) {
long millSecs = System.currentTimeMillis();
int temp = (int) (Math.random() * 10000);
StringBuffer outputImgName = new StringBuffer();
outputImgName.append(outputDir).append("/").append(millSecs).append(temp).append(".jpg");
if (face != null) {
Imgcodecs.imwrite(outputImgName.toString(), face);
logger.info(">>>>>>write image into->" + outputDir);
}
}
}
運(yùn)行DetectFace.java
到此這篇關(guān)于Java+OpenCV實(shí)現(xiàn)圖片中的人臉識別的文章就介紹到這了,更多相關(guān)Java OpenCV人臉識別內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Intellij Idea中批量導(dǎo)入第三方j(luò)ar包的全過程
引入jar包一般都是針對小的java項(xiàng)目,這篇文章主要給大家介紹了關(guān)于Intellij Idea中批量導(dǎo)入第三方j(luò)ar包的相關(guān)資料,文中通過圖文介紹的非常詳細(xì),需要的朋友可以參考下2021-10-10
Java實(shí)現(xiàn)對稱加密DES和AES的示例代碼
這篇文章主要介紹了如何使用Java實(shí)現(xiàn)采用對稱密碼算法的應(yīng)用軟件,所用算法包括DES算法和AES算法,文中的示例代碼講解詳細(xì),感興趣的可以了解一下2023-04-04
Mybatis結(jié)果集映射與生命周期詳細(xì)介紹
結(jié)果集映射指的是將數(shù)據(jù)表中的字段與實(shí)體類中的屬性關(guān)聯(lián)起來,這樣 MyBatis 就可以根據(jù)查詢到的數(shù)據(jù)來填充實(shí)體對象的屬性,幫助我們完成賦值操作2022-10-10
Spring Security實(shí)現(xiàn)退出登錄和退出處理器
本文主要介紹了Spring Security實(shí)現(xiàn)退出登錄和退出處理器,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05
Java Spring Security認(rèn)證與授權(quán)及注銷和權(quán)限控制篇綜合解析
Spring Security 是 Spring 家族中的一個安全管理框架,實(shí)際上,在 Spring Boot 出現(xiàn)之前,Spring Security 就已經(jīng)發(fā)展了多年了,但是使用的并不多,安全管理這個領(lǐng)域,一直是 Shiro 的天下2021-10-10
mybatis-plus雪花算法自動生成機(jī)器id原理及源碼
Mybatis-Plus是一個Mybatis的增強(qiáng)工具,它在Mybatis的基礎(chǔ)上做了增強(qiáng),卻不做改變,Mybatis-Plus是為簡化開發(fā)、提高開發(fā)效率而生,但它也提供了一些很有意思的插件,比如SQL性能監(jiān)控、樂觀鎖、執(zhí)行分析等,下面一起看看mybatis-plus雪花算法自動生成機(jī)器id原理解析2021-06-06
java反射校驗(yàn)參數(shù)是否是基礎(chǔ)類型步驟示例
這篇文章主要為大家介紹了java反射校驗(yàn)參數(shù)是否是基礎(chǔ)類型步驟示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12
springboot登陸頁面圖片驗(yàn)證碼簡單的web項(xiàng)目實(shí)現(xiàn)
這篇文章主要介紹了springboot登陸頁面圖片驗(yàn)證碼簡單的web項(xiàng)目實(shí)現(xiàn),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-04-04

