【opencv】LBP(局部二进制模式)算法的实现

2024-02-13 10:58

本文主要是介绍【opencv】LBP(局部二进制模式)算法的实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本章我们学习LBP图像的原理和使用,因为接下来教程我们要使用LBP图像的直方图来进行脸部识别。

参考资料:

http://docs.opencv.org/modules/contrib/doc/facerec/facerec_tutorial.html      (非常重要的参考文档!!!)

http://www.cnblogs.com/mikewolf2002/p/3438166.html

      LBP的基本思想是以图像中某个像素为中心,对相邻像素进行阈值比较。如果中心像素的亮度大于等于它的相邻像素,把相邻像素标记为1,否则标记为0。我们可以用二进制数字来表示LBP图中的每个像素的LBP编码,比如下图中的中心像素,它的LBP编码为:00010011,其十进制值为19。

image

用公式表示就是:

image

其中(xc,yc)是中心像素,ic是灰度值,in是相邻像素的灰度值,s是一个符号函数:

image

在OpenCV的LBP算法中,固定的领域大小不能对不同规模的细节进行编码。所以基本的LBP算法被进一步推广为使用不同大小和形状的领域,采取圆形的领域并结合双线性插值运算,我们可以获得任意半径和任意数目的领域像素点。使用圆形的LBP算子:

对于一个点image, 它的近邻点 image用以下公式计算:

image

其中R是半径,p是样本点的个数。

如果就算的结果不在像素坐标上,我们则使用双线性插值(确定他的值)进行近似处理。

image

下面的代码中,我们分别实现了通常LBP图和圆形算子LBP图。

      elbp是圆形算子LBP函数,elbp1是通常LBP图,我们分别对lena的图像进行了处理,结果如下所示,从途中可以看出来,使用圆形算子的效果锐度更强。

