c++ - 双三次插值结果与 FFMPEG 不同

标签 c++ ffmpeg interpolation image-resizing bicubic

我刚刚实现了双三次插值来调整图像大小。
我有一个 6x6 像素(灰度)的测试图像,它的列是黑白的(x3)。
我正在将我的代码结果与工具 ffmpeg 的结果进行比较,它们不正确。我不明白为什么,我想我可能计算错误的像素邻域或者调整大小的像素与原始像素的距离。
有人可以查看我的代码(我会简化它以便更好地阅读)并告诉我错误在哪里?

    // Iterate through each line
    for(int lin = 0; lin < dstHeight; lin++){
        // Original coordinates
        float linInOriginal = (lin - 0.5) / scaleHeightRatio;

        // Calculate original pixels coordinates to interpolate
        int linTopFurther = clamp(floor(linInOriginal) - 1, 0, srcHeight - 1);
        int linTop = clamp(floor(linInOriginal), 0, srcHeight - 1);
        int linBottom = clamp(ceil(linInOriginal), 0, srcHeight - 1);
        int linBottomFurther = clamp(ceil(linInOriginal) + 1, 0, srcHeight - 1);

        // Calculate distance to the top left pixel
        float linDist = linInOriginal - floor(linInOriginal);

        // Iterate through each column
        for(int col = 0; col < dstWidth; col++){
            // Original coordinates
            float colInOriginal = (col - 0.5) / scaleWidthRatio;

            // Calculate original pixels coordinates to interpolate
            int colLeftFurther = clamp(floor(colInOriginal) - 1, 0, srcWidth - 1);
            int colLeft = clamp(floor(colInOriginal), 0, srcWidth - 1);
            int colRight = clamp(ceil(colInOriginal), 0, srcWidth - 1);
            int colRightFurther = clamp(ceil(colInOriginal) + 1, 0, srcWidth - 1);

            // Calculate distance to the top left pixel
            float colDist = colInOriginal - floor(colInOriginal);

            // Gets the original pixels values
            // 1st row
            uint8_t p00 = srcSlice[0][linTopFurther * srcWidth + colLeftFurther];
            // ...

            // 2nd row
            uint8_t p01 = srcSlice[0][linTop * srcWidth + colLeftFurther];
            // ...

            // 3rd row
            // ...

            // 4th row
            // ...

            // Bilinear interpolation operation
            // Y
            float value = cubicInterpolate(
                cubicInterpolate(static_cast<float>(p00), static_cast<float>(p10), static_cast<float>(p20), static_cast<float>(p30), colDist),
                cubicInterpolate(static_cast<float>(p01), static_cast<float>(p11), static_cast<float>(p21), static_cast<float>(p31), colDist),
                cubicInterpolate(static_cast<float>(p02), static_cast<float>(p12), static_cast<float>(p22), static_cast<float>(p32), colDist),
                cubicInterpolate(static_cast<float>(p03), static_cast<float>(p13), static_cast<float>(p23), static_cast<float>(p33), colDist),
                linDist);

            dstSlice[0][lin * dstWidth + col] = double2uint8_t(clamp(value, 0.0f, 255.0f));
        }
    }

最佳答案

我忘记设置插值矩阵的二阶变量的值。它们被设置为 0,因此生成的插值将类似于双线性插值。

关于c++ - 双三次插值结果与 FFMPEG 不同,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49130302/

相关文章:

c++ - 标准库中聚合可初始化性的类型特征?

c++ - 如何有条件地实例化一个对象?

c++存储数组下一个位置的地址

audio - 使用 ffmpeg、linux 提取音频

javascript - 在 Rails 中,如何在 HTML 页面上将 javascript 显示为文本?

c++ - 如何在 OS X Yosemite 上安装 FreeGlut(3.0.0 或 2.8.0)

cmd - 使用 cmd 或 media info/ffmpeg 的文件列表和文件大小

android - 捕获 "Unfortunately ' 应用程序'已停止工作“错误

c# - Unity3d 随时间围绕特定点旋转

OpenGL gluLookAt() 未按预期工作