[图像处理] MFC载入图片并进行二值化处理和灰度处理及其效果显示

文章目录

  • 工程效果
  • 重要代码
  • 完整代码
  • 参考

工程效果

载入图片,并在左侧显示原始图片、二值化图片和灰度图片。
双击左侧的图片控件,可以在右侧的大控件中,显示双击的图片。

初始画面:
在这里插入图片描述
载入图片:
在这里插入图片描述
双击左侧的第二个控件,显示图片:
在这里插入图片描述
//对图片显示在控件中的位置没有进行优化。

重要代码

主要是用的MFC Image控件。

载入图片:

void CGDITESTDlg::OnBnClickedBtnStart()
{
	//获取图像文件
	CFileDialog filedlg(
		TRUE,
		_T("png"),
		NULL,
		0,
		TEXT("image Files(*.bmp;*.jpg;*.png)|*.bmp;*.jpg;*.png|all files(*.*)|(*.*)||"),
		this);
	filedlg.DoModal();
	m_cstr_filepath = filedlg.GetPathName();
	m_cstr_filename = filedlg.GetFileName();
	if (m_cstr_filepath.IsEmpty() || m_cstr_filename.IsEmpty())
	{
		AfxMessageBox(TEXT("打开文件失败"));
		return ;
	}	
	image_origin.Load(m_cstr_filepath);
	image_binarization.Load(m_cstr_filepath);
	image_grey.Load(m_cstr_filepath);

	//获取图像尺寸
	imgw = image_origin.GetWidth();
	imgh = image_origin.GetHeight();
	img_t = imgw / imgh;

	//根据图像尺寸,调整控件客户区尺寸比例
	CRect rect_ctl;
	m_ctl_pic_origin.GetClientRect(&rect_ctl);
	float rectw = rect_ctl.Width();
	float recth = rect_ctl.Height();
	float rect_t = rectw / recth;
	if(rect_t < img_t)
	{
		rect_ctl = CRect(rect_ctl.TopLeft(), CSize(rect_ctl.Width(), rect_ctl.Width() / img_t ));
	}
	else
	{
		rect_ctl = CRect(rect_ctl.TopLeft(), CSize(rect_ctl.Height() * img_t, rect_ctl.Height()));
	}
	//显示图像到控件客户区
	CDC* pdc = m_ctl_pic_origin.GetDC();
	m_ctl_pic_origin.SetBitmap(NULL);	
	image_origin.Draw(pdc->m_hDC, rect_ctl);

	//图像二值化处理
	ImageBinarizationProcess(image_binarization);
	//根据图像尺寸,调整控件客户区尺寸比例
	CRect rect_ctl_binarization;
	m_ctl_pic_binarization.GetClientRect(&rect_ctl_binarization);
	float rectw_ = rect_ctl_binarization.Width();
	float recth_ = rect_ctl_binarization.Height();
	float rect_t_ = rectw / recth;
	if (rect_t_ < img_t)
	{
		rect_ctl_binarization = CRect(rect_ctl_binarization.TopLeft(), CSize(rect_ctl_binarization.Width(), rect_ctl_binarization.Width() / img_t));
	}
	else
	{
		rect_ctl_binarization = CRect(rect_ctl_binarization.TopLeft(), CSize(rect_ctl_binarization.Height() * img_t, rect_ctl_binarization.Height()));
	}
	//显示图像到控件客户区
	CDC* pdc_ = m_ctl_pic_binarization.GetDC();
	m_ctl_pic_binarization.SetBitmap(NULL);
	image_binarization.Draw(pdc_->m_hDC, rect_ctl_binarization);

	//图像灰度处理
	ImageGreyProcess(image_grey);
	//根据图像尺寸,调整控件客户区尺寸比例
	CRect rect_ctl_grey;
	m_ctl_pic_grey.GetClientRect(&rect_ctl_grey);
	float rectw__ = rect_ctl_grey.Width();
	float recth__ = rect_ctl_grey.Height();
	float rect_t__ = rectw / recth;
	if (rect_t__ < img_t)
	{
		rect_ctl_grey = CRect(rect_ctl_grey.TopLeft(), CSize(rect_ctl_grey.Width(), rect_ctl_grey.Width() / img_t));
	}
	else
	{
		rect_ctl_grey = CRect(rect_ctl_grey.TopLeft(), CSize(rect_ctl_grey.Height() * img_t, rect_ctl_grey.Height()));
	}
	//显示图像到控件客户区
	CDC* pdc__ = m_ctl_pic_grey.GetDC();
	m_ctl_pic_grey.SetBitmap(NULL);
	image_grey.Draw(pdc__->m_hDC, rect_ctl_grey);

	isLoadedImage = true;
	
	m_ctl_pic_origin.ReleaseDC(pdc);
	m_ctl_pic_binarization.ReleaseDC(pdc);
	m_ctl_pic_grey.ReleaseDC(pdc);
}

