opencv3機(jī)器學(xué)習(xí)之EM算法示例詳解
引言
不同于其它的機(jī)器學(xué)習(xí)模型,EM算法是一種非監(jiān)督的學(xué)習(xí)算法,它的輸入數(shù)據(jù)事先不需要進(jìn)行標(biāo)注。相反,該算法從給定的樣本集中,能計(jì)算出高斯混和參數(shù)的最大似然估計(jì)。也能得到每個(gè)樣本對(duì)應(yīng)的標(biāo)注值,類(lèi)似于kmeans聚類(lèi)(輸入樣本數(shù)據(jù),輸出樣本數(shù)據(jù)的標(biāo)注)。實(shí)際上,高斯混和模型GMM和kmeans都是EM算法的應(yīng)用。
在opencv3.0中,EM算法的函數(shù)是trainEM,函數(shù)原型為:
bool trainEM(InputArray samples, OutputArray logLikelihoods=noArray(),OutputArray labels=noArray(),OutputArray probs=noArray())
四個(gè)參數(shù):
samples: 輸入的樣本,一個(gè)單通道的矩陣。從這個(gè)樣本中,進(jìn)行高斯混和模型估計(jì)。
logLikelihoods: 可選項(xiàng),輸出一個(gè)矩陣,里面包含每個(gè)樣本的似然對(duì)數(shù)值。
labels: 可選項(xiàng),輸出每個(gè)樣本對(duì)應(yīng)的標(biāo)注。
probs: 可選項(xiàng),輸出一個(gè)矩陣,里面包含每個(gè)隱性變量的后驗(yàn)概率
這個(gè)函數(shù)沒(méi)有輸入?yún)?shù)的初始化值,是因?yàn)樗鼤?huì)自動(dòng)執(zhí)行kmeans算法,將kmeans算法得到的結(jié)果作為參數(shù)初始化。
這個(gè)trainEM函數(shù)實(shí)際把E步驟和M步驟都包含進(jìn)去了,我們也可以對(duì)兩個(gè)步驟分開(kāi)執(zhí)行,OPENCV3.0中也提供了分別執(zhí)行的函數(shù):
bool trainE(InputArray samples, InputArray means0,
InputArray covs0=noArray(),
InputArray weights0=noArray(),
OutputArray logLikelihoods=noArray(),
OutputArray labels=noArray(),
OutputArray probs=noArray())bool trainM(InputArray samples, InputArray probs0,
OutputArray logLikelihoods=noArray(),
OutputArray labels=noArray(),
OutputArray probs=noArray())trainEM函數(shù)的功能和kmeans差不多,都是實(shí)現(xiàn)自動(dòng)聚類(lèi),輸出每個(gè)樣本對(duì)應(yīng)的標(biāo)注值。但它比kmeans還多出一個(gè)功能,就是它還能起到訓(xùn)練分類(lèi)器的作用,用于后續(xù)新樣本的預(yù)測(cè)。
預(yù)測(cè)函數(shù)原型為:
Vec2d predict2(InputArray sample, OutputArray probs) const
sample: 待測(cè)樣本
probs : 和上面一樣,一個(gè)可選的輸出值,包含每個(gè)隱性變量的后驗(yàn)概率
返回一個(gè)Vec2d類(lèi)型的數(shù),包括兩個(gè)元素的double向量,第一個(gè)元素為樣本的似然對(duì)數(shù)值,第二個(gè)元素為最大可能混和分量的索引值。
在本文中,我們用兩個(gè)實(shí)例來(lái)學(xué)習(xí)opencv中的EM算法的應(yīng)用。
一、opencv3.0中自帶的例子
既包括聚類(lèi)trianEM,也包括預(yù)測(cè)predict2
代碼:
#include "stdafx.h"
#include "opencv2/opencv.hpp"
#include <iostream>
using namespace std;
using namespace cv;
using namespace cv::ml;
//使用EM算法實(shí)現(xiàn)樣本的聚類(lèi)及預(yù)測(cè)
int main()
{
const int N = 4; //分成4類(lèi)
const int N1 = (int)sqrt((double)N);
//定義四種顏色,每一類(lèi)用一種顏色表示
const Scalar colors[] =
{
Scalar(0, 0, 255), Scalar(0, 255, 0),
Scalar(0, 255, 255), Scalar(255, 255, 0)
};
int i, j;
int nsamples = 100; //100個(gè)樣本點(diǎn)
Mat samples(nsamples, 2, CV_32FC1); //樣本矩陣,100行2列,即100個(gè)坐標(biāo)點(diǎn)
Mat img = Mat::zeros(Size(500, 500), CV_8UC3); //待測(cè)數(shù)據(jù),每一個(gè)坐標(biāo)點(diǎn)為一個(gè)待測(cè)數(shù)據(jù)
samples = samples.reshape(2, 0);
//循環(huán)生成四個(gè)類(lèi)別樣本數(shù)據(jù),共樣本100個(gè),每類(lèi)樣本25個(gè)
for (i = 0; i < N; i++)
{
Mat samples_part = samples.rowRange(i*nsamples / N, (i + 1)*nsamples / N);
//設(shè)置均值
Scalar mean(((i%N1) + 1)*img.rows / (N1 + 1),
((i / N1) + 1)*img.rows / (N1 + 1));
//設(shè)置標(biāo)準(zhǔn)差
Scalar sigma(30, 30);
randn(samples_part, mean, sigma); //根據(jù)均值和標(biāo)準(zhǔn)差,隨機(jī)生成25個(gè)正態(tài)分布坐標(biāo)點(diǎn)作為樣本
}
samples = samples.reshape(1, 0);
// 訓(xùn)練分類(lèi)器
Mat labels; //標(biāo)注,不需要事先知道
Ptr<EM> em_model = EM::create();
em_model->setClustersNumber(N);
em_model->setCovarianceMatrixType(EM::COV_MAT_SPHERICAL);
em_model->setTermCriteria(TermCriteria(TermCriteria::COUNT + TermCriteria::EPS, 300, 0.1));
em_model->trainEM(samples, noArray(), labels, noArray());
//對(duì)每個(gè)坐標(biāo)點(diǎn)進(jìn)行分類(lèi),并根據(jù)類(lèi)別用不同的顏色畫(huà)出
Mat sample(1, 2, CV_32FC1);
for (i = 0; i < img.rows; i++)
{
for (j = 0; j < img.cols; j++)
{
sample.at<float>(0) = (float)j;
sample.at<float>(1) = (float)i;
//predict2返回的是double值,用cvRound進(jìn)行四舍五入得到整型
//此處返回的是兩個(gè)值Vec2d,取第二個(gè)值作為樣本標(biāo)注
int response = cvRound(em_model->predict2(sample, noArray())[1]);
Scalar c = colors[response]; //為不同類(lèi)別設(shè)定顏色
circle(img, Point(j, i), 1, c*0.75, FILLED);
}
}
//畫(huà)出樣本點(diǎn)
for (i = 0; i < nsamples; i++)
{
Point pt(cvRound(samples.at<float>(i, 0)), cvRound(samples.at<float>(i, 1)));
circle(img, pt, 2, colors[labels.at<int>(i)], FILLED);
}
imshow("EM聚類(lèi)結(jié)果", img);
waitKey(0);
return 0;
}結(jié)果:

