图片缩放算法

项目背景:博主之前做过一个摄像头采集数据,然后在LCD上显示视频数据的项目,假如我们摄像头采集的一帧数据的分辨率比我们的LCD的分辨率要大,那么LCD则无法显示整个图像,这时候我们就要把这么一帧图片进行缩放,然后再显示在LCD上。

 

本文采用的是近邻取样插值方法缩放图片,比如原图有100个点,想缩放10倍的话,那就要取10个点,(点的编号)比如0 10 20 30 。。。100(其他点就不要了)。

下面我们直接通过程序来分析:首先我们用  输入分辨率/压缩后的分辨率  可得到一个缩放比例值k,根据这个k值找出我们要保存原图上哪些竖线横线。横线和竖线的交点就是我们要保存的点。

/**********************************************************************
 * 函数名称: PicZoom
 * 功能描述: 近邻取样插值方法缩放图片
 *            注意该函数会分配内存来存放缩放后的图片,用完后要用free函数释放掉
 *            "近邻取样插值"的原理请参考网友"lantianyu520"所著的"图像缩放算法"
 * 输入参数: ptOriginPic - 内含原始图片的象素数据
 *            ptBigPic    - 内含缩放后的图片的象素数据
 * 输出参数: 无
 * 返 回 值: 0 - 成功, 其他值 - 失败
 ***********************************************************************/
int PicZoom(PT_PixelDatas ptOriginPic, PT_PixelDatas ptZoomPic)
{
    unsigned long dwDstWidth = ptZoomPic->iWidth;//压缩后分辨率的宽度值
    unsigned long* pdwSrcXTable;
	unsigned long x;
	unsigned long y;
	unsigned long dwSrcY;
	unsigned char *pucDest;
	unsigned char *pucSrc;
	unsigned long dwPixelBytes = ptOriginPic->iBpp/8;

	if (ptOriginPic->iBpp != ptZoomPic->iBpp)
	{
		return -1;
	}

        //pdwSrcXTable数组用来存放要保存的是哪些“竖线”(保存上图红色竖线的位置)
    pdwSrcXTable = malloc(sizeof(unsigned long) * dwDstWidth);
    if (NULL == pdwSrcXTable)
    {
        DBG_PRINTF("malloc error!\n");
        return -1;
    }
        //pdwSrcXTable数组用来存放要保存的是哪些“竖线”(保存上图红色竖线的位置)
    for (x = 0; x < dwDstWidth; x++)//生成表 pdwSrcXTable
    {
        pdwSrcXTable[x]=(x*ptOriginPic->iWidth/ptZoomPic->iWidth);
    }

        //循环“压缩后图片的分辨率的高度”次,也就是开始把要留下的点放进输出buffer了
    for (y = 0; y < ptZoomPic->iHeight; y++)
    {		
        //找出每一次要保存的“横线”	
        dwSrcY = (y * ptOriginPic->iHeight / ptZoomPic->iHeight);

        //目的的起始地址
		pucDest = ptZoomPic->aucPixelDatas + y*ptZoomPic->iLineBytes;
        //源的起始地址(横线的位置是每个点的起始地址,竖线的位置是偏移位置)
		pucSrc  = ptOriginPic->aucPixelDatas + dwSrcY*ptOriginPic->iLineBytes;
		
        //每条横线上有dwDstWidth个点要保存
        for (x = 0; x <dwDstWidth; x++)
        {
            //结合pdwSrcXTable数组找出要保存的点
			 memcpy(pucDest+x*dwPixelBytes, pucSrc+pdwSrcXTable[x]*dwPixelBytes, dwPixelBytes);
        }
    }

    free(pdwSrcXTable);
	return 0;
}

 

 

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
图片算法是一种常见的图像处理技术,它能够将一张图像按照一定比例进行,从而得到所需的尺寸。在此,提供一份C语言代码,用于实现图片算法。 代码实现(以双线性插值为例): ``` #include <stdio.h> #include <stdlib.h> #include <string.h> #include <math.h> #define STB_IMAGE_IMPLEMENTATION #include "stb_image.h" #define STB_IMAGE_WRITE_IMPLEMENTATION #include "stb_image_write.h" typedef unsigned char BYTE; int main() { int width, height, bpp; BYTE* buffer = stbi_load("test.png", &width, &height, &bpp, 0); if (!buffer) { printf("Failed to load image\n"); return 1; } int new_width = width * 0.8; //按照0.8倍率,可改为其他比率 int new_height = height * 0.8; BYTE* new_buffer = (BYTE*)malloc(new_width * new_height * bpp); if (!new_buffer) { printf("Failed to allocate memory\n"); return 1; } float x_ratio = (width - 1) / (float)(new_width - 1); float y_ratio = (height - 1) / (float)(new_height - 1); float x_diff, y_diff; int index, offset = 0; BYTE* p1, * p2, * p3, * p4; for (int y = 0; y < new_height; ++y) { for (int x = 0; x < new_width; ++x) { index = y * new_width + x; float x_full = x * x_ratio; float y_full = y * y_ratio; int x1 = (int)x_full; int y1 = (int)y_full; int x2 = x1 + 1; int y2 = y1 + 1; x_diff = x_full - x1; y_diff = y_full - y1; p1 = &buffer[(y1 * width + x1) * bpp]; p2 = &buffer[(y1 * width + x2) * bpp]; p3 = &buffer[(y2 * width + x1) * bpp]; p4 = &buffer[(y2 * width + x2) * bpp]; for (int i = 0; i < bpp; ++i) { new_buffer[index * bpp + i] = (BYTE)( p1[i] * (1 - x_diff) * (1 - y_diff) + p2[i] * (x_diff) * (1 - y_diff) + p3[i] * (y_diff) * (1 - x_diff) + p4[i] * (x_diff * y_diff)); } } } stbi_write_png("new_test.png", new_width, new_height, bpp, new_buffer, new_width * bpp); stbi_image_free(buffer); free(new_buffer); return 0; } ``` 以上是图片算法的C语言代码,我们使用了STB_IMAGE库来加载图片和保存图片。该代码中使用的是双线性插值算法,通过计算原图像上的四个点的像素值,得出目标像素的像素值。在实现过程中,我们使用了两个循环,遍历目标图片的所有像素,为每个像素计算相应的值,并存储到新图像的缓冲区中。最后,将新图像保存到磁盘中。需要注意的是,该算法中只针对灰度图像或RGB图像进行处理,对于其他类型图像可能需要做出适当更改。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值