图像二值化处理:
算法来源:C++MFC打开图片、彩图,以及对图像进行简单算法处理

void CGDITESTDlg::ImageBinarizationProcess(CImage &image)
{
	BYTE* pimagedata = (BYTE*)image.GetBits();	//获取到图片内存点的位置
	int width = image.GetWidth();
	int height = image.GetHeight();
	int pit = image.GetPitch();//图像每行字节数
	int bytes_per_pixel = image.GetBPP() / 8; //获取每像素的位数除以8得到每个像素占的字节数
	std::vector<int> gray(256); //初始化时自动存0,用来存放256种颜色出现的次数
	for (int i = 0; i < height; i++)
	{
		for (int j = 0; j < width; j++)
		{
			gray.at((int)*(pimagedata + pit * i + bytes_per_pixel * j)) += 1;
		}
	}

	int max = 0;
	int sec = 0;
	int localmax = 0;
	int localsec = 0;

	for (int i = 0; i < 256; i++)
	{
		if (gray[i] > max)
		{
			max = gray[i];
			localmax = i;
		}
	}
	for (int i = 0; i < 256; i++)
	{
		if (gray[i] > sec && abs(i - localmax) > 10)
		{
			sec = gray[i];
			localsec = i;
		}
	}
	int mid = (localmax + localsec) / 2;
	for (int i = 0; i < height; i++)
	{
		for (int j = 0; j < width; j++)
		{
			if ((int)(*(pimagedata + pit * i + j * bytes_per_pixel)) < mid)
			{
				*(pimagedata + pit * i + j * bytes_per_pixel) = 0;
				*(pimagedata + pit * i + j * bytes_per_pixel + 1) = 0;
				*(pimagedata + pit * i + j * bytes_per_pixel + 2) = 0;
			}
			else
			{
				*(pimagedata + pit * i + j * bytes_per_pixel) = 255;
				*(pimagedata + pit * i + j * bytes_per_pixel + 1) = 255;
				*(pimagedata + pit * i + j * bytes_per_pixel + 2) = 255;
			}
		}
	}
}

灰度处理:
算法来源:C++MFC打开图片、彩图,以及对图像进行简单算法处理

void CGDITESTDlg::ImageGreyProcess(CImage& image)
{
	BYTE* pimagedata = (BYTE*)image.GetBits();	//获取到图片内存点的位置
	int width = image.GetWidth();
	int height = image.GetHeight();
	int pit = image.GetPitch();//图像每行字节数
	int bytes_per_pixel = image.GetBPP() / 8; //获取每像素的位数除以8得到每个像素占的字节数
	for (int i = 0; i < height; i++)
	{
		for (int j = 0; j < width; j++)
		{
			*(pimagedata + pit * i + j * bytes_per_pixel) *= 0.114;
			*(pimagedata + pit * i + j * bytes_per_pixel + 1) *= 0.587;
			*(pimagedata + pit * i + j * bytes_per_pixel + 2) *= 0.299;
		}
	}

}

双击左侧控件的响应:
在这里插入图片描述

