当前位置: 欣欣网 > 码农

OpenCV中C++11 lambda方式像素遍历

2024-03-20码农

点击上方 蓝字 关注我们

微信公众号: OpenCV学堂

关注获取更多计算机视觉与深度学习知识

C++11 lambda语法

C++11中引入了lambda表达式,它支持定义一个内联(inline)的函数,作为一个本地的对象或者一个参数。有了lambda表达式,就可以很方便的使用stl标准库,它的标准语法如下:

[...](...) mutable throwSpec -> returnType {...}

参数解释:

[…]里面包含的是可以引用的本地变量
(…)里面包含的是函数的变量参数
returnType是返回类型

一个简单的例子如下:

auto sum_xy = [](int x, int y) -> int { return x + y;};std::cout <<" sum_xy = "<< sum_xy(11, 12) << std::endl;

引入本地变量

int a = 25, b = 9;auto sum_xy = [&a, &b](int x, int y) -> int { return x + y + a + b;};std::cout <<" sum_xy = "<< sum_xy(11, 12) << std::endl;

注意: 如果直接使用,没有把本地变量放到变量列表中去,就会出现一个常见的 语法错误

封闭函数局部变量不能在lambda体中引用,除非位于捕获列表中

Mat的for Each遍历

说实话我也没有注意过,OpenCV4从哪个版本开始支持,反正已经支持了,通过Mat的forEach方式结合C++11 lambda表达式,实现对Mat对象快速像素遍历。语法如下:

void cv::Mat::forEach(const Functor & operation)

其中operation是一个C++11 lambda表达式,同时也是一个匿名的C++函数。基于Mat的for Each实现的像素遍历代码如下:

// wxh = 3840x2560cv::Mat image = cv::imread("D:/test_pixs.jpg");typedef cv::Point3_<uint8_t> Pixel;// forEach方式的像素遍历double start = (double)cv::getTickCount();image.forEach<Pixel>([](Pixel &p, constint * position) -> void {p.x = 255 - p.x;p.y = 255 - p.y;p.z = 255 - p.z;});double time = (((double)cv::getTickCount() - start)) / cv::getTickFrequency();printf(" forEach time : %.4f seconds\n", time);

传统高效的OpenCV指针方式的像素遍历访问代码如下

// raw pointer access.start = (double)cv::getTickCount();for (int r = 0; r < image.rows; ++r) { Pixel* ptr = image.ptr<Pixel>(r, 0);const Pixel* ptr_end = ptr + image.cols;for (; ptr != ptr_end; ++ptr) { ptr->x = 255 - ptr->x; ptr->y = 255 - ptr->y; ptr->z = 255 - ptr->z; }}time = (((double)cv::getTickCount() - start)) / cv::getTickFrequency();printf(" raw pointer access time : %.4f seconds\n", time);

运行结果对比如下:

从执行的时间可以看出,针对一张3840x2560大小的图像、forEach方式遍历的确比较靠谱。

系统化学习直接扫码查看

推荐阅读