SIMD——请不要妄图优化opencv的函数

2023-11-21 15:10

本文主要是介绍SIMD——请不要妄图优化opencv的函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

大纲

  • ORB-SLAM特征提取之SIMD优化
    • 一、优化手段
      • 1、NEON
      • 2、SSE
    • 二、cv::gaussianblur函数优化
    • 三、cv::FAST函数优化
    • 四、原因分析

ORB-SLAM特征提取之SIMD优化

一、优化手段

1、NEON

NEON是基于ARM架构的一种128位的SIMD(Single Instruction, Multiple Data,单指令、多数据)的拓展结构,具体的细节在这里不做过多的介绍,我们来讨论NEON在实际工程中的两种使用方式:

(1)汇编

相信学习过单片机的朋友对汇编语言一定都是又恨又怕,复杂的指令与奇怪的逻辑令人望而却步,我们来看一小段基于汇编的NEON指令:

 "vld1.8       {d8,d9}, [%[in0]]\n\t""add          %[in0], %[step]\n\t""vld1.8       {d10,d11}, [%[in1]]\n\t""add          %[in1], %[step]\n\t""vld1.8       {d12,d13}, [%[in0]]\n\t""add          %[in0], %[step]\n\t""vld1.8       {d14,d15}, [%[in1]]\n\t"

什么东西?这是什么东西!说实话,我也看不太懂,所以基于汇编的NEON指令难度大,可移植性差,但是由于汇编语言的特性,决定了它具有很高的效率。Anyway,我们来看下一种方式。

