resize函数有五种插值算法
转自http://blog.csdn.net/fengbingchun/article/details/17335477
最新版OpenCV2.4.7中,cv::resize函数有五种插值算法:最近邻、双线性、双三次、基于像素区域关系、兰索斯插值。下面用for循环代替cv::resize函数来说明其详细的插值实现过程,其中部分代码摘自于cv::resize函数中的源代码。
每种插值算法的前部分代码是相同的,如下:
- cv::Mat matSrc, matDst1, matDst2;
- matSrc = cv::imread("lena.jpg", 2 | 4);
- matDst1 = cv::Mat(cv::Size(800, 1000), matSrc.type(), cv::Scalar::all(0));
- matDst2 = cv::Mat(matDst1.size(), matSrc.type(), cv::Scalar::all(0));
- double scale_x = (double)matSrc.cols / matDst1.cols;
- double scale_y = (double)matSrc.rows / matDst1.rows;
1、最近邻:公式,
- for (int i = 0; i < matDst1.cols; ++i)
- {
- int sx = cvFloor(i * scale_x);
- sx = std::min(sx, matSrc.cols - 1);
- for (int j = 0; j < matDst1.rows; ++j)
- {
- int sy = cvFloor(j * scale_y);
- sy = std::min(sy, matSrc.rows - 1);
- matDst1.at<cv::Vec3b>(j, i) = matSrc.at<cv::Vec3b>(sy, sx);
- }
- }
- cv::imwrite("nearest_1.jpg", matDst1);
- cv::resize(matSrc, matDst2, matDst1.size(), 0, 0, 0);
- cv::imwrite("nearest_2.jpg", matDst2);
2、双线性:由相邻的四像素(2*2)计算得出,公式,
- uchar* dataDst = matDst1.data;
- int stepDst = matDst1.step;
- uchar* dataSrc = matSrc.data;
- int stepSrc = matSrc.step;
- int iWidthSrc = matSrc.cols;
- int iHiehgtSrc = matSrc.rows;
- for (int j = 0; j < matDst1.rows; ++j)
- {
- float fy = (float)((j + 0.5) * scale_y - 0.5);
- int sy = cvFloor(fy);
- fy -= sy;
- sy = std::min(sy, iHiehgtSrc - 2);
- sy = std::max(0, sy);
- short cbufy[2];
- cbufy[0] = cv::saturate_cast<short>((1.f - fy) * 2048);
- cbufy[1] = 2048 - cbufy[0];
- for (int i = 0; i < matDst1.cols; ++i)
- {
- float fx = (float)((i + 0.5) * scale_x - 0.5);
- int sx = cvFloor(fx);
- fx -= sx;
- if (sx < 0) {
- fx = 0, sx = 0;
- }
- if (sx >= iWidthSrc - 1) {
- fx = 0, sx = iWidthSrc - 2;
- }
- short cbufx[2];
- cbufx[0] = cv::saturate_cast<short>((1.f - fx) * 2048);
- cbufx[1] = 2048 - cbufx[0];
- for (int k = 0; k < matSrc.channels(); ++k)
- {
- *(dataDst+ j*stepDst + 3*i + k) = (*(dataSrc + sy*stepSrc + 3*sx + k) * cbufx[0] * cbufy[0] +
- *(dataSrc + (sy+1)*stepSrc + 3*sx + k) * cbufx[0] * cbufy[1] +
- *(dataSrc + sy*stepSrc + 3*(sx+1) + k) * cbufx[1] * cbufy[0] +
- *(dataSrc + (sy+1)*stepSrc + 3*(sx+1) + k) * cbufx[1] * cbufy[1]) >> 22;
- }
- }
- }
- cv::imwrite("linear_1.jpg", matDst1);
- cv::resize(matSrc, matDst2, matDst1.size(), 0, 0, 1);
- cv::imwrite("linear_2.jpg", matDst2);
3、双三次:由相邻的4*4像素计算得出,公式类似于双线性
- int iscale_x = cv::saturate_cast<int>(scale_x);
- int iscale_y = cv::saturate_cast<int>(scale_y);
- for (int j = 0; j < matDst1.rows; ++j)
- {
- float fy = (float)((j + 0.5) * scale_y - 0.5);
- int sy = cvFloor(fy);
- fy -= sy;
- sy = std::min(sy, matSrc.rows - 3);
- sy = std::max(1, sy);
- const float A = -0.75f;
- float coeffsY[4];
- coeffsY[0] = ((A*(fy + 1) - 5*A)*(fy + 1) + 8*A)*(fy + 1) - 4*A;
- coeffsY[1] = ((A + 2)*fy - (A + 3))*fy*fy + 1;
- coeffsY[2] = ((A + 2)*(1 - fy) - (A + 3))*(1 - fy)*(1 - fy) + 1;
- coeffsY[3] = 1.f - coeffsY[0] - coeffsY[1] - coeffsY[2];
- short cbufY[4];
- cbufY[0] = cv::saturate_cast<short>(coeffsY[0] * 2048);
- cbufY[1] = cv::saturate_cast<short>(coeffsY[1] * 2048);
- cbufY[2] = cv::saturate_cast<short>(coeffsY[2] * 2048);
- cbufY[3] = cv::saturate_cast<short>(coeffsY[3] * 2048);
- for (int i = 0; i < matDst1.cols; ++i)
- {
- float fx = (float)((i + 0.5) * scale_x - 0.5);
- int sx = cvFloor(fx);
- fx -= sx;
- if (sx < 1) {
- fx = 0, sx = 1;
- }
- if (sx >= matSrc.cols - 3) {
- fx = 0, sx = matSrc.cols - 3;
- }
- float coeffsX[4];
- coeffsX[0] = ((A*(fx + 1) - 5*A)*(fx + 1) + 8*A)*(fx + 1) - 4*A;
- coeffsX[1] = ((A + 2)*fx - (A + 3))*fx*fx + 1;
- coeffsX[2] = ((A + 2)*(1 - fx) - (A + 3))*(1 - fx)*(1 - fx) + 1;
- coeffsX[3] = 1.f - coeffsX[0] - coeffsX[1] - coeffsX[2];
- short cbufX[4];
- cbufX[0] = cv::saturate_cast<short>(coeffsX[0] * 2048);
- cbufX[1] = cv::saturate_cast<short>(coeffsX[1] * 2048);
- cbufX[2] = cv::saturate_cast<short>(coeffsX[2] * 2048);
- cbufX[3] = cv::saturate_cast<short>(coeffsX[3] * 2048);
- for (int k = 0; k < matSrc.channels(); ++k)
- {
- matDst1.at<cv::Vec3b>(j, i)[k] = abs((matSrc.at<cv::Vec3b>(sy-1, sx-1)[k] * cbufX[0] * cbufY[0] + matSrc.at<cv::Vec3b>(sy, sx-1)[k] * cbufX[0] * cbufY[1] +
- matSrc.at<cv::Vec3b>(sy+1, sx-1)[k] * cbufX[0] * cbufY[2] + matSrc.at<cv::Vec3b>(sy+2, sx-1)[k] * cbufX[0] * cbufY[3] +
- matSrc.at<cv::Vec3b>(sy-1, sx)[k] * cbufX[1] * cbufY[0] + matSrc.at<cv::Vec3b>(sy, sx)[k] * cbufX[1] * cbufY[1] +
- matSrc.at<cv::Vec3b>(sy+1, sx)[k] * cbufX[1] * cbufY[2] + matSrc.at<cv::Vec3b>(sy+2, sx)[k] * cbufX[1] * cbufY[3] +
- matSrc.at<cv::Vec3b>(sy-1, sx+1)[k] * cbufX[2] * cbufY[0] + matSrc.at<cv::Vec3b>(sy, sx+1)[k] * cbufX[2] * cbufY[1] +
- matSrc.at<cv::Vec3b>(sy+1, sx+1)[k] * cbufX[2] * cbufY[2] + matSrc.at<cv::Vec3b>(sy+2, sx+1)[k] * cbufX[2] * cbufY[3] +
- matSrc.at<cv::Vec3b>(sy-1, sx+2)[k] * cbufX[3] * cbufY[0] + matSrc.at<cv::Vec3b>(sy, sx+2)[k] * cbufX[3] * cbufY[1] +
- matSrc.at<cv::Vec3b>(sy+1, sx+2)[k] * cbufX[3] * cbufY[2] + matSrc.at<cv::Vec3b>(sy+2, sx+2)[k] * cbufX[3] * cbufY[3] ) >> 22);
- }
- }
- }
- cv::imwrite("cubic_1.jpg", matDst1);
- cv::resize(matSrc, matDst2, matDst1.size(), 0, 0, 2);
- cv::imwrite("cubic_2.jpg", matDst2);
4、基于像素区域关系:共分三种情况,图像放大时类似于双线性插值,图像缩小(x轴、y轴同时缩小)又分两种情况,此情况下可以避免波纹出现。
- cv::resize(matSrc, matDst2, matDst1.size(), 0, 0, 3);
- cv::imwrite("area_2.jpg", matDst2);
- double inv_scale_x = 1. / scale_x;
- double inv_scale_y = 1. / scale_y;
- int iscale_x = cv::saturate_cast<int>(scale_x);
- int iscale_y = cv::saturate_cast<int>(scale_y);
- bool is_area_fast = std::abs(scale_x - iscale_x) < DBL_EPSILON && std::abs(scale_y - iscale_y) < DBL_EPSILON;
- if (scale_x >= 1 && scale_y >= 1) //zoom out
- {
- if (is_area_fast) //integer multiples
- {
- for (int j = 0; j < matDst1.rows; ++j)
- {
- int sy = j * scale_y;
- for (int i = 0; i < matDst1.cols; ++i)
- {
- int sx = i * scale_x;
- matDst1.at<cv::Vec3b>(j, i) = matSrc.at<cv::Vec3b>(sy, sx);
- }
- }
- cv::imwrite("area_1.jpg", matDst1);
- return 0;
- }
- for (int j = 0; j < matDst1.rows; ++j)
- {
- double fsy1 = j * scale_y;
- double fsy2 = fsy1 + scale_y;
- double cellHeight = cv::min(scale_y, matSrc.rows - fsy1);
- int sy1 = cvCeil(fsy1), sy2 = cvFloor(fsy2);
- sy2 = std::min(sy2, matSrc.rows - 1);
- sy1 = std::min(sy1, sy2);
- float cbufy[2];
- cbufy[0] = (float)((sy1 - fsy1) / cellHeight);
- cbufy[1] = (float)(std::min(std::min(fsy2 - sy2, 1.), cellHeight) / cellHeight);
- for (int i = 0; i < matDst1.cols; ++i)
- {
- double fsx1 = i * scale_x;
- double fsx2 = fsx1 + scale_x;
- double cellWidth = std::min(scale_x, matSrc.cols - fsx1);
- int sx1 = cvCeil(fsx1), sx2 = cvFloor(fsx2);
- sx2 = std::min(sx2, matSrc.cols - 1);
- sx1 = std::min(sx1, sx2);
- float cbufx[2];
- cbufx[0] = (float)((sx1 - fsx1) / cellWidth);
- cbufx[1] = (float)(std::min(std::min(fsx2 - sx2, 1.), cellWidth) / cellWidth);
- for (int k = 0; k < matSrc.channels(); ++k)
- {
- matDst1.at<cv::Vec3b>(j, i)[k] = (uchar)(matSrc.at<cv::Vec3b>(sy1, sx1)[k] * cbufx[0] * cbufy[0] +
- matSrc.at<cv::Vec3b>(sy1 + 1, sx1)[k] * cbufx[0] * cbufy[1] +
- matSrc.at<cv::Vec3b>(sy1, sx1 + 1)[k] * cbufx[1] * cbufy[0] +
- matSrc.at<cv::Vec3b>(sy1 + 1, sx1 + 1)[k] * cbufx[1] * cbufy[1]);
- }
- }
- }
- cv::imwrite("area_1.jpg", matDst1);
- return 0;
- }
- //zoom in,it is emulated using some variant of bilinear interpolation
- for (int j = 0; j < matDst1.rows; ++j)
- {
- int sy = cvFloor(j * scale_y);
- float fy = (float)((j + 1) - (sy + 1) * inv_scale_y);
- fy = fy <= 0 ? 0.f : fy - cvFloor(fy);
- short cbufy[2];
- cbufy[0] = cv::saturate_cast<short>((1.f - fy) * 2048);
- cbufy[1] = 2048 - cbufy[0];
- for (int i = 0; i < matDst1.cols; ++i)
- {
- int sx = cvFloor(i * scale_x);
- float fx = (float)((i + 1) - (sx + 1) * inv_scale_x);
- fx = fx < 0 ? 0.f : fx - cvFloor(fx);
- if (sx < 0) {
- fx = 0, sx = 0;
- }
- if (sx >= matSrc.cols - 1) {
- fx = 0, sx = matSrc.cols - 2;
- }
- short cbufx[2];
- cbufx[0] = cv::saturate_cast<short>((1.f - fx) * 2048);
- cbufx[1] = 2048 - cbufx[0];
- for (int k = 0; k < matSrc.channels(); ++k)
- {
- matDst1.at<cv::Vec3b>(j, i)[k] = (matSrc.at<cv::Vec3b>(sy, sx)[k] * cbufx[0] * cbufy[0] +
- matSrc.at<cv::Vec3b>(sy + 1, sx)[k] * cbufx[0] * cbufy[1] +
- matSrc.at<cv::Vec3b>(sy, sx + 1)[k] * cbufx[1] * cbufy[0] +
- matSrc.at<cv::Vec3b>(sy + 1, sx + 1)[k] * cbufx[1] * cbufy[1]) >> 22;
- }
- }
- }
- cv::imwrite("area_1.jpg", matDst1);
5、兰索斯插值:由相邻的8*8像素计算得出,公式类似于双线性
- int iscale_x = cv::saturate_cast<int>(scale_x);
- int iscale_y = cv::saturate_cast<int>(scale_y);
- for (int j = 0; j < matDst1.rows; ++j)
- {
- float fy = (float)((j + 0.5) * scale_y - 0.5);
- int sy = cvFloor(fy);
- fy -= sy;
- sy = std::min(sy, matSrc.rows - 5);
- sy = std::max(3, sy);
- const double s45 = 0.70710678118654752440084436210485;
- const double cs[][2] = {{1, 0}, {-s45, -s45}, {0, 1}, {s45, -s45}, {-1, 0}, {s45, s45}, {0, -1}, {-s45, s45}};
- float coeffsY[8];
- if (fy < FLT_EPSILON) {
- for (int t = 0; t < 8; t++)
- coeffsY[t] = 0;
- coeffsY[3] = 1;
- } else {
- float sum = 0;
- double y0 = -(fy + 3) * CV_PI * 0.25, s0 = sin(y0), c0 = cos(y0);
- for (int t = 0; t < 8; ++t)
- {
- double dy = -(fy + 3 -t) * CV_PI * 0.25;
- coeffsY[t] = (float)((cs[t][0] * s0 + cs[t][1] * c0) / (dy * dy));
- sum += coeffsY[t];
- }
- sum = 1.f / sum;
- for (int t = 0; t < 8; ++t)
- coeffsY[t] *= sum;
- }
- short cbufY[8];
- cbufY[0] = cv::saturate_cast<short>(coeffsY[0] * 2048);
- cbufY[1] = cv::saturate_cast<short>(coeffsY[1] * 2048);
- cbufY[2] = cv::saturate_cast<short>(coeffsY[2] * 2048);
- cbufY[3] = cv::saturate_cast<short>(coeffsY[3] * 2048);
- cbufY[4] = cv::saturate_cast<short>(coeffsY[4] * 2048);
- cbufY[5] = cv::saturate_cast<short>(coeffsY[5] * 2048);
- cbufY[6] = cv::saturate_cast<short>(coeffsY[6] * 2048);
- cbufY[7] = cv::saturate_cast<short>(coeffsY[7] * 2048);
- for (int i = 0; i < matDst1.cols; ++i)
- {
- float fx = (float)((i + 0.5) * scale_x - 0.5);
- int sx = cvFloor(fx);
- fx -= sx;
- if (sx < 3) {
- fx = 0, sx = 3;
- }
- if (sx >= matSrc.cols - 5) {
- fx = 0, sx = matSrc.cols - 5;
- }
- float coeffsX[8];
- if (fx < FLT_EPSILON) {
- for ( int t = 0; t < 8; t++ )
- coeffsX[t] = 0;
- coeffsX[3] = 1;
- } else {
- float sum = 0;
- double x0 = -(fx + 3) * CV_PI * 0.25, s0 = sin(x0), c0 = cos(x0);
- for (int t = 0; t < 8; ++t)
- {
- double dx = -(fx + 3 -t) * CV_PI * 0.25;
- coeffsX[t] = (float)((cs[t][0] * s0 + cs[t][1] * c0) / (dx * dx));
- sum += coeffsX[t];
- }
- sum = 1.f / sum;
- for (int t = 0; t < 8; ++t)
- coeffsX[t] *= sum;
- }
- short cbufX[8];
- cbufX[0] = cv::saturate_cast<short>(coeffsX[0] * 2048);
- cbufX[1] = cv::saturate_cast<short>(coeffsX[1] * 2048);
- cbufX[2] = cv::saturate_cast<short>(coeffsX[2] * 2048);
- cbufX[3] = cv::saturate_cast<short>(coeffsX[3] * 2048);
- cbufX[4] = cv::saturate_cast<short>(coeffsX[4] * 2048);
- cbufX[5] = cv::saturate_cast<short>(coeffsX[5] * 2048);
- cbufX[6] = cv::saturate_cast<short>(coeffsX[6] * 2048);
- cbufX[7] = cv::saturate_cast<short>(coeffsX[7] * 2048);
- for (int k = 0; k < matSrc.channels(); ++k)
- {
- matDst1.at<cv::Vec3b>(j, i)[k] = abs((matSrc.at<cv::Vec3b>(sy-3, sx-3)[k] * cbufX[0] * cbufY[0] + matSrc.at<cv::Vec3b>(sy-2, sx-3)[k] * cbufX[0] * cbufY[1] +
- matSrc.at<cv::Vec3b>(sy-1, sx-3)[k] * cbufX[0] * cbufY[2] + matSrc.at<cv::Vec3b>(sy, sx-3)[k] * cbufX[0] * cbufY[3] +
- matSrc.at<cv::Vec3b>(sy+1, sx-3)[k] * cbufX[0] * cbufY[4] + matSrc.at<cv::Vec3b>(sy+2, sx-3)[k] * cbufX[0] * cbufY[5] +
- matSrc.at<cv::Vec3b>(sy+3, sx-3)[k] * cbufX[0] * cbufY[6] + matSrc.at<cv::Vec3b>(sy+4, sx-3)[k] * cbufX[0] * cbufY[7] +
- matSrc.at<cv::Vec3b>(sy-3, sx-2)[k] * cbufX[1] * cbufY[0] + matSrc.at<cv::Vec3b>(sy-2, sx-2)[k] * cbufX[1] * cbufY[1] +
- matSrc.at<cv::Vec3b>(sy-1, sx-2)[k] * cbufX[1] * cbufY[2] + matSrc.at<cv::Vec3b>(sy, sx-2)[k] * cbufX[1] * cbufY[3] +
- matSrc.at<cv::Vec3b>(sy+1, sx-2)[k] * cbufX[1] * cbufY[4] + matSrc.at<cv::Vec3b>(sy+2, sx-2)[k] * cbufX[1] * cbufY[5] +
- matSrc.at<cv::Vec3b>(sy+3, sx-2)[k] * cbufX[1] * cbufY[6] + matSrc.at<cv::Vec3b>(sy+4, sx-2)[k] * cbufX[1] * cbufY[7] +
- matSrc.at<cv::Vec3b>(sy-3, sx-1)[k] * cbufX[2] * cbufY[0] + matSrc.at<cv::Vec3b>(sy-2, sx-1)[k] * cbufX[2] * cbufY[1] +
- matSrc.at<cv::Vec3b>(sy-1, sx-1)[k] * cbufX[2] * cbufY[2] + matSrc.at<cv::Vec3b>(sy, sx-1)[k] * cbufX[2] * cbufY[3] +
- matSrc.at<cv::Vec3b>(sy+1, sx-1)[k] * cbufX[2] * cbufY[4] + matSrc.at<cv::Vec3b>(sy+2, sx-1)[k] * cbufX[2] * cbufY[5] +
- matSrc.at<cv::Vec3b>(sy+3, sx-1)[k] * cbufX[2] * cbufY[6] + matSrc.at<cv::Vec3b>(sy+4, sx-1)[k] * cbufX[2] * cbufY[7] +
- matSrc.at<cv::Vec3b>(sy-3, sx)[k] * cbufX[3] * cbufY[0] + matSrc.at<cv::Vec3b>(sy-2, sx)[k] * cbufX[3] * cbufY[1] +
- matSrc.at<cv::Vec3b>(sy-1, sx)[k] * cbufX[3] * cbufY[2] + matSrc.at<cv::Vec3b>(sy, sx)[k] * cbufX[3] * cbufY[3] +
- matSrc.at<cv::Vec3b>(sy+1, sx)[k] * cbufX[3] * cbufY[4] + matSrc.at<cv::Vec3b>(sy+2, sx)[k] * cbufX[3] * cbufY[5] +
- matSrc.at<cv::Vec3b>(sy+3, sx)[k] * cbufX[3] * cbufY[6] + matSrc.at<cv::Vec3b>(sy+4, sx)[k] * cbufX[3] * cbufY[7] +
- matSrc.at<cv::Vec3b>(sy-3, sx+1)[k] * cbufX[4] * cbufY[0] + matSrc.at<cv::Vec3b>(sy-2, sx+1)[k] * cbufX[4] * cbufY[1] +
- matSrc.at<cv::Vec3b>(sy-1, sx+1)[k] * cbufX[4] * cbufY[2] + matSrc.at<cv::Vec3b>(sy, sx+1)[k] * cbufX[4] * cbufY[3] +
- matSrc.at<cv::Vec3b>(sy+1, sx+1)[k] * cbufX[4] * cbufY[4] + matSrc.at<cv::Vec3b>(sy+2, sx+1)[k] * cbufX[4] * cbufY[5] +
- matSrc.at<cv::Vec3b>(sy+3, sx+1)[k] * cbufX[4] * cbufY[6] + matSrc.at<cv::Vec3b>(sy+4, sx+1)[k] * cbufX[4] * cbufY[7] +
- matSrc.at<cv::Vec3b>(sy-3, sx+2)[k] * cbufX[5] * cbufY[0] + matSrc.at<cv::Vec3b>(sy-2, sx+2)[k] * cbufX[5] * cbufY[1] +
- matSrc.at<cv::Vec3b>(sy-1, sx+2)[k] * cbufX[5] * cbufY[2] + matSrc.at<cv::Vec3b>(sy, sx+2)[k] * cbufX[5] * cbufY[3] +
- matSrc.at<cv::Vec3b>(sy+1, sx+2)[k] * cbufX[5] * cbufY[4] + matSrc.at<cv::Vec3b>(sy+2, sx+2)[k] * cbufX[5] * cbufY[5] +
- matSrc.at<cv::Vec3b>(sy+3, sx+2)[k] * cbufX[5] * cbufY[6] + matSrc.at<cv::Vec3b>(sy+4, sx+2)[k] * cbufX[5] * cbufY[7] +
- matSrc.at<cv::Vec3b>(sy-3, sx+3)[k] * cbufX[6] * cbufY[0] + matSrc.at<cv::Vec3b>(sy-2, sx+3)[k] * cbufX[6] * cbufY[1] +
- matSrc.at<cv::Vec3b>(sy-1, sx+3)[k] * cbufX[6] * cbufY[2] + matSrc.at<cv::Vec3b>(sy, sx+3)[k] * cbufX[6] * cbufY[3] +
- matSrc.at<cv::Vec3b>(sy+1, sx+3)[k] * cbufX[6] * cbufY[4] + matSrc.at<cv::Vec3b>(sy+2, sx+3)[k] * cbufX[6] * cbufY[5] +
- matSrc.at<cv::Vec3b>(sy+3, sx+3)[k] * cbufX[6] * cbufY[6] + matSrc.at<cv::Vec3b>(sy+4, sx+3)[k] * cbufX[6] * cbufY[7] +
- matSrc.at<cv::Vec3b>(sy-3, sx+4)[k] * cbufX[7] * cbufY[0] + matSrc.at<cv::Vec3b>(sy-2, sx+4)[k] * cbufX[7] * cbufY[1] +
- matSrc.at<cv::Vec3b>(sy-1, sx+4)[k] * cbufX[7] * cbufY[2] + matSrc.at<cv::Vec3b>(sy, sx+4)[k] * cbufX[7] * cbufY[3] +
- matSrc.at<cv::Vec3b>(sy+1, sx+4)[k] * cbufX[7] * cbufY[4] + matSrc.at<cv::Vec3b>(sy+2, sx+4)[k] * cbufX[7] * cbufY[5] +
- matSrc.at<cv::Vec3b>(sy+3, sx+4)[k] * cbufX[7] * cbufY[6] + matSrc.at<cv::Vec3b>(sy+4, sx+4)[k] * cbufX[7] * cbufY[7] ) >> 22);// 4194304
- }
- }
- }
- cv::imwrite("Lanczos_1.jpg", matDst1);
- cv::resize(matSrc, matDst2, matDst1.size(), 0, 0, 4);
- cv::imwrite("Lanczos_2.jpg", matDst2);
以上代码的实现结果与cv::resize函数相同,但是执行效率非常低,只是为了详细说明插值过程。OpenCV中默认采用C++ Concurrency进行优化加速,你也可以采用TBB、OpenMP等进行优化加速。
resize函数有五种插值算法的更多相关文章
- 第196天:js---调用函数的五种方式
一.普通方式 /*普通模式*/ // 声明一个函数,并调用 function func() { console.log("Hello World"); } func(); 二.函数 ...
- matlab自定义函数的五种表示(前2种重点)
1.命令文件/函数文件+函数文件:多个M文件 2.函数文件+子函数:一个M文件 3. inline:无需M文件 4.符号表达式+subs方式:无需M文件 5.字符串+subs方式:无需M文件 第一种: ...
- 五种I/O模型和select函数简介
一.五种I/O模型 1.阻塞I/O 我们在前面所说的I/O模型都是阻塞I/O,即调用recv系统调用,如果没有数据则阻塞等待,当数据到来则将数据从内核空间(套接口缓冲区)拷贝到用户空间(recv函数提 ...
- js 五种绑定彻底弄懂this,默认绑定、隐式绑定、显式绑定、new绑定、箭头函数绑定详解
壹 ❀ 引 可以说this与闭包.原型链一样,属于JavaScript开发中老生常谈的问题了,百度一搜,this相关的文章铺天盖地.可开发好几年,被几道this题安排明明白白的人应该不在少数(我就是 ...
- 转:五种I/O模型和select函数简介
源地址:http://blog.csdn.net/jnu_simba/article/details/9070955 一.五种I/O模型 1.阻塞I/O 我们在前面所说的I/O模型都是阻塞I/O,即调 ...
- JS五种绑定彻底弄懂this,默认绑定、隐式绑定、显式绑定、new绑定、箭头函数绑定详解(转载)
目录 壹 ❀ 引 贰 ❀ this默认绑定 叁 ❀ this隐式绑定 1.隐式绑定 2.隐式丢失 肆 ❀ this显式绑定 伍 ❀ new绑定 陆 ❀ this绑定优先级 柒 ❀ 箭头函数的this ...
- opencv2.4.13.7的resize函数使用(c++)
先来看一下resize函数的原型,如下. C++: void resize(InputArray src, OutputArray dst, Size dsize, double fx=0, doub ...
- 简述linux同步与异步、阻塞与非阻塞概念以及五种IO模型
1.概念剖析 相信很多从事linux后台开发工作的都接触过同步&异步.阻塞&非阻塞这样的概念,也相信都曾经产生过误解,比如认为同步就是阻塞.异步就是非阻塞,下面我们先剖析下这几个概念分 ...
- php五种常见的设计模式(转载)
很多人都想着写博客来记录编程生活中的点滴,我也不例外,但想了好长时间不知道写什么........万事开头难,先转载一篇吧..... 设计模式 一书将设计模式引入软件社区,该书的作者是 Erich Ga ...
随机推荐
- vue-router使用next()跳转到指定路径时会无限循环
我在路由为 /path 的页面这样写 beforeRouteLeave (to, from, next) { console.log('离开路路由') if(to.fullPath==='/home' ...
- Javascript-- jQuery Ajax应用
使用ajax()方法加载服务器数据 使用ajax()方法是最底层.功能最强大的请求服务器数据的方法,它不仅可以获取服务器返回的数据,还能向服务器发送请求并传递数值,它的调用格式如下: jQuery.a ...
- MBA 工商管理课程-风险型决策方法
(二)风险型决策方法 适用的条件 未来情况不止一种,管理者不知道到底哪种情况会发生,但知道每种情况发生的概率 常用方法: ...
- python_Notepad++编码集的说明
window环境 [以ANSI格式编码]:简对应python中的"gbk"编码 [以UTF-8无BOM格式编码]:对应python中的"utf-8"编码 [以U ...
- 3.4 常用的两种 layer 层 3.7 字体与文本
3.4 常用的两种 layer 层 //在cocos2d-x中,经常使用到的两种 layer 层 : CCLayer 和 CCLayerColor //CCLayer 的创建 CCLayer* la ...
- Linux-CentOS7 安装VMware Workstation 12
转自:http://blog.csdn.net/aoshilang2249/article/details/48656107 1.下载VMware 衔接地址 http://www.vmware.com ...
- codevs 2503 失恋28天-缝补礼物
题目描述 Description 话说上回他给女孩送了n件礼物,由于是廉价的所以全部都坏掉了,女孩很在意这些礼物,所以决定自己缝补,但是人生苦短啊,女孩时间有限,她总共有m分钟能去缝补礼物.由于损坏程 ...
- Android手机里的垃圾文件和文件夹清理
SD卡中各个文件夹功能的最详尽分析SD卡用久了会有好多文件夹出现,大家看看都是干什么用~ 1..android_secure 是官方app2sd的产物,删了之后装到sd卡中的软件就无法使用了.2.. ...
- as3设计模式乱用之工厂模式
好久没写技术相关的日记了,一忙,二懒,三则被这单调的生活熏得没什么感悟. 其实这篇日记早就想写了,项目开发初期的时候,带学生.经常看到那种乱用设计模式的现象.一方面,公司面试人的时候喜欢问设计模式,另 ...
- OTL调用Oracle存储过程
OTL很早前用过,今天写东西要调存储过程,程序写完了,调试死活通不过,折腾了一早晨. 最后才发现错误,这里总结一下: 1.代码写的不规范. 有个参数后边少写了个“,”以至于总是抱错.而单独写的测试例子 ...