PotatoPie 4.0 实验教程(29) —— FPGA实现摄像头图像均值滤波处理

news2025/1/11 11:53:29

图像的均值滤波简介

图像均值滤波处理是一种常见的图像处理技术,用于降低图像中噪声的影响并平滑图像。该方法通过在图像中滑动一个固定大小的窗口(通常是一个正方形或矩形),将窗口中所有像素的值取平均来计算窗口中心像素的新值。这种操作会使图像中的高频噪声减弱,并且可以使图像的轮廓变得更加模糊,从而使得图像更加平滑。

图像均值滤波的理论依据是基于局部区域的像素值平均化。当图像中的一个像素值受到噪声的影响时,它的值可能会与其周围像素的值不一致。通过取局部区域像素值的平均,可以有效地减小噪声的影响,从而得到更加平滑的图像。

数学上,对于一个大小为 N×N 的滤波器(通常称为卷积核),图像中的每个像素被替换为其周围 N×N 区域中所有像素值的平均值。滤波器的大小决定了平滑效果的程度,较大的滤波器会导致更大程度的平滑,但可能会损失图像的细节信息。

均值滤波步骤和算法实现

均值滤波是一种常用的图像平滑处理方法,其步骤如下:

  1. 定义滤波模板: 通常使用一个固定大小的滤波模板(例如3×3或5×5)。

  2. 对每个像素应用滤波: 将模板中心的像素放置在图像的每个像素位置,并计算模板覆盖的像素的平均值。

  3. 更新图像值: 将计算得到的平均值作为当前像素的新值。

  4. 处理图像边界: 对于图像边界的像素,由于模板无法完全覆盖,需要根据特定的策略来处理,例如忽略边界像素、使用镜像边界像素或补零处理等。

  5. 输出结果: 输出经过均值滤波处理后的图像。

这种滤波方法有助于去除图像中的噪声和细节,使图像变得更加平滑。

python实现图像的均值滤波处理源码

import cv2
import numpy as np
import os
import matplotlib.pyplot as plt

# 获取当前脚本所在目录
current_directory = os.path.dirname(os.path.abspath(__file__))

# 构建图像文件的完整路径
image_path = os.path.join(current_directory, 'Lena.jpg')

# 读取图像
image_in = cv2.imread(image_path)

# 将彩色图像转换为灰度图像
image_gray = cv2.cvtColor(image_in, cv2.COLOR_BGR2GRAY)

# 获取图像尺寸
row, col = image_gray.shape

# 将灰度图像转换为浮点型
image_gray = image_gray.astype(float)

# 初始化用于存储均值滤波结果的图像
average_image = np.zeros_like(image_gray)

# 对灰度图像应用均值滤波
for i in range(1, row-1):
    for j in range(1, col-1):
        average_image[i, j] = (image_gray[i-1, j-1] + image_gray[i-1, j] + image_gray[i-1, j+1] +
                               image_gray[i, j-1] + image_gray[i, j] + image_gray[i, j+1] +
                               image_gray[i+1, j-1] + image_gray[i+1, j] + image_gray[i+1, j+1]) / 9

# 添加椒盐噪声到灰度图像
noise_salt_pepper = np.random.choice([0, 1], size=(row, col), p=[0.99, 0.01])  # 进一步降低椒盐噪声密度
image_gray_salt_pepper = image_gray.copy()
image_gray_salt_pepper[noise_salt_pepper == 1] = 255  # 将椒盐噪声点的灰度值设为255

# 初始化用于存储均值滤波结果的图像
average_image_salt_pepper = np.zeros_like(image_gray)

# 对添加椒盐噪声后的灰度图像应用均值滤波
for i in range(1, row-1):
    for j in range(1, col-1):
        average_image_salt_pepper[i, j] = (image_gray_salt_pepper[i-1, j-1] + image_gray_salt_pepper[i-1, j] + image_gray_salt_pepper[i-1, j+1] +
                                           image_gray_salt_pepper[i, j-1] + image_gray_salt_pepper[i, j] + image_gray_salt_pepper[i, j+1] +
                                           image_gray_salt_pepper[i+1, j-1] + image_gray_salt_pepper[i+1, j] + image_gray_salt_pepper[i+1, j+1]) / 9

# 添加高斯噪声到灰度图像
noise_gaussian = np.random.normal(0, 0.08, (row, col))  # 进一步降低高斯噪声的强度
image_gray_gaussian = image_gray + noise_gaussian * 255
image_gray_gaussian = np.clip(image_gray_gaussian, 0, 255)

# 初始化用于存储均值滤波结果的图像
average_image_gaussian = np.zeros_like(image_gray)

