使用VC++设计程序:实现常见的三种图像插值算法:最近邻插值,双线性插值,立方卷积插值

图像放大的三种插值算法

获取源工程可访问gitee可在此工程的基础上进行学习。
该工程的其他文章:
01- 一元熵值、二维熵值
02- 图像平移变换,图像缩放、图像裁剪、图像对角线镜像以及图像的旋转
03-邻域平均平滑算法、中值滤波算法、K近邻均值滤波器
04-分段线性变换,直方图均衡化、锐化处理
05-基于拉普拉斯算子、Canny的边缘检测功能、实现Otsu分割方法

文章目录

  • 图像放大的三种插值算法
    • 实验内容
    • 一、 插值算法的原理
      • 1. 最近邻插值
      • 2. 双线性插值
      • 3. 立方卷积插值
    • 二、 实验代码与现象图
      • 1. 最近邻插值
      • 2. 双线性插值
      • 3.立方卷积插值

实验内容

B–(3)研究放大图像时使用的插值算法,使用VC++实现常见的三种图像插值算法:最近邻插值,双线性插值,立方卷积插值。通过实验验证其效果。

一、 插值算法的原理

1. 最近邻插值

最近邻插值是一种简单的插值方法,它选择离目标位置最近的已知像素值。对于二维图像上的插值,最近邻插值的公式为:

I new ( x , y ) = I old ( round ( x ) , round ( y ) ) I_{\text{new}}(x, y) = I_{\text{old}}\left(\text{round}(x), \text{round}(y)\right) Inew(x,y)=Iold(round(x),round(y))

其中:

  • $ I_{\text{new}}(x, y) $ 是目标位置的新像素值。
  • $ I_{\text{old}}(x’, y’) $是原始图像中最近邻的已知像素值,其中 $ (x’, y’) $ 由 $ (x, y) $ 四舍五入得到。

2. 双线性插值

双线性插值考虑了目标位置周围的四个最近的已知像素值,并根据其相对位置进行加权平均。对于二维图像上的插值,双线性插值的公式为:

I new ( x , y ) = ( 1 − α ) ( 1 − β ) I old ( x 1 , y 1 ) + α ( 1 − β ) I old ( x 2 , y 1 ) + ( 1 − α ) β I old ( x 1 , y 2 ) + α β I old ( x 2 , y 2 ) I_{\text{new}}(x, y) = (1 - \alpha)(1 - \beta)I_{\text{old}}(x_1, y_1) + \alpha(1 - \beta)I_{\text{old}}(x_2, y_1) + (1 - \alpha)\beta I_{\text{old}}(x_1, y_2) + \alpha \beta I_{\text{old}}(x_2, y_2) Inew(x,y)=(1α)(1β)Iold(x1,y1)+α(1β)Iold(x2,y1)+(1α)βIold(x1,y2)+αβIold(x2,y2)
其中:

  • $ I_{\text{new}}(x, y) $ 是目标位置的新像素值。
  • $ I_{\text{old}}(x_i, y_i)$是原始图像中四个最近邻的已知像素值,其中 $ (x_i, y_i) $ 是目标位置的四个相邻像素的坐标。
  • $ \alpha = x - x_1 $ 和 $ \beta = y - y_1$。

双线性插值算法涉及到目标位置周围的四个最近邻的已知像素值。这四个最近邻的像素可以通过目标位置坐标的整数部分和小数部分来确定。假设目标位置的坐标为 ((x, y)),则这四个最近邻的坐标可以表示为 ((x_1, y_1), (x_2, y_1), (x_1, y_2), (x_2, y_2)),其中:

  • x 1 = ⌊ x ⌋ x_1 = \lfloor x \rfloor x1=x x x x的整数部分;
  • x 2 = ⌈ x ⌉ x_2 = \lceil x \rceil x2=x x x x的整数部分加一;
  • y 1 = ⌊ y ⌋ y_1 = \lfloor y \rfloor y1=y y y y 的整数部分;
  • y 2 = ⌈ y ⌉ y_2 = \lceil y \rceil y2=y y y y 的整数部分加一。

这四个最近邻的坐标形成了一个矩形区域,如下图所示:

(x1, y1) ---------- (x2, y1)
 |                        |
 |                        |
 |                        |