#include "opencv2/core/core.hpp"
#include "opencv2/contrib/contrib.hpp"
#include "opencv2/highgui/highgui.hpp"#include <iostream>
#include <fstream>
#include <sstream>using namespace cv;
using namespace std;void elbp(Mat& src, Mat &dst, int radius, int neighbors){for(int n=0; n<neighbors; n++){// 采样点的计算float x = static_cast<float>(-radius * sin(2.0*CV_PI*n/static_cast<float>(neighbors)));float y = static_cast<float>(radius * cos(2.0*CV_PI*n/static_cast<float>(neighbors)));// 上取整和下取整的值int fx = static_cast<int>(floor(x));int fy = static_cast<int>(floor(y));int cx = static_cast<int>(ceil(x));int cy = static_cast<int>(ceil(y));// 小数部分float ty = y - fy;float tx = x - fx;// 设置插值权重float w1 = (1 - tx) * (1 - ty);float w2 =      tx  * (1 - ty);float w3 = (1 - tx) *      ty;float w4 =      tx  *      ty;// 循环处理图像数据for(int i=radius; i < src.rows-radius;i++){for(int j=radius;j < src.cols-radius;j++) {// 计算插值float t = static_cast<float>(w1*src.at<uchar>(i+fy,j+fx) + w2*src.at<uchar>(i+fy,j+cx) + w3*src.at<uchar>(i+cy,j+fx) + w4*src.at<uchar>(i+cy,j+cx));// 进行编码dst.at<uchar>(i-radius,j-radius) += ((t > src.at<uchar>(i,j)) || (std::abs(t-src.at<uchar>(i,j)) < std::numeric_limits<float>::epsilon())) << n;}}}}void elbp1(Mat& src, Mat &dst){// 循环处理图像数据for(int i=1; i < src.rows-1;i++){for(int j=1;j < src.cols-1;j++) {uchar tt = 0;int tt1 = 0;uchar u = src.at<uchar>(i,j);if(src.at<uchar>(i-1,j-1)>u) { tt += 1 <<tt1; } tt1++;if(src.at<uchar>(i-1,j)>u) { tt += 1 <<tt1; } tt1++;if(src.at<uchar>(i-1,j+1)>u) { tt += 1 <<tt1; } tt1++;if(src.at<uchar>(i,j+1)>u) { tt += 1 <<tt1; } tt1++;if(src.at<uchar>(i+1,j+1)>u) { tt += 1 <<tt1; } tt1++;if(src.at<uchar>(i+1,j)>u) { tt += 1 <<tt1; } tt1++;if(src.at<uchar>(i+1,j-1)>u) { tt += 1 <<tt1; } tt1++;if(src.at<uchar>(i-1,j)>u) { tt += 1 <<tt1; } tt1++;dst.at<uchar>(i-1,j-1) = tt;}}}int main(){Mat img = cv::imread("../lenna.jpg", 0);namedWindow("image");imshow("image", img);int radius, neighbors;radius = 1;neighbors = 8;//创建一个LBP//注意为了溢出,我们行列都在原有图像上减去2个半径Mat dst = Mat(img.rows-2*radius, img.cols-2*radius,CV_8UC1, Scalar(0));elbp1(img,dst);namedWindow("normal");imshow("normal", dst);Mat dst1 = Mat(img.rows-2*radius, img.cols-2*radius,CV_8UC1, Scalar(0));elbp(img,dst1,1,8);namedWindow("circle");imshow("circle", dst1);while(1)cv::waitKey(0);}

imageimageimage

我们换另外一张图,该图包括不同光照下的四副照片,再来看看LBP图的效果(可以看到,LBP在光照不均匀的人脸识别中可以取得很好的应用!):

image

image

image

这篇关于【opencv】LBP(局部二进制模式)算法的实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/705351

相关文章

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

Redis客户端连接机制的实现方案

《Redis客户端连接机制的实现方案》本文主要介绍了Redis客户端连接机制的实现方案,包括事件驱动模型、非阻塞I/O处理、连接池应用及配置优化,具有一定的参考价值,感兴趣的可以了解一下... 目录1. Redis连接模型概述2. 连接建立过程详解2.1 连php接初始化流程2.2 关键配置参数3. 最大连

Python实现网格交易策略的过程

《Python实现网格交易策略的过程》本文讲解Python网格交易策略,利用ccxt获取加密货币数据及backtrader回测,通过设定网格节点,低买高卖获利,适合震荡行情,下面跟我一起看看我们的第一... 网格交易是一种经典的量化交易策略,其核心思想是在价格上下预设多个“网格”,当价格触发特定网格时执行买

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统

Python对接支付宝支付之使用AliPay实现的详细操作指南

《Python对接支付宝支付之使用AliPay实现的详细操作指南》支付宝没有提供PythonSDK,但是强大的github就有提供python-alipay-sdk,封装里很多复杂操作,使用这个我们就... 目录一、引言二、准备工作2.1 支付宝开放平台入驻与应用创建2.2 密钥生成与配置2.3 安装ali

Spring Security 单点登录与自动登录机制的实现原理

《SpringSecurity单点登录与自动登录机制的实现原理》本文探讨SpringSecurity实现单点登录(SSO)与自动登录机制,涵盖JWT跨系统认证、RememberMe持久化Token... 目录一、核心概念解析1.1 单点登录(SSO)1.2 自动登录(Remember Me)二、代码分析三、

PyCharm中配置PyQt的实现步骤

《PyCharm中配置PyQt的实现步骤》PyCharm是JetBrains推出的一款强大的PythonIDE,结合PyQt可以进行pythion高效开发桌面GUI应用程序,本文就来介绍一下PyCha... 目录1. 安装China编程PyQt1.PyQt 核心组件2. 基础 PyQt 应用程序结构3. 使用 Q

OpenCV在Java中的完整集成指南分享

《OpenCV在Java中的完整集成指南分享》本文详解了在Java中集成OpenCV的方法,涵盖jar包导入、dll配置、JNI路径设置及跨平台兼容性处理,提供了图像处理、特征检测、实时视频分析等应用... 目录1. OpenCV简介与应用领域1.1 OpenCV的诞生与发展1.2 OpenCV的应用领域2

Python实现批量提取BLF文件时间戳

《Python实现批量提取BLF文件时间戳》BLF(BinaryLoggingFormat)作为Vector公司推出的CAN总线数据记录格式,被广泛用于存储车辆通信数据,本文将使用Python轻松提取... 目录一、为什么需要批量处理 BLF 文件二、核心代码解析:从文件遍历到数据导出1. 环境准备与依赖库

在Java中使用OpenCV实践

《在Java中使用OpenCV实践》用户分享了在Java项目中集成OpenCV4.10.0的实践经验,涵盖库简介、Windows安装、依赖配置及灰度图测试,强调其在图像处理领域的多功能性,并计划后续探... 目录前言一 、OpenCV1.简介2.下载与安装3.目录说明二、在Java项目中使用三 、测试1.测