# 对添加高斯噪声后的灰度图像应用均值滤波
for i in range(1, row-1):
    for j in range(1, col-1):
        average_image_gaussian[i, j] = (image_gray_gaussian[i-1, j-1] + image_gray_gaussian[i-1, j] + image_gray_gaussian[i-1, j+1] +
                                         image_gray_gaussian[i, j-1] + image_gray_gaussian[i, j] + image_gray_gaussian[i, j+1] +
                                         image_gray_gaussian[i+1, j-1] + image_gray_gaussian[i+1, j] + image_gray_gaussian[i+1, j+1]) / 9

# 显示原始灰度图像和均值滤波后的图像
plt.figure(figsize=(12, 10))

plt.subplot(3, 2, 1)
plt.imshow(image_gray, cmap='gray')
plt.title('Original Gray Image')

plt.subplot(3, 2, 2)
plt.imshow(average_image, cmap='gray')
plt.title('Average Image')

plt.subplot(3, 2, 3)
plt.imshow(image_gray_salt_pepper, cmap='gray')
plt.title('Salt & Pepper Image')

plt.subplot(3, 2, 4)
plt.imshow(average_image_salt_pepper, cmap='gray')
plt.title('Average Salt & Pepper Image')

plt.subplot(3, 2, 5)
plt.imshow(image_gray_gaussian, cmap='gray')
plt.title('Gaussian Image')

plt.subplot(3, 2, 6)
plt.imshow(average_image_gaussian, cmap='gray')
plt.title('Average Gaussian Image')

plt.tight_layout()
plt.show()

这段代码实现了以下功能:

  1. 读取一张彩色图像并将其转换为灰度图像。
  2. 对灰度图像应用均值滤波,以平滑图像并降低噪声。
  3. 添加了椒盐噪声到灰度图像中,模拟图像中的随机噪声。
  4. 对添加了椒盐噪声的灰度图像应用均值滤波,以降低椒盐噪声对图像的影响。
  5. 添加了高斯噪声到灰度图像中,模拟图像中的连续噪声。
  6. 对添加了高斯噪声的灰度图像应用均值滤波,以降低高斯噪声对图像的影响。
  7. 显示原始灰度图像、均值滤波后的图像、添加椒盐噪声后的图像、添加高斯噪声后的图像以及它们各自的均值滤波结果。

这段代码可以用于图像处理中的噪声去除和平滑处理。

MATLAB实现图像的均值滤波处理源码

clear;  % 清空工作区变量
clear all;  % 清除所有变量
clc;  % 清空命令窗口

% 获取当前脚本所在目录
current_directory = fileparts(mfilename('fullpath'));

% 构建图像文件的完整路径
image_path = fullfile(current_directory, 'Lena.jpg');

% 读取图像
image_in = imread(image_path);

% 将彩色图像转换为灰度图像
image_gray = rgb2gray(image_in);

% 获取图像尺寸
[row,col] = size(image_gray);

% 将灰度图像转换为双精度类型
image_gray = im2double(image_gray); 

% 初始化用于存储均值滤波结果的图像
average_image = zeros(row,col);

% 对灰度图像应用均值滤波
for i = 2:1:row-1
    for j = 2:1:col-1
        average_image(i,j) = (...
            image_gray(i-1,j-1) + image_gray(i-1,j) + image_gray(i-1,j+1) + ...
            image_gray(i,j-1)   + image_gray(i,j)   + image_gray(i,j+1)   + ...
            image_gray(i+1,j-1) + image_gray(i+1,j) + image_gray(i+1,j+1)) / 9;
    end
end

% 添加椒盐噪声到灰度图像
image_gray_salt_pepper = imnoise(image_gray,'salt & pepper',0.05); 
image_gray_salt_pepper = im2double(image_gray_salt_pepper); 

% 初始化用于存储均值滤波结果的图像
average_image_salt_pepper = zeros(row,col);

% 对添加椒盐噪声后的灰度图像应用均值滤波
for i = 2:1:row-1
    for j = 2:1:col-1
        average_image_salt_pepper(i,j) = (...
            image_gray_salt_pepper(i-1,j-1) + image_gray_salt_pepper(i-1,j) + image_gray_salt_pepper(i-1,j+1) + ...
            image_gray_salt_pepper(i,j-1)   + image_gray_salt_pepper(i,j)   + image_gray_salt_pepper(i,j+1)   + ...
            image_gray_salt_pepper(i+1,j-1) + image_gray_salt_pepper(i+1,j) + image_gray_salt_pepper(i+1,j+1)) / 9;
    end