void CGDITESTDlg::OnLButtonDblClk(UINT nFlags, CPoint point)
{
	if (!isLoadedImage) //如果没有加载图片,则不会执行后续代码
		return;
	//未使用,仅测试
	CPoint ptCursor;
	GetCursorPos(&ptCursor);//获取执行此函数时的鼠标位置,屏幕坐标
	
	ClientToScreen(&point); //point是双击时的鼠标位置,坐标系是窗口客户区,所以要转换成屏幕坐标
	
	//未使用,仅测试
	CPoint ptCursor1(GetCurrentMessage()->pt); //双击时的鼠标位置,屏幕坐标

	GetDlgItem(IDC_PIC_ORIGIN)->GetWindowRect(&rc_origin);//控件的rect,屏幕坐标
	if (rc_origin.PtInRect(point))//如果右键在picture control区域抬起则放大显示灰度图片
	{
		CRect rect_ctl;
		m_ctl_pic_dsp.GetClientRect(&rect_ctl);
		float rectw = rect_ctl.Width();
		float recth = rect_ctl.Height();
		float rect_t = rectw / recth;

		if (rect_t < img_t)
		{
			rect_ctl = CRect(rect_ctl.TopLeft(), CSize(rect_ctl.Width(), rect_ctl.Width() / img_t));
		}
		else
		{
			rect_ctl = CRect(rect_ctl.TopLeft(), CSize(rect_ctl.Height() * img_t, rect_ctl.Height()));
		}

		CDC* pdc = m_ctl_pic_dsp.GetDC();
		m_ctl_pic_dsp.SetBitmap(NULL);
		image_origin.Draw(pdc->m_hDC, rect_ctl);
		m_ctl_pic_dsp.ReleaseDC(pdc);
		return;
	}

	GetDlgItem(IDC_PIC_PROCESS)->GetWindowRect(&rc_binarization);
	if (rc_binarization.PtInRect(point))//如果右键在picture control区域抬起则放大显示灰度图片
	{
		CRect rect_ctl;
		m_ctl_pic_dsp.GetClientRect(&rect_ctl);
		float rectw = rect_ctl.Width();
		float recth = rect_ctl.Height();
		float rect_t = rectw / recth;

		if (rect_t < img_t)
		{
			rect_ctl = CRect(rect_ctl.TopLeft(), CSize(rect_ctl.Width(), rect_ctl.Width() / img_t));
		}
		else
		{
			rect_ctl = CRect(rect_ctl.TopLeft(), CSize(rect_ctl.Height() * img_t, rect_ctl.Height()));
		}
		CDC* pdc = m_ctl_pic_dsp.GetDC();
		m_ctl_pic_dsp.SetBitmap(NULL);
		image_binarization.Draw(pdc->m_hDC, rect_ctl);
		m_ctl_pic_dsp.ReleaseDC(pdc);
		return;
	}

	GetDlgItem(IDC_PIC_PROCESS2)->GetWindowRect(&rc_grey);
	if (rc_grey.PtInRect(point))//如果右键在picture control区域抬起则放大显示灰度图片
	{
		CRect rect_ctl;
		m_ctl_pic_dsp.GetClientRect(&rect_ctl);
		float rectw = rect_ctl.Width();
		float recth = rect_ctl.Height();
		float rect_t = rectw / recth;

		if (rect_t < img_t)
		{
			rect_ctl = CRect(rect_ctl.TopLeft(), CSize(rect_ctl.Width(), rect_ctl.Width() / img_t));
		}
		else
		{
			rect_ctl = CRect(rect_ctl.TopLeft(), CSize(rect_ctl.Height() * img_t, rect_ctl.Height()));
		}
		CDC* pdc = m_ctl_pic_dsp.GetDC();
		m_ctl_pic_dsp.SetBitmap(NULL);
		image_grey.Draw(pdc->m_hDC, rect_ctl);
		m_ctl_pic_dsp.ReleaseDC(pdc);
		return;
	}

	CDialogEx::OnLButtonDblClk(nFlags, point);
}

以上是通过窗口的双击时间回调函数,判断双击时鼠标的坐标,是否在控件的坐标Rect中,如果是,则执行对应代码。

也可以把双击的空间的notify属性设置为true,然后再时间中设置双击消息回调函数。
在这里插入图片描述
在这里插入图片描述

完整代码

MFC简单的图片处理工程-Gitee

参考

C++MFC打开图片、彩图,以及对图像进行简单算法处理

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

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

相关文章

【uC/OS-III篇】uC/OS-III 移植到 STM32 简明教程