(x1, y2) ---------- (x2, y2)

3. 立方卷积插值

立方卷积插值考虑了目标位置周围的八个最近的已知像素值,并使用立方卷积核进行加权平均。对于二维图像上的插值,立方卷积插值的公式较为复杂,其中涉及到立方卷积核的权重计算。

I new ( x , y ) = ∑ i = − 1 2 ∑ j = − 1 2 w ( i , j ) I old ( x + i , y + j ) I_{\text{new}}(x, y) = \sum_{i = -1}^{2} \sum_{j = -1}^{2} w(i, j)I_{\text{old}}(x + i, y + j) Inew(x,y)=i=12j=12w(i,j)Iold(x+i,y+j)

其中:

  • $ I_{\text{new}}(x, y) $ 是目标位置的新像素值。
  • $I_{\text{old}}(x + i, y + j) $ 是原始图像中八个最近邻的已知像素值,其中 $i $ 和 $j $取值为 -1, 0, 1, 2。
  • $ w(i, j) $ 是立方卷积核的权重,通常采用一些特定的卷积核形式,如 Bicubic 插值。

在立方卷积插值中,涉及到的是8个点。以下是正确的描述:

对于立方卷积插值,考虑目标位置 ( x , y ) (x, y) (x,y) 周围的8个最近邻点,其坐标可以表示为 ( x i , y j ) (x_i, y_j) (xi,yj),其中:

x i = x − 1 , 0 , 1 , 2 x_i = x - 1, 0, 1, 2 xi=x1,0,1,2
y j = y − 1 , 0 , 1 , 2 y_j = y - 1, 0, 1, 2 yj=y1,0,1,2

这样,可以得到一个包含8个点的矩阵,如下所示:

(x-1, y-1)  |  (x, y-1)  |  (x+1, y-1)  |  (x+2, y-1)
(x-1, y)    |  (x, y)    |  (x+1, y)    |  (x+2, y)
(x-1, y+1)  |  (x, y+1)  |  (x+1, y+1)  |  (x+2, y+1)
(x-1, y+2)  |  (x, y+2)  |  (x+1, y+2)  |  (x+2, y+2)

这里的坐标形成了一个4x4的矩阵,但在立方卷积插值中,只需要考虑中间的8个点。这8个点的坐标将用于计算插值权重。

二、 实验代码与现象图

1. 最近邻插值

void CImageProcessingView::OnGeoResizing()
{
 // 实验 图像缩放
 //MessageBox("请在这里添加图像缩放的代码");
 // 获得当前文档对象
CImageProcessingDoc* pDoc = GetDocument();
 // 判断图像是否已被加载
 if( pDoc->m_pDibInit->IsEmpty() )
 {
  MessageBox("图像未加载");
  return;
 }
 int width = pDoc->m_pDibInit->GetWidth();
 int height = pDoc->m_pDibInit->GetHeight();
 int bitCount = pDoc->m_pDibInit->GetBitCount();
 
 // 将 m_pDibInit 拷贝至 m_pDibTest
 pDoc->m_pDibTest->CloneDib(pDoc->m_pDibInit);

 // 考虑将图像放大两倍的情况 
 float nResizing = 2;

 // 获得新的图像高度
 int newWidth = width*nResizing;
 int newHeight = height*nResizing;
 pDoc->m_pDibTest->SetWidthHeight(newWidth, newHeight);
 //*****************************图像的插值1最近邻插值算法************//
 int i=0;
 int j=0;
 float src_x, src_y;
 RGBQUAD Quad1;
 for(i=0;i<newWidth;i++)
  for(j=0;j<newHeight;j++)
  {
   src_x = (i / nResizing) + 0.5; //四舍五入
   src_y = (j / nResizing) + 0.5;
   Quad1=pDoc->m_pDibInit->GetPixelColor(src_x,src_y);
   pDoc->m_pDibTest->SetPixelColor(i,j,&Quad1);
  }
 // 交换 m_pDibInit 与 m_pDibTest 指针
 CDib* pTmp = pDoc->m_pDibInit;
 pDoc->m_pDibInit = pDoc->m_pDibTest;
 pDoc->m_pDibTest = pTmp; 
 
 // 设置脏标记
 pDoc->SetModifiedFlag(TRUE);
 // 更新视图
 pDoc->UpdateAllViews(NULL);
}