end

% 添加高斯噪声到灰度图像
image_gray_gaussian = imnoise(image_gray,'gaussian',0.05); 
image_gray_gaussian = im2double(image_gray_gaussian); 

% 初始化用于存储均值滤波结果的图像
average_image_gaussian = zeros(row,col);

% 对添加高斯噪声后的灰度图像应用均值滤波
for i = 2:1:row-1
    for j = 2:1:col-1
        average_image_gaussian(i,j) = (...
            image_gray_gaussian(i-1,j-1) + image_gray_gaussian(i-1,j) + image_gray_gaussian(i-1,j+1) + ...
            image_gray_gaussian(i,j-1)   + image_gray_gaussian(i,j)   + image_gray_gaussian(i,j+1)   + ...
            image_gray_gaussian(i+1,j-1) + image_gray_gaussian(i+1,j) + image_gray_gaussian(i+1,j+1)) / 9;
    end
end

% 显示原始灰度图像和均值滤波后的图像
figure
subplot(321);
imshow(image_gray), title('原始灰度图像');
subplot(322);
imshow(average_image), title('均值滤波后的图像');
subplot(323);
imshow(image_gray_salt_pepper), title('添加椒盐噪声后的灰度图像');
subplot(324);
imshow(average_image_salt_pepper), title('添加椒盐噪声后的均值滤波图像');
subplot(325);
imshow(image_gray_gaussian), title('添加高斯噪声后的灰度图像');
subplot(326);
imshow(average_image_gaussian), title('添加高斯噪声后的均值滤波图像');

这段代码实现了图像的均值滤波处理,具体功能如下:

  1. 清空工作区变量和命令窗口: clearclear allclc函数用于清空工作区变量和命令窗口,确保工作环境清晰。

  2. 获取当前脚本所在目录: filepartsmfilenamefullpath函数用于获取当前脚本所在的目录,并通过fullfile函数构建图像文件的完整路径,以便读取图像文件。

  3. 读取图像: imread函数读取名为”Lena.jpg”的图像文件,并将图像数据存储在变量image_in中。

  4. 转换为灰度图像: rgb2gray函数将彩色图像转换为灰度图像,以便后续处理。

  5. 获取图像尺寸: size函数获取灰度图像的尺寸,包括行数和列数。

  6. 转换为双精度类型: im2double函数将灰度图像转换为双精度类型,以便进行数学运算。

  7. 初始化存储均值滤波结果的图像矩阵: 创建一个与输入图像大小相同的矩阵average_image,用于存储均值滤波后的图像。

  8. 应用均值滤波: 使用两层嵌套的循环遍历图像中的每个像素,并对其应用3×3的均值滤波模板,得到均值滤波后的像素值。

  9. 添加噪声并进行均值滤波处理: 将灰度图像分别添加了椒盐噪声和高斯噪声,并对噪声图像分别进行均值滤波处理。

  10. 显示结果图像: 将原始灰度图像和三种处理后的图像(原始灰度图像、椒盐噪声+均值滤波、高斯噪声+均值滤波)以子图形式展示在一个大图中。

FPGA工程分析

工程层次图

demo18相比,只是多了一个img_mean_fltr的模块,也就是下面这一段代码,在从SDRAM读出来之后,经它处理后再输出hdmi_tx模块。

img_mean_fltr u_mean_fltr
(
    .i_clk(clk_pixel),
    .i_rst_n(sys_rst_n),
    .i_hs(VGA_HS),
    .i_vs(VGA_VS),
    .i_de(VGA_DE),
    .i_r(VGA_RGB[23:16]),
    .i_g(VGA_RGB[15:8] ),
    .i_b(VGA_RGB[7:0]  ),         
    .o_hs(mean_hs),
    .o_vs(mean_vs),
    .o_de(mean_de),   
    .o_r(mean_data[23:16]),
    .o_g(mean_data[15:8] ),
    .o_b(mean_data[7:0]  ) 
);

img_mean_fltr模块代码分析

从层次图可以看到这个模块的结果跟前面的 《PotatoPie 4.0 实验教程(28) —— FPGA实现sobel算子对摄像头图像进行边缘提取》整体结构是一样的。

首先例化图像缓冲模块,用于将图像从一个时钟一个像素转为一个时钟输出三行三列9个像素。以R通道为例