uC/OS-III 移植到 STM32 简明教程 一、uC/OS-III 介绍 二、获取UCOS-III源码 三、建立项目工程 四、解决工程编译报错 五、修改项目文件 下一篇博客&#xff1a; 【uC/OS-III篇】uC/OS-III 创建第一个任务&#xff08;For STM32&#xff09; 一、uC/OS-III 介绍 uC/OS-III…

docker部署开源软件的国内镜像站点

下载镜像 docker pull registry.cn-beijing.aliyuncs.com/wuxingge123/le_monitor:latestdocker-compose部署 vim docker-compose.yml version: 3 services:le_monitor:container_name: le_monitorimage: registry.cn-beijing.aliyuncs.com/wuxingge123/le_monitor:latestpo…

【JDK常用的API】包装类

&#x1f36c; 博主介绍&#x1f468;‍&#x1f393; 博主介绍&#xff1a;大家好&#xff0c;我是 hacker-routing &#xff0c;很高兴认识大家~ ✨主攻领域&#xff1a;【渗透领域】【应急响应】 【Java】 【VulnHub靶场复现】【面试分析】 &#x1f389;点赞➕评论➕收藏 …

SQL Server 数据库常见提权总结

前面总结了linux和Windows的提权方式以及Mysql提权&#xff0c;这篇文章讲讲SQL Server数据库的提权。 目录 基础知识 权限判定 系统数据库 存储过程 常见系统存储过程 常见扩展存储过程 xp_cmdshell扩展存储过程提权 xp_dirtree写入文件提权 sp_oacreate提权 xp_re…

每日面经分享(Spring Boot: part2 DAO层)

1. Spring Boot DAO层的作用 a. 封装数据访问逻辑&#xff1a;DAO层的主要责任是封装与数据访问相关的逻辑。负责处理与数据库的交互&#xff0c;包括数据的增删改查等操作。通过将数据访问逻辑统一封装在DAO层中&#xff0c;可以提高代码的可维护性和可重用性。 b. 解耦业务逻…

学习笔记】java项目—苍穹外卖day05

文章目录 苍穹外卖-day05课程内容1. Redis入门1.1 Redis简介1.2 Redis下载与安装1.2.1 Redis下载1.2.2 Redis安装 1.3 Redis服务启动与停止1.3.1 服务启动命令1.3.2 客户端连接命令1.3.3 修改Redis配置文件1.3.4 Redis客户端图形工具 2. Redis数据类型2.1 五种常用数据类型介绍…

vsphere高可用实验

实验要求&#xff1a; 部署高可用集群&#xff0c;在2个EXSI主机上&#xff0c;将该虚拟机断电。这台虚拟机会在另一台主机上自动起来 实验环境要求&#xff1a; 2台EXSI&#xff0c;一台ISCSI&#xff0c;一台vcenter&#xff0c;在一台EXSI上安装一台虚拟机&#xff0c;要求…

武汉大学开设 “雷军班”:计算机专业、今年招收 15 名本科生。武汉大学已经联合小米成立了机器系

更多精彩内容在公众号。 3月25日&#xff0c;武汉大学官方网站发布了一则新闻&#xff0c;报道了校长张平文对计算机学院的调研活动。在报道中&#xff0c;张平文校长特别强调了关于“雷军班”及机器人系的发展规划。他表示&#xff0c;希望计算机学院能够立足于更高层次&#…

AI预测福彩3D第22弹【2024年3月31日预测--第5套算法开始计算第4次测试】

今天&#xff0c;咱们继续进行本套算法的测试&#xff0c;今天为第四次测试&#xff0c;仍旧是采用冷温热趋势结合AI模型进行预测。好了&#xff0c;废话不多说了。直接上结果~ 仍旧是分为两个方案&#xff0c;1大1小。 经过人工神经网络计算并进行权重赋值打分后&#xff0c;3…

MTMT:构建比特币生态平行世界 打造铭文生态繁荣

近年来&#xff0c;随着铭文市场的火爆以及比特币ETF成功通过&#xff0c;比特币生态正经历着一场复兴&#xff0c;尤其是铭文市场作为新一代Web3的叙事&#xff0c;带来了全新的生产方式&#xff0c;可以预见&#xff0c;铭文就像流动性挖矿对于上一轮DeFi Summer的推动一样会…