实验现象.
图像放大2倍
在这里插入图片描述

2. 双线性插值

 int i=0;
 int j=0;
 for(i=0;i<newWidth;i++)
  for(j=0;j<newHeight;j++)
  {
   int x=i/nResizing, y = j / nResizing;  //定位,找周围四个点
   float u = i / nResizing,v = j / nResizing; //注意这个类型

   u =u - x;          //所占权重系数
   v =v - y;

   int gray0 , gray1 , gray2 , gray3 , GRAY ;
   gray0 = pDoc->m_pDibInit->GetPixelGray(x, y);
   //边界处理 非补零
   if (x + 1 < 256)
    gray1 = pDoc->m_pDibInit->GetPixelGray(x+1, y);
   else
    gray1 = pDoc->m_pDibInit->GetPixelGray(x, y);
   if (y + 1 < 256)
    gray2 = pDoc->m_pDibInit->GetPixelGray(x, y + 1);
   else
    gray2 = pDoc->m_pDibInit->GetPixelGray(x, y);
   if (x + 1 < 256 && y + 1 < 256)
    gray3 = pDoc->m_pDibInit->GetPixelGray(x + 1, y + 1);
   else
    gray3 = pDoc->m_pDibInit->GetPixelGray(x, y);
   GRAY = (1 - u) * (1 - v) * gray0 + (1 - u) * v * gray2 + u * (1 - v) * gray1 + u * v * gray3;
   pDoc->m_pDibTest->SetPixelGray(i, j, GRAY);
  }
 // 交换 m_pDibInit 与 m_pDibTest 指针
 CDib* pTmp = pDoc->m_pDibInit;
 pDoc->m_pDibInit = pDoc->m_pDibTest;
 pDoc->m_pDibTest = pTmp;

实验效果图.

在这里插入图片描述

3.立方卷积插值

int a = -0.5;   //系数
 for(int i=0;i<newWidth;i++)
  for (int j = 0; j < newHeight; j++)
  {
   int x[4] = { 0 }, y[4] = { 0 };
   x[1] = i / nResizing, y[1] = j / nResizing;  //找点,(1,1)位置
   x[0] = x[1] - 1; y[0] = y[1] - 1;
   x[2] = x[1] + 1; y[2] = y[1] + 1;
   x[3] = x[1] + 2; y[3] = y[1] + 2;

   float u = (i) / nResizing, v = (j) / nResizing;

   u =u - x[1];   //所占权重系数(大于0小于1)
   v =v - y[1];

   int gray[4][4] = { 0 };
   float w_x[4] = { 0 }, w_y[4] = { 0 };
   w_x[0] = 1 + u; w_x[1] = u; w_x[2] = 1 - u; w_x[3] = 2 - u;
   w_y[0] = 1 + v; w_y[1] = v; w_y[2] = 1 - v; w_y[3] = 2 - v;

   float W_x[4] = { 0 }, W_y[4] = { 0 };
   for (int k = 0; k < 4; k++)    //计算x和y的权重
   {
    if (w_x[k] <= 1 && w_x[k] >= -1)
     W_x[k] = (a + 2) * pow(w_x[k], 3) - (a + 3) * pow(w_x[k], 2) + 1;
    else if (w_x[k] > 1 && w_x[k] < 2) 
     W_x[k] = a * pow(w_x[k], 3) - 5 * a * pow(w_x[k], 2) + 8 * a * w_x[k] - 4 * a;
    else
     W_x[k] = 0;

    if (w_y[k] <= 1 && w_y[k] >= -1)
     W_y[k] = (a + 2) * pow(w_y[k], 3) - (a + 3) * pow(w_y[k], 2) + 1;
    else if (w_y[k] > 1 && w_y[k] < 2)
     W_y[k] = a * pow(w_y[k], 3) - 5 * a * pow(w_y[k], 2) + 8 * a * w_y[k] - 4 * a;
    else
     W_y[k] = 0;
   }

   for(int k=0;k<4;k++)
    for (int m = 0; m < 4; m++)
    {
     if (x[k] > 0 && x[k]<height && y[m] > 0 && y[m] < width) //判断边界
      gray[k][m] = pDoc->m_pDibInit->GetPixelGray(x[k], y[m]);
     else
      gray[k][m] = 0;
    }

   int Pix_gray = 0;  //最终计算得到的灰度值
   for(int k=0;k<4;k++)
    for (int m = 0; m < 4; m++)
    {
     Pix_gray += gray[k][m] * W_x[k] * W_y[m];
    }
   pDoc->m_pDibTest->SetPixelGray(i, j, Pix_gray);
  }