二、trainEM實(shí)現(xiàn)自動(dòng)聚類(lèi)進(jìn)行圖片目標(biāo)檢測(cè)
只用trainEM實(shí)現(xiàn)自動(dòng)聚類(lèi)功能,進(jìn)行圖片中的目標(biāo)檢測(cè)
代碼:
#include "stdafx.h"
#include "opencv2/opencv.hpp"
#include <iostream>
using namespace std;
using namespace cv;
using namespace cv::ml;
int main()
{
const int MAX_CLUSTERS = 5;
Vec3b colorTab[] =
{
Vec3b(0, 0, 255),
Vec3b(0, 255, 0),
Vec3b(255, 100, 100),
Vec3b(255, 0, 255),
Vec3b(0, 255, 255)
};
Mat data, labels;
Mat pic = imread("d:/woman.png");
for (int i = 0; i < pic.rows; i++)
for (int j = 0; j < pic.cols; j++)
{
Vec3b point = pic.at<Vec3b>(i, j);
Mat tmp = (Mat_<float>(1, 3) << point[0], point[1], point[2]);
data.push_back(tmp);
}
int N =3; //聚成3類(lèi)
Ptr<EM> em_model = EM::create();
em_model->setClustersNumber(N);
em_model->setCovarianceMatrixType(EM::COV_MAT_SPHERICAL);
em_model->setTermCriteria(TermCriteria(TermCriteria::COUNT + TermCriteria::EPS, 300, 0.1));
em_model->trainEM(data, noArray(), labels, noArray());
int n = 0;
//顯示聚類(lèi)結(jié)果,不同的類(lèi)別用不同的顏色顯示
for (int i = 0; i < pic.rows; i++)
for (int j = 0; j < pic.cols; j++)
{
int clusterIdx = labels.at<int>(n);
pic.at<Vec3b>(i, j) = colorTab[clusterIdx];
n++;
}
imshow("pic", pic);
waitKey(0);
return 0;
}測(cè)試圖片