Mybatis-特殊SQL的执行

1. 模糊查询 在MyBatis中进行模糊查询时&#xff0c;有以下三种常见的实现方式&#xff1a; 1.1. 错误示范 先来个准备操作&#xff0c;并做一个错误示例 根据姓名&#xff0c;模糊查询用户&#xff0c;(x小x) 更新数据表 SQLMapper.java package com.sakurapaid.mybatis3…

python 进程、线程、协程基本使用

1、进程、线程以及协程【1】进程概念【2】线程的概念线程的生命周期进程与线程的区别 【3】协程(Coroutines) 2、多线程实现【1】threading模块【2】互斥锁【3】线程池【4】线程应用 3、多进程实现4、协程实现【1】yield与协程【2】asyncio模块【3】3.8版本【4】aiohttp 1. 并发…

《C++工程方向面试记录》

此系列&#xff0c;是本人免费提供的&#xff0c;大厂面试模拟&#xff0c;经本人同意后发出。如要申请&#xff0c;请查看《大厂面试模拟&#xff08;免费&#xff09;》 面试者画像&#xff1a; 本硕中部985&#xff0c;工作2-3年&#xff0c;本硕均非计算机专业。毕业后一直…

南京观海微电子---Vitis HLS设计流程介绍——Vitis HLS教程

1. 传统的FPGA设计流程 传统的RTL设计流程如下图所示&#xff1a; 传统的FPGA RTL设计流程主要是采用VHDL、VerilogHDL或System Verilog进行工程的开发&#xff0c;同时也是通过硬件描述语言来编写测试案例&#xff08;Test Bench&#xff09;对开发的工程进行仿真验证。 随后…

SpringBoot接收参数的方式

Get 请求 1.1 以方法的形参接收参数 1.这种方式一般适用参数比较少的情况 RestController RequestMapping("/user") Slf4j public class UserController {GetMapping("/detail")public Result<User> getUserDetail(String name,String phone) {log.…

二维码门楼牌管理应用平台建设:实现民主参与的新途径

文章目录 前言一、二维码门楼牌管理应用平台的兴起二、投票表决功能的实现三、居民参与度的提升四、面临的挑战与前景展望 前言 在数字化时代&#xff0c;二维码技术的应用已经渗透到我们生活的方方面面。近期&#xff0c;二维码门楼牌管理应用平台的建设成为了社区治理的一大…

leecode 331 |验证二叉树的前序序列化 | gdb 调试找bug

计算的本质是数据的计算 数据的计算需要采用格式化的存储&#xff0c; 规则的数据结果&#xff0c;可以快速的按照指定要求存储数据 这里就不得不说二叉树了&#xff0c;二叉树应用场景真的很多 本题讲的是&#xff0c;验证二叉树的前序序列化 换言之&#xff0c;不采用建立树的…

Kubernetes(K8S)学习(二):K8S常用组件

K8S常用组件 一、 Controllers1、ReplicationController(RC)2、ReplicaSet(RS)3、Deployment 二、Labels and Selectors三、Namespace&#xff08;命名空间&#xff09;1、简介2、测试2.1、创建namespace2.2、创建pod 四、Network1、集群内&#xff1a;同一个Pod中的容器通信2、…

数据挖掘|贝叶斯分类器及其Python实现

分类分析|贝叶斯分类器及其Python实现 0. 分类分析概述1. Logistics回归模型2. 贝叶斯分类器2.1 贝叶斯定理2.2 朴素贝叶斯分类器2.2.1 高斯朴素贝叶斯分类器2.2.2 多项式朴素贝叶斯分类器 2.3 朴素贝叶斯分类的主要优点2.4 朴素贝叶斯分类的主要缺点 3. 贝叶斯分类器在生产中的…

力扣刷题Days29-第二题-70.爬楼梯(js)

只有学习&#xff0c;没有自己的思路解题哈哈哈 1&#xff0c;题目 假设你正在爬楼梯。需要 n 阶你才能到达楼顶。 每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢&#xff1f; 2&#xff0c;代码 这种解法的本质是斐波那契数列 /*** param {number} n* re…