实验效果图.
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/189672.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

VSCode 警告:v-on event ‘@toggleClick‘ must be hyphenated

&#x1f680; 作者主页&#xff1a; 有来技术 &#x1f525; 开源项目&#xff1a; youlai-mall &#x1f343; vue3-element-admin &#x1f343; youlai-boot &#x1f33a; 仓库主页&#xff1a; Gitee &#x1f4ab; Github &#x1f4ab; GitCode &#x1f496; 欢迎点赞…

【Spring篇】spring核心——AOP面向切面编程

目录 想要彻底理解AOP&#xff0c;我觉得你的先要了解框架的模块化思想&#xff0c;为此先记录框架在讲AOP 什么是java框架&#xff1f;为什么要出现框架&#xff1f; 我总结以下七点来讲述和帮助理解java框架思想 什么是AOP&#xff1f; 如何理解上面这句话呢&#xff1…

【开源】基于Vue+SpringBoot的农家乐订餐系统

项目编号&#xff1a; S 043 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S043&#xff0c;文末获取源码。} 项目编号&#xff1a;S043&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 用户2.2 管理员 三、系统展示四、核…

Redis面试题:redis做为缓存,数据的持久化是怎么做的?两种持久化方式有什么区别呢?这两种方式,哪种恢复的比较快呢?

目录 面试官&#xff1a;redis做为缓存&#xff0c;数据的持久化是怎么做的&#xff1f; 面试官&#xff1a;这两种持久化方式有什么区别呢&#xff1f; 面试官&#xff1a;这两种方式&#xff0c;哪种恢复的比较快呢&#xff1f; 面试官&#xff1a;redis做为缓存&#xff…

【【Linux编程介绍之关键配置和常用用法】】

Linux编程介绍之关键配置和常用用法 Hello World ! 我们所说的编写代码包括两部分&#xff1a;代码编写和编译&#xff0c;在Windows下可以使用Visual Studio来完成这两部&#xff0c;可以在 Visual Studio 下编写代码然后直接点击编译就可以了。但是在 Linux 下这两部分是分开…

文章解读与仿真程序复现思路——电网技术EI\CSCD\北大核心《前景导向的主动配电网智能储能软开关规划方法》

这个标题涉及到电力系统中的主动配电网&#xff08;Active Distribution Network&#xff09;以及与之相关的智能储能软开关的规划方法。下面是对标题中各个关键词的解释&#xff1a; 前景导向的&#xff08;Future-oriented&#xff09;&#xff1a; 这表明该方法是以未来发展…

球面的表面积

此推导需要用到重积分的知识&#xff0c;另外关于曲面的面积公式可以看我之前的博客

线性表,也是Java中数组的知识点!

线性表定义&#xff1a; 由n (n≥0)个数据特性相同的元素构成的有限序列称为线性表&#xff0c;(n0)的时候被称为空表。 线性表的顺序表示 线性表的顺序存储又被称为顺序表 优点 无需为表示表中元素之间的逻辑关系而增加额外的存储空间可以随意读取任意位置的元素 缺点 插入…

【Mybatis】预编译/即时sql 数据库连接池

回顾 Mybatis是一个持久层框架.有两种方式(这两种方式可以共存) 1.注解 2.xml 一.传递参数 以使用#{} 来接受参数为例 (以上两种方式一样适用的) 1)传递单个参数 #{} 可以为任意名称 2)多个参数 默认的参数名称就是接口方法声明的形参 3)参数为对象 默认给每个对象的每个属性都…

中海油“海安杯”一站到底知识竞赛真的很有特色