(2)使用arm提供的Intrinsics函数
可以认为是内联函数,但是在编译时编译器会将函数转化为neon指令。调用该函数需要包含头文件arm_neon.h,该头文件包含了neon各种操作函数。当然,我在这里同样来展示一段代码:

 for(; j < img.cols - 16 - 3; j += 16, ptr += 16){uint8x16_t m0,m1;uint8x16_t v0 = vld1q_u8((const uint8_t*)ptr);uint8x16_t v1 = vqsubq_u8(v0,t);v0 = vqaddq_u8(v0,t);

了解过NEON的同学这段代码应该不难看懂,就是简单的加减法,我们可以看到,这种内联方式的NEON代码指令难度低,可移植行也就比汇编形式的高很多。

2、SSE

SSE是Intel x86架构CPU的SIMD指令的简称,与NEON一样具有汇编和内联函数两种形式。

二、cv::gaussianblur函数优化

由于本次实验的电脑是intel的CPU,所以采用SSE优化的方式对高斯模糊函数进行优化,在优化之前,记录了cv::gaussianblur函数的效率如下图所示:在这里插入图片描述9.41ms,在我的认知领域内,应该不是具有很高效率的时常,所以我对该函数进行了SSE优化,对gaussainblur函数的源码进行了修改,执行时间如下:在这里插入图片描述惊人的148ms。当然,如果你需要的源码,可以私信我,我发给你自己run一下。总的来说,这并不是一次非常愉快的优化过程,同样的,接下来的SSE优化也令我大跌眼镜。

三、cv::FAST函数优化

FAST提取图像中的特征点是ORB-SLAM系列中使用的一种方法,具体原理不再赘述,因为这毕竟不是一篇介绍SLAM算法的blog,我们来看opencv实现的FAST函数耗时如何:在这里插入图片描述
结果是2.65ms,不得不说opencv优化的已经很好了。注意!!!这里使用了和ORBSLAM2中相同的阈值:20。在ORBSLAM2中该函数在网格的循环遍历中被调用,ORBSLAM2将图像分为30x30个网格,在每个网格中进行FAST角点检测,经过测量,每个网格中耗时约为0.02ms,为什么差异这么大呢,首先于ORBSLAM中独特的网格划分有关系,其次与图像的大小也有关系。接下来,我们来看我进行了SSE优化的代码,相关代码已经有NEON实现,你需要做的只是将NEON转化为SSE即可:在这里插入图片描述50ms。。唉,当我看到这个结果的时候,心里有一种说不出的酸楚…

四、原因分析

实验失败不可怕,可怕的是不分析为什么失败,经过资料的查询与文献的阅读我大概的掌握了其中的奥义。

(1)OpenCV的函数自带SIMD加速,无论是gaussainblur还是FAST都自带了SMID的并行加速,并且OpenCV代码结构和顺序更加合理,能够更好的发挥CPU的全部性能
(2)OpenCV的优化是按照运行环境中CPU的架构来决定的,我们知道到INTEL的SIMD指令集avx512是高于SSE的,由于我运行的环境中支持avx512,所以使用该指令集的优化代码效率必然高于SSE。
(3)最终的建议是:别动OpenCV的源码为好,除非用汇编…,这得是神人来写了。

这篇关于SIMD——请不要妄图优化opencv的函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

MySQL count()聚合函数详解

《MySQLcount()聚合函数详解》MySQL中的COUNT()函数,它是SQL中最常用的聚合函数之一,用于计算表中符合特定条件的行数,本文给大家介绍MySQLcount()聚合函数,感兴趣的朋... 目录核心功能语法形式重要特性与行为如何选择使用哪种形式?总结深入剖析一下 mysql 中的 COUNT

MyBatisPlus如何优化千万级数据的CRUD

《MyBatisPlus如何优化千万级数据的CRUD》最近负责的一个项目,数据库表量级破千万,每次执行CRUD都像走钢丝,稍有不慎就引起数据库报警,本文就结合这个项目的实战经验,聊聊MyBatisPl... 目录背景一、MyBATis Plus 简介二、千万级数据的挑战三、优化 CRUD 的关键策略1. 查

MySQL 中 ROW_NUMBER() 函数最佳实践

《MySQL中ROW_NUMBER()函数最佳实践》MySQL中ROW_NUMBER()函数,作为窗口函数为每行分配唯一连续序号,区别于RANK()和DENSE_RANK(),特别适合分页、去重... 目录mysql 中 ROW_NUMBER() 函数详解一、基础语法二、核心特点三、典型应用场景1. 数据分

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN

Python get()函数用法案例详解

《Pythonget()函数用法案例详解》在Python中,get()是字典(dict)类型的内置方法,用于安全地获取字典中指定键对应的值,它的核心作用是避免因访问不存在的键而引发KeyError错... 目录简介基本语法一、用法二、案例:安全访问未知键三、案例:配置参数默认值简介python是一种高级编

python 常见数学公式函数使用详解(最新推荐)

《python常见数学公式函数使用详解(最新推荐)》文章介绍了Python的数学计算工具,涵盖内置函数、math/cmath标准库及numpy/scipy/sympy第三方库,支持从基础算术到复杂数... 目录python 数学公式与函数大全1. 基本数学运算1.1 算术运算1.2 分数与小数2. 数学函数

Python如何将OpenCV摄像头视频流通过浏览器播放

《Python如何将OpenCV摄像头视频流通过浏览器播放》:本文主要介绍Python如何将OpenCV摄像头视频流通过浏览器播放的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完... 目录方法1:使用Flask + MJPEG流实现代码使用方法优点缺点方法2:使用WebSocket传输视

使用Python和OpenCV库实现实时颜色识别系统

《使用Python和OpenCV库实现实时颜色识别系统》:本文主要介绍使用Python和OpenCV库实现的实时颜色识别系统,这个系统能够通过摄像头捕捉视频流,并在视频中指定区域内识别主要颜色(红... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间详解

OpenCV实现实时颜色检测的示例

《OpenCV实现实时颜色检测的示例》本文主要介绍了OpenCV实现实时颜色检测的示例,通过HSV色彩空间转换和色调范围判断实现红黄绿蓝颜色检测,包含视频捕捉、区域标记、颜色分析等功能,具有一定的参考... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间

Python中help()和dir()函数的使用

《Python中help()和dir()函数的使用》我们经常需要查看某个对象(如模块、类、函数等)的属性和方法,Python提供了两个内置函数help()和dir(),它们可以帮助我们快速了解代... 目录1. 引言2. help() 函数2.1 作用2.2 使用方法2.3 示例(1) 查看内置函数的帮助(