測(cè)試結(jié)果:

以上就是opencv3機(jī)器學(xué)習(xí)之EM算法的詳細(xì)內(nèi)容,更多關(guān)于opencv3 EM算法的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- Gradio機(jī)器學(xué)習(xí)模型快速部署工具接口狀態(tài)
- 通過(guò)gradio和攝像頭獲取照片和視頻實(shí)現(xiàn)過(guò)程
- Python機(jī)器學(xué)習(xí)性能度量利用鳶尾花數(shù)據(jù)繪制P-R曲線
- Python機(jī)器學(xué)習(xí)利用鳶尾花數(shù)據(jù)繪制ROC和AUC曲線
- python機(jī)器學(xué)習(xí)pytorch?張量基礎(chǔ)教程
- python機(jī)器學(xué)習(xí)pytorch自定義數(shù)據(jù)加載器
- 前端AI機(jī)器學(xué)習(xí)在瀏覽器中訓(xùn)練模型
- Gradio機(jī)器學(xué)習(xí)模型快速部署工具應(yīng)用分享
相關(guān)文章
C語(yǔ)言實(shí)現(xiàn)套接字操作的示例代碼
在C語(yǔ)言中,套接字(Socket)是一種進(jìn)行網(wǎng)絡(luò)通信的接口,這篇文章就來(lái)和大家介紹一下C語(yǔ)言中是如何進(jìn)行套接字操作的,需要的可以參考一下2023-06-06
利用C++11原子量如何實(shí)現(xiàn)自旋鎖詳解
當(dāng)自旋鎖嘗試獲取鎖時(shí)以忙等待(busy waiting)的形式不斷地循環(huán)檢查鎖是否可用,下面這篇文章主要給大家介紹了關(guān)于利用C++11原子量如何實(shí)現(xiàn)自旋鎖的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2018-06-06
C++ COM編程之QueryInterface函數(shù)(二)
這篇文章主要介紹了C++ COM編程之QueryInterface函數(shù)(二),本文是第二篇,第一篇請(qǐng)參閱相關(guān)文檔,需要的朋友可以參考下2014-10-10
C++類(lèi)的靜態(tài)成員變量與靜態(tài)成員函數(shù)詳解
下面小編就為大家?guī)?lái)一篇C++類(lèi)的靜態(tài)成員變量與靜態(tài)成員函數(shù)的文章。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2021-11-11
關(guān)于VS+QT5應(yīng)用程序換圖標(biāo)的解決方案
這篇文章主要介紹了VS+QT5應(yīng)用程序換圖標(biāo)的處理方案,本文給大家提供了兩種解決方案供大家參考,每種方法給大家講解的都非常詳細(xì),需要的朋友可以參考下2021-12-12
Qt?自定義屬性Q_PROPERTY不顯示float類(lèi)型的解決
這篇文章主要介紹了Qt?自定義屬性Q_PROPERTY不顯示float類(lèi)型的問(wèn)題及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-11-11
C++結(jié)構(gòu)體數(shù)組實(shí)現(xiàn)貪吃蛇
這篇文章主要為大家詳細(xì)介紹了C++結(jié)構(gòu)體數(shù)組實(shí)現(xiàn)貪吃蛇,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-03-03