中海油“海安杯”一站到底知识竞赛规格高&#xff0c;赛制复杂&#xff0c;天纵知识竞赛系统为此次知识竞赛提供了软件支持。本次竞赛设置选手区和擂台区两个区域。比赛共分为五个轮次&#xff0c;五个轮次选手区所有参赛选手均需答题。 第一轮&#xff1a;“脱颖而出” 所有参…

ruoyi-plus-vue docker 部署

本文以 ruoyi-vue-plus 5.x docker 部署为基础 安装虚拟机 部署文档 安装docker 安装docker 安装docker-compose 配置idea环境 上传 /doicker 文件夹 到服务器&#xff1b;赋值 777权限 chmod -R 777 /docker idea构建 jar 包 利用 idea 构建镜像; 创建基础服务 docker…

深入了解 Pinia:现代 Vue 应用的状态管理利器

&#x1f90d; 前端开发工程师&#xff08;主业&#xff09;、技术博主&#xff08;副业&#xff09;、已过CET6 &#x1f368; 阿珊和她的猫_CSDN个人主页 &#x1f560; 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 &#x1f35a; 蓝桥云课签约作者、已在蓝桥云…

2017年11月16日 Go生态洞察:Go用户调查深度解析

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页——&#x1f405;&#x1f43e;猫头虎的博客&#x1f390; &#x1f433; 《面试题大全专栏》 &#x1f995; 文章图文…

CSDN助手:一键下载CSDN博客:高效保存,随时阅读

文章目录 &#x1f4d6; 介绍 &#x1f4d6;&#x1f3e1; 环境 &#x1f3e1;&#x1f4d2; 使用方法 &#x1f4d2;⚓️ 相关链接 ⚓️ &#x1f4d6; 介绍 &#x1f4d6; 这是我自己无聊的时候写的一个应用&#xff0c;以前UI有点丑&#xff0c;这次重写了一下UI 功能如下 …

使用VC++设计程序对一幅256级灰度图像进行全局固定阈值分割、自适应阈值分割

图像分割–全局固定阈值分割、自适应阈值分割 获取源工程可访问gitee可在此工程的基础上进行学习。 该工程的其他文章&#xff1a; 01- 一元熵值、二维熵值 02- 图像平移变换&#xff0c;图像缩放、图像裁剪、图像对角线镜像以及图像的旋转 03-邻域平均平滑算法、中值滤波算法、…

Linux python安装 虚拟环境 virtualenv,以及 git clone的 文件数据, 以及 下资源配置

根目录创建 venvs 文件夹 sudo mkdir /venvs 进入 /venvs 目录 cd /venvsp 创建虚拟环境&#xff0c;前提要按照 python3 安装 的 命令 sudo apt install python3 sudo python3 -m venv 虚拟环境名 激活虚拟环境 source /venvs/zen-venv/bin/activate 安装flask pip install fl…

Java + openCV更换证件照背景色

最近在小红书上看到很多更换证件照背景色的需求&#xff0c;联想到以前自己也更换过证件照背景色而且还是付费的&#xff0c;碰巧最近在看一本书《JavaOpenCV高效入门》&#xff0c;于是查找资料&#xff0c;找到了通过技术解决这个需求的办法。 先看效果图&#xff08;图片来自…

每日一题2023.11.26——个位数统计【PTA】

题目要求&#xff1a; 输入格式&#xff1a; 每个输入包含 1 个测试用例&#xff0c;即一个不超过 1000 位的正整数 N。 输出格式&#xff1a; 对 N 中每一种不同的个位数字&#xff0c;以 D:M 的格式在一行中输出该位数字 D 及其在 N 中出现的次数 M。要求按 D 的升序输出。…

设备树是什么?

设备树&#xff1a; 设备树DTS(Device Tree Source) 描述设备信息的独立的文件。 为什么要引入设备树&#xff1f; 随着芯片的发展&#xff0c;Linux内核中就包含着越来越多这些描述设备的代码&#xff0c;导致Linux内核代码会很臃肿。因此引入了设备树文件&#xff0c;从…

Runloop解析

RunLoop 前言 ​ 本文介绍RunLoop的概念&#xff0c;并使用swift和Objective-C来描述RunLoop机制。 简介 ​ RunLoop——运行循环&#xff08;死循环&#xff09;&#xff0c;它提供了一个事件循环机制在程序运行过程中处理各种事件&#xff0c;例如用户交互、网络请求、定…