img_buf u_r_buf
(
  .i_clk      (i_clk        ),
  .i_rst_n    (i_rst_n      ),
  .i_de      (i_de        ),
  .i_data      (i_r        ),
  .o_de      (          ),
  .o_p11    (r_p11      ),
  .o_p12    (r_p12      ),
  .o_p13    (r_p13      ),  
  .o_p21    (r_p21      ),
  .o_p22    (r_p22      ),
  .o_p23    (r_p23      ),    
  .o_p31    (r_p31      ),
  .o_p32    (r_p32      ),
  .o_p33    (r_p33      )
);

代码后面又分别对G、B通道进行了行缓存的例化。

然后计算中心像素周围的另外8个像素的和,以R通道为例,

sum_r <= r_p11 + r_p12 + r_p13 + r_p21 + r_p23 + r_p31 + r_p32 + r_p33;

最后进行求平均值处理,右移3就相当于除以8。

avg_r <= sum_r >> 3;

管脚约束

与PotatoPie 4.0 实验教程(18) —— FPGA实现OV5640摄像头采集以SDRAM作为显存进行HDMI输出显示相同,不作赘述。

时序约束

与PotatoPie 4.0 实验教程(18) —— FPGA实现OV5640摄像头采集以SDRAM作为显存进行HDMI输出显示相同,不作赘述。

实验效果

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

GateWay具体的使用之全链路跟踪TraceId日志

1.创建全局过滤器&#xff0c;在请求头上带入traceId参数&#xff0c;穿透到下游服务. package com.by.filter;import cn.hutool.core.collection.CollUtil; import cn.hutool.core.util.IdUtil; import cn.hutool.core.util.ObjectUtil; import cn.hutool.jwt.JWTValidator;…

【Kotlin】Channel简介

1 前言 Channel 是一个并发安全的阻塞队列&#xff0c;可以通过 send 函数往队列中塞入数据&#xff0c;通过 receive 函数从队列中取出数据。 当队列被塞满时&#xff0c;send 函数将被挂起&#xff0c;直到队列有空闲缓存&#xff1b;当队列空闲时&#xff0c;receive 函数将…

vue3 vite 路由去中心化(modules文件夹自动导入router)

通过路由去中心化可实现多人写作开发&#xff0c;不怕文件不停修改导致的冲突&#xff0c;modules中的文件可自动导入到index.js中 // 自动导入模块 const files import.meta.globEager(./modules/**.js); const modules {} for (const key in files) {modules[key.replace…

前端工程化Vue使用Node.js设置国内高速npm镜像源(踩坑记录版)

前端工程化Vue使用Node.js设置国内高速npm镜像源&#xff08;踩坑记录版&#xff09; 此篇仅为踩坑记录&#xff0c;并未成功更换高速镜像源&#xff0c;实际解决方法见文末跳转链接。 1.自身源镜像 自身镜像源创建Vue项目下载速度感人 2.更改镜像源 2.1 通过命令行配置 前提…

在Redux Toolkit中使用redux-persist进行状态持久化

在 Redux Toolkit 中使用 redux-persist 持久化插件的步骤如下: 安装依赖 npm install redux-persist配置 persistConfig 在 Redux store 配置文件中(例如 rootReducer.js)&#xff0c;导入必要的模块并配置持久化选项: import { combineReducers } from redux; import { p…

【MySQL关系型数据库】基本命令、配置、连接池

目录 MySQL数据库 第一章 1、什么是数据库 2、数据库分类 3、不同数据库的特点 4、MySQL常见命令&#xff1a; 5、MySQL基本语法 第二章 1、MySQL的常见数据类型 1、数值类型 2、字符类型 3、时间日期类型 2、SQL语句分类 1、DDL&#xff08;数据定义语言&#x…

mysql-sql-练习题-2-窗口函数

窗口函数 访问量max sum建表窗口函数连接 直播间人数 第1、3名建表排名sum 访问量max sum 每个用户截止到每月为止&#xff0c;最大单月访问次数&#xff0c;累计到该月的总访问次数 建表 create table visit(uid1 varchar(5) comment 用户id,month1 varchar(10) comment 月…

28.Gateway-网关过滤器

GatewayFilter是网关中提供的一种过滤器&#xff0c;可以多进入网关的请求和微服务返回的响应做处理。 GatewayFilter(当前路由过滤器&#xff0c;DefaultFilter) spring中提供了31种不同的路由过滤器工厂。 filters针对部分路由的过滤器。 default-filters针对所有路由的默认…

锂电池SOH预测 | 基于BP神经网络的锂电池SOH预测(附matlab完整源码)

锂电池SOH预测 锂电池SOH预测完整代码锂电池SOH预测 锂电池的SOH(状态健康度)预测是一项重要的任务,它可以帮助确定电池的健康状况和剩余寿命,从而优化电池的使用和维护策略。 SOH预测可以通过多种方法实现,其中一些常用的方法包括: 容量衰减法:通过监测电池的容量衰减…

