1

这篇文章讨论了如何用 OpenCV 转置图像,在这里我想更进一步:假设图像是灰度图像,使用 C++ 转置它(或矩阵)的最快方法是什么?我的解决方案如下:

        // image data is stored in an image buffer image*buffer_

    unsigned char *mem = (unsigned char *) malloc(image.bufferSize_);

    int height = image.Height();
    int width = image.Width();
    for(int i=0; i<height; i++)
    {
        unsigned char *ptr =image.buffer_+i*width;
        for(int j=0; j<width; j++)
            *(mem+j*height+i) = *(ptr+j);
    }


    memcpy(image.buffer_,mem,image.bufferSize_);
    free(mem);

上面代码的一些解释:我们创建一个包含基本图像信息以及图像像素(in image.buffer_)的图像对象。当图像像素存储在 中时image.buffer_,我们假设图像像素是逐行保存的。关于进一步改进上述代码的任何想法?

4

1 回答 1

1

在不触及 malloc/free 部分的情况下,复制部分可能如下所示:

    size_t len = image.bufferSize_,
           len1 = len - 1;

    unsigned char *src = image.buffer_,
                  *dest = mem,
                  *end = dest + len;

    for(size_t i = 0; i < len; i++)
    {
        *dest++ = *src;  // dest moves to next row
        src += height;   // src moves to next column

        // src wraps around and moves to next row
        if (src > end) src -= len1;
    }

这等效于具有按列的目标迭代器和按行的源迭代器。

在没有实际测试的情况下,我觉得这会更快:它在内部循环中有 3 个用于偏移计算的操作,而在您的版本中有 4 个(在两个版本中都有 2 个取消引用操作)。

编辑

另一项改进和更正:

    //...
    unsigned char *src = image.buffer_,
                  *src_end = src + len,
                  *dest = mem,
                  *dest_end = dest + len;

    while (dest != dest_end)
    {
        *dest++ = *src;  // dest moves to next row
        src += height;   // src moves to next column

        // src wraps around and moves to next row
        if (src > src_end) src -= len1;
    }

i++这样每次迭代(在for循环中)可以节省一个操作。以前也src被比错了end

于 2013-09-18T10:42:51.303 回答