分类器 html
分类器是一种计算机程序。 算法
他的设计目标是在经过学习后,可自动将数据分到已知类别。 安全
平面线性分类器 机器学习
一个简单的分类问题,如图有一些圆圈和一些正方形,如何找一条最优的直线将他们分开? 函数
咱们能够找到不少种方法画出这条直线,但怎样的直线才是最优的呢? 学习
距离样本太近的直线不是最优的,由于这样的直线对噪声敏感度高,泛化性较差。 所以咱们的目标是找到一条直线,离最近的点距离最远。 测试
怎么寻找距离最远的直线?枚举全部直线,而后计算其样本最小距离?这样显然不是一个好办法,这将产生大量的计算开销。 优化
咱们利用另外一种方法,对直线的正负偏移量1,这样就产生了一个区域(下图的Maximum margin覆盖的区域),区域边界上的点到直线的距离是固定的,如今的问题是最近的点是否恰好在边界上或者在边界外。 ui
还记得点到线的公式么? spa
对于直线Ax+By+C=0,点(x0, y0)到直线的距离:
distance = |Ax0+By0+C| / (A2 + B2)1/2
那么区域边缘到直线的距离:
distance = (|Ax+By+C| + 1)/ (A2 + B2)1/2 = 1/ (A2 + B2)1/2。
并须要知足对于全部样本类别yi 知足:yi (Ax+By+C) > = 1,也就是全部样本都不在该区域之内。
因而咱们能够找到适当的A、B、C,从而获得:
Maximum margin = 2/ (A2 + B2)1/2。
超平面推广
同理,咱们将这必定理推广到任意维度。其超平面表达式为:
一维是线、二维是面、三维是体……四维呢?五维呢?好吧统称超平面吧……
其中 叫作 权重向量 ,
叫作 偏置向量。
用这种表达式来表达线Ax+By+C = 0的话,能够这么表示:
f(x) = (C, 0) + (A, B)T (x, y);
其中(C, 0) 是偏置向量 ,(A, B)是权重向量
。
因为最优超平面能够有不少种表达方式,咱们定义:
β0 + βTx = 0,
为最优超平面表达式。因而咱们能够获得他的Maximum margin区域边界表达式应该为:
咱们称在这边界上的点为:支持向量(Supper Vector)。
由于点到超平面距离公式为:
在边界上,即支持向量到超平面距离:
因此Maximum margin为两倍距离,即:
将M求倒数1/M 则可将求最大转换成求最小。因而有:
其中 表示样本的类别标记。
这是一个拉格朗日优化问题,能够经过拉格朗日乘数法获得最优超平面的权重向量 和偏置
。
什么是SVM
支持向量机 (SVM) 是一个类分类器,正式的定义是一个可以将不一样类样本在样本空间分隔的超平面。 换句话说,给定一些标记好的训练样本 (监督式学习),SVM算法输出一个最优化的分隔超平面。
1995年Cortes和Vapnik于首先提出SVM,它在解决小样本、非线性及高维模式识别中表现出许多特有的优点,并可以推广应用到函数拟合等其余机器学习问题中。
使用SVM
#include <opencv2/core/core.hpp> #include <opencv2/highgui/highgui.hpp> #include <opencv2/ml/ml.hpp> using namespace cv; int main() { // 用于保存可视化数据的矩阵 int width = 512, height = 512; Mat image = Mat::zeros(height, width, CV_8UC3); // 建立一些训练样本 float labels[4] = {1.0, -1.0, -1.0, -1.0}; Mat labelsMat(3, 1, CV_32FC1, labels); float trainingData[4][2] = { {501, 10}, {255, 10}, {501, 255}, {10, 501} }; Mat trainingDataMat(3, 2, CV_32FC1, trainingData); // 设置SVM参数 CvSVMParams params; params.svm_type = CvSVM::C_SVC; params.kernel_type = CvSVM::LINEAR; params.term_crit = cvTermCriteria(CV_TERMCRIT_ITER, 100, 1e-6); // 对SVM进行训练 CvSVM SVM; SVM.train(trainingDataMat, labelsMat, Mat(), Mat(), params); Vec3b green(0,255,0), blue (255,0,0); // 将SVM判定的分划区域绘制出来 for (int i = 0; i < image.rows; ++i) for (int j = 0; j < image.cols; ++j) { Mat sampleMat = (Mat_<float>(1,2) << i,j); float response = SVM.predict(sampleMat); if (response == 1) image.at<Vec3b>(j, i) = green; else if (response == -1) image.at<Vec3b>(j, i) = blue; } // 绘制训练数据点 int thickness = -1; int lineType = 8; circle( image, Point(501, 10), 5, Scalar( 0, 0, 0), thickness, lineType); circle( image, Point(255, 10), 5, Scalar(255, 255, 255), thickness, lineType); circle( image, Point(501, 255), 5, Scalar(255, 255, 255), thickness, lineType); circle( image, Point( 10, 501), 5, Scalar(255, 255, 255), thickness, lineType); // 绘制支持向量 thickness = 2; lineType = 8; int c = SVM.get_support_vector_count(); for (int i = 0; i < c; ++i) { const float* v = SVM.get_support_vector(i); circle( image, Point( (int) v[0], (int) v[1]), 6, Scalar(128, 128, 128), thickness, lineType); } imwrite("result.png", image); imshow("简单SVM分类", image); waitKey(0); }
创建训练样本
这里经过Mat构造函数,创建了一个简单的训练样本。
//创建一个标签矩阵 float labels[4] = {1.0, -1.0, -1.0, -1.0}; Mat labelsMat(3, 1, CV_32FC1, labels); //创建一个训练样本矩阵 float trainingData[4][2] = { {501, 10}, {255, 10}, {501, 255}, {10, 501} }; Mat trainingDataMat(3, 2, CV_32FC1, trainingData);
因为CvSVM::train 要求样本数据存储在float 类型的Mat中,因此创建了float类型的Mat样本。
设置SVM参数
struct CvSVMParams
SVM 训练参数结构。
该结构必须被初始化后,传给CvSVM。
CvSVMParams::CvSVMParams
构造函数
C++: CvSVMParams::CvSVMParams() C++: CvSVMParams::CvSVMParams(int svm_type, int kernel_type, double degree, double gamma, double coef0, double Cvalue, double nu, double p, CvMat* class_weights, CvTermCriteria term_crit)
参数
- svm_type –
指定SVM的类型,下面是可能的取值:
- CvSVM::C_SVC C类支持向量分类机。n类分组 (n
2),容许用异常值惩罚因子C进行不彻底分类。
- CvSVM::NU_SVC
类支持向量分类机。n相似然不彻底分类的分类器。参数为
取代C(其值在区间【0,1】中,nu越大,决策边界越平滑)。
- CvSVM::ONE_CLASS 单分类器,全部的训练数据提取自同一个类里,而后SVM创建了一个分界线以分割该类在特征空间中所占区域和其它类在特征空间中所占区域。
- CvSVM::EPS_SVR
类支持向量回归机。训练集中的特征向量和拟合出来的超平面的距离须要小于p。异常值惩罚因子C被采用。
- CvSVM::NU_SVR
类支持向量回归机。
代替了 p。
可从 [LibSVM] 获取更多细节。
- kernel_type –
SVM的内核类型,下面是可能的取值:
- CvSVM::LINEAR 线性内核。没有任何向映射至高维空间,线性区分(或回归)在原始特征空间中被完成,这是最快的选择。
.
- CvSVM::POLY 多项式内核:
.
- CvSVM::RBF 基于径向的函数,对于大多数状况都是一个较好的选择:
.
- CvSVM::SIGMOID Sigmoid函数内核:
.
- degree – 内核函数(POLY)的参数degree。
- gamma – 内核函数(POLY/ RBF/ SIGMOID)的参数
。
- coef0 – 内核函数(POLY/ SIGMOID)的参数coef0。
- Cvalue – SVM类型(C_SVC/ EPS_SVR/ NU_SVR)的参数C。
- nu – SVM类型(NU_SVC/ ONE_CLASS/ NU_SVR)的参数
。
- p – SVM类型(EPS_SVR)的参数
。
- class_weights – C_SVC中的可选权重,赋给指定的类,乘以C之后变成
。因此这些权重影响不一样类别的错误分类惩罚项。权重越大,某一类别的误分类数据的惩罚项就越大。
- term_crit – SVM的迭代训练过程的停止条件,解决部分受约束二次最优问题。您能够指定的公差和/或最大迭代次数。
默认的构造函数初始化有如下值:
CvSVMParams::CvSVMParams() : svm_type(CvSVM::C_SVC), kernel_type(CvSVM::RBF), degree(0), gamma(1), coef0(0), C(1), nu(0), p(0), class_weights(0) { term_crit = cvTermCriteria( CV_TERMCRIT_ITER+CV_TERMCRIT_EPS, 1000, FLT_EPSILON ); }
OpenCV的SVM
class CvSVM
向量支持机
CvSVM::CvSVM
训练构造函数。
C++: CvSVM::CvSVM() C++: CvSVM::CvSVM(const Mat& trainData, const Mat& responses, const Mat& varIdx=Mat(), const Mat& sampleIdx=Mat(), CvSVMParams params=CvSVMParams() ) C++: CvSVM::CvSVM(const CvMat* trainData, const CvMat* responses, const CvMat* varIdx=0, const CvMat* sampleIdx=0, CvSVMParams params=CvSVMParams() )
参数
- trainData — 训练数据,必须是CV_32FC1 (32位浮点类型,单通道)。数据必须是CV_ROW_SAMPLE的,即特征向量以行来存储。
- responses — 响应数据,一般是1D向量存储在CV_32SC1 (仅仅用在分类问题上)或者CV_32FC1格式。
- varIdx — 指定感兴趣的特征。能够是整数(32sC1)向量,例如以0为开始的索引,或者8位(8uC1)的使用的特征或者样本的掩码。用户也能够传入NULL指针,用来表示训练中使用全部变量/样本。
- sampleIdx — 指定感兴趣的样本。描述同上。
- params — SVM参数。
CvSVM::train
训练一个SVM。
C++: bool CvSVM::train(const Mat& trainData, const Mat& responses, const Mat& varIdx=Mat(), const Mat& sampleIdx=Mat(), CvSVMParams params=CvSVMParams() ) C++: bool CvSVM::train(const CvMat* trainData, const CvMat* responses, const CvMat* varIdx=0, const CvMat* sampleIdx=0, CvSVMParams params=CvSVMParams() )参数参考构造函数。
CvSVM::train_auto
根据可选参数训练一个SVM。
C++: bool CvSVM::train_auto(const Mat& trainData, const Mat& responses, const Mat& varIdx, const Mat& sampleIdx, CvSVMParams params, int k_fold=10, CvParamGrid Cgrid=CvSVM::get_default_grid(CvSVM::C), CvParamGrid gammaGrid=CvSVM::get_default_grid(CvSVM::GAMMA), CvParamGrid pGrid=CvSVM::get_default_grid(CvSVM::P), CvParamGrid nuGrid=CvSVM::get_default_grid(CvSVM::NU), CvParamGrid coeffGrid=CvSVM::get_default_grid(CvSVM::COEF), CvParamGrid degreeGrid=CvSVM::get_default_grid(CvSVM::DEGREE), bool balanced=false) C++: bool CvSVM::train_auto(const CvMat* trainData, const CvMat* responses, const CvMat* varIdx, const CvMat* sampleIdx, CvSVMParams params, int kfold=10, CvParamGrid Cgrid=get_default_grid(CvSVM::C), CvParamGrid gammaGrid=get_default_grid(CvSVM::GAMMA), CvParamGrid pGrid=get_default_grid(CvSVM::P), CvParamGrid nuGrid=get_default_grid(CvSVM::NU), CvParamGrid coeffGrid=get_default_grid(CvSVM::COEF), CvParamGrid degreeGrid=get_default_grid(CvSVM::DEGREE), bool balanced=false )
参数
- k_fold – 交叉验证参数。训练集被分红k_fold的自子集。其中一个子集是用来测试模型,其余子集则成为训练集。因此,SVM算法复杂度是执行k_fold的次数。
- *Grid – 对应的SVM迭代网格参数。
- balanced – 若是是true则这是一个2类分类问题。这将会建立更多的平衡交叉验证子集。
这个方法根据CvSVMParams中的最佳参数C, gamma, p, nu, coef0, degree自动训练SVM模型。参数被认为是最佳的交叉验证,其测试集预估错误最小。
若是没有须要优化的参数,相应的网格步骤应该被设置为小于或等于1的值。例如,为了不gamma的优化,设置gamma_grid.step = 0,gamma_grid.min_val, gamma_grid.max_val 为任意数值。因此params.gamma 由gamma得出。
最后,若是参数优化是必需的,可是相应的网格却不肯定,你可能须要调用函数CvSVM::get_default_grid(),建立一个网格。例如,对于gamma,调用CvSVM::get_default_grid(CvSVM::GAMMA)。
该函数为分类运行 (params.svm_type=CvSVM::C_SVC 或者 params.svm_type=CvSVM::NU_SVC) 和为回归运行 (params.svm_type=CvSVM::EPS_SVR 或者 params.svm_type=CvSVM::NU_SVR)效果同样好。若是params.svm_type=CvSVM::ONE_CLASS,没有优化,并指定执行通常的SVM。
CvSVM::predict
预测样本的相应数据。
C++: float CvSVM::predict(const Mat& sample, bool returnDFVal=false ) const
C++: float CvSVM::predict(const CvMat* sample, bool returnDFVal=false ) const C++: float CvSVM::predict(const CvMat* samples, CvMat* results) const
参数
- sample – 须要预测的输入样本。
- samples – 须要预测的输入样本们。
- returnDFVal – 指定返回值类型。若是值是true,则是一个2类分类问题,该方法返回的决策函数值是边缘的符号距离。
- results – 相应的样本输出预测的响应。
这个函数用来预测一个新样本的响应数据(response)。在分类问题中,这个函数返回类别编号;在回归问题中,返回函数值。输入的样本必须与传给trainData的训练样本一样大小。若是训练中使用了varIdx参数,必定记住在predict函数中使用跟训练特征一致的特征。
后缀const是说预测不会影响模型的内部状态,因此这个函数能够很安全地从不一样的线程调用。
CvSVM::get_default_grid
生成一个SVM网格参数。
C++: CvParamGrid CvSVM::get_default_grid(int param_id)
参数
- param_id –
SVM参数的IDs必须是下列中的一个:
- CvSVM::C
- CvSVM::GAMMA
- CvSVM::P
- CvSVM::NU
- CvSVM::COEF
- CvSVM::DEGREE
网格参数将根据这个ID生成。
CvSVM::get_params
返回当前SVM的参数。
C++: CvSVMParams CvSVM::get_params() const这个函数主要是在使用CvSVM::train_auto()时去得到最佳参数。
CvSVM::get_support_vector
检索必定数量的支持向量和特定的向量。
C++: int CvSVM::get_support_vector_count() const C++: const float* CvSVM::get_support_vector(int i) const
参数 i – 指定支持向量的索引。 该方法能够用于检索一组支持向量。
CvSVM::get_var_count
返回变量的个数。
C++: int CvSVM::get_var_count() const
分割结果
OpenCV的SVM是基于台湾大学林智仁开发的LIBSVM开发包的。若是你还不过瘾能够看看下面林智仁的演示程序(须要JAVA支持):
http://www.csie.ntu.edu.tw/~cjlin/libsvm/
在这个实验中,咱们成功让机器找到了区分样品的线性划分,并将其支持向量显示出来。
被山寨的原文
Introduction to Support Vector Machines . OpenCV.org
Support Vector Machines API . OpenCV.org