.NET 检测地址/主机/域名是否正常

&#x1f331;PING 地址/主机名/域名 /// <summary>/// PING/// </summary>/// <param name"ip">ip</param>/// <returns></returns>public static bool PingIp(string ip){System.Net.NetworkInformation.Ping p new System.N…

Qt/C++ 波形绘制双缓冲下改善PaintEvent连续绘制卡顿问题(完整代码解析)

音频波形可视化&#xff1a;该控件用于将音频样本数据可视化为波形&#xff0c;常用于音频处理软件中以展示音频信号的时间域特性。 动态数据绘制&#xff1a;控件能够响应外部数据的变化并重新绘制波形&#xff0c;适用于实时或动态的音频数据流。 自定义绘制逻辑&#xff1…

git变更远端仓库名之后如何修改本地仓库配置的另一种方法?(删remote指针、添加、绑定master)

背景 如果某个远端的仓库地址变化后&#xff0c;本地仓库可以修改对应的remote。 之前谈过几种方法&#xff0c;比如重新设置一个新的remote的指针&#xff0c;绑定到新地址。然后删除origin&#xff0c;然后把新指针mv到origin。比如直接seturl修改&#xff08;git remote se…

Apache Seata如何解决TCC 模式的幂等、悬挂和空回滚问题

title: 阿里 Seata 新版本终于解决了 TCC 模式的幂等、悬挂和空回滚问题 author: 朱晋君 keywords: [Seata、TCC、幂等、悬挂、空回滚] description: Seata 在 1.5.1 版本解决了 TCC 模式的幂等、悬挂和空回滚问题&#xff0c;这篇文章主要讲解 Seata 是怎么解决的。 今天来聊一…

容器安全-镜像扫描

前言 容器镜像安全是云原生应用交付安全的重要一环&#xff0c;对上传的容器镜像进行及时安全扫描&#xff0c;并基于扫描结果选择阻断应用部署&#xff0c;可有效降低生产环境漏洞风险。容器安全面临的风险有&#xff1a;镜像风险、镜像仓库风险、编排工具风险&#xff0c;小…

STM32HAL库++ESP8266+cJSON连接阿里云物联网平台

实验使用资源&#xff1a;正点原子F1 USART1&#xff1a;PA9P、A10&#xff08;串口打印调试&#xff09; USART3&#xff1a;PB10、PB11&#xff08;WiFi模块&#xff09; DHT11&#xff1a;PG11&#xff08;采集数据、上报&#xff09; LED0、1&#xff1a;PB5、PE5&#xff…

神经网络的优化器

神经网络的优化器是用于训练神经网络的一类算法&#xff0c;它们的核心目的是通过改变神经网络的权值参数来最小化或最大化一个损失函数。优化器对损失函数的搜索过程对于神经网络性能至关重要。 作用&#xff1a; 参数更新&#xff1a;优化器通过计算损失函数相对于权重参数的…

c++理论篇(一) ——浅谈tcp缓存与tcp的分包与粘包

介绍 在网络通讯中,Linux系统为每一个socket创建了接收缓冲区与发送缓冲区,对于TCP协议来说,这两个缓冲区是必须的.应用程序在调用send/recv函数时,Linux内核会把数据从应用进程拷贝到socket的发送缓冲区中,应用程序在调用recv/read函数时,内核把接收缓冲区中的数据拷贝到应用…

Xcode 15构建问题

构建时出现的异常&#xff1a; 解决方式&#xff1a; 将ENABLE_USER_SCRIPT_SANDBOXING设为“no”即可&#xff01;

【Linux命令行艺术】1. 初见命令行

&#x1f4da;博客主页&#xff1a;爱敲代码的小杨. ✨专栏&#xff1a;《Java SE语法》 | 《数据结构与算法》 | 《C生万物》 |《MySQL探索之旅》 |《Web世界探险家》 ❤️感谢大家点赞&#x1f44d;&#x1f3fb;收藏⭐评论✍&#x1f3fb;&#xff0c;您的三连就是我持续更…

【网络基础】深入理解UDP协议:从报文格式到应用本质

文章目录 前言Udp协议段格式1. 几乎所有协议首要解决的两个问题&#xff1a;a) 如何分离&#xff08;封装&#xff09;b) 如何进行向上交付 2. 理解报文本身3. 对Udp报文字段的解释4. Udp的特点如何理解 面向数据报&#xff1a; 5. IO类接口的本质&#xff1a;sento、recvfromU…