MFC进度条学习笔记

最近工作中有使用到进度条的知识,就来学习学习这个控件~~
先来看看想要达到的效果:
在这里插入图片描述
好的,现在开始正题。
我们这里的进度条,在程序里面叫做CProgressCtrl,它是继承自CWnd类的基础控件类,用来展示工作进展度。

1、主线程的处理方式。
1.1先给我们的控件添加一个环境变量,起个名字就叫做:

	CProgressCtrl m_progressCtrl;

1.2 接着,在对话框初始化的时候做一些初始化工作:

void CdemoDlg::Init()
{
	m_progressCtrl.SetRange(0, 100);//设置进度条数值变化范围
	m_progressCtrl.SetPos(0);			  //设置进度条默认初始进度
	m_progressCtrl.SetStep(m_nStep);

	//文本框百分比显示
	CString str;
	int nPos = m_progressCtrl.GetPos(); // 获取进度条的当前位置
	str.Format("%d", nPos);
	GetDlgItem(IDC_STATIC_PERCENT)->SetWindowText(str + _T("%"));//显示进度条初始进度
}

1.3 然后,需要在头文件里面添加几个重要的方法和变量:

private:
	int m_nStopType;	//0加载完毕结束1外界干扰结束
public:
	int m_nStep;		//步长
	void UpdateProgress();	//用来更新我们的进度条
	void StopUpdate();		//用来结束我们的进度条
	afx_msg void OnBnClickedButtonPerLoad();	//手动点击
	afx_msg void OnBnClickedButtonAutoLoad();	//自动
	afx_msg void OnBnClickedButtonStopLoad();	//停止
	afx_msg void OnTimer(UINT_PTR nIDEvent);	//定时器

1.4 实现那些关键的函数:

void CdemoDlg::UpdateProgress()
{
	m_progressCtrl.SetStep(m_nStep); 	 //设置进度条步长
	m_progressCtrl.StepIt();   			 //按照当前步长更新位置
	int nPos = m_progressCtrl.GetPos();  // 获取进度条的当前位置

	CString str;
	str.Format("%d", nPos);
	GetDlgItem(IDC_STATIC_PERCENT)->SetWindowText(str + _T("%"));//显示进度条进度

	if (100 == nPos)
	{
		StopUpdate();
	}
}
void CdemoDlg::StopUpdate()
{
	KillTimer(1);
	if (0 == m_nStopType)
	{
		MessageBox("加载完毕", "温馨提示", MB_OKCANCEL);
	}
	else if (1 == m_nStopType)
	{
		int nPos = m_progressCtrl.GetPos();
		CString tips;
		tips.Format("您已取消加载,目前进度:%d%%", nPos);
		MessageBox(tips, "温馨提示", MB_OKCANCEL);
	}
}
void CdemoDlg::OnBnClickedButtonPerLoad()
{
	UpdateProgress();
}

void CdemoDlg::OnBnClickedButtonAutoLoad()
{
	SetTimer(1, 10, NULL); //开启自动加载
}

void CdemoDlg::OnBnClickedButtonStopLoad()
{
	m_nStopType = 1;
	StopUpdate();
}

void CdemoDlg::OnTimer(UINT_PTR nIDEvent)
{
	if (1 == nIDEvent)
	{
		UpdateProgress();
	}
	CDialogEx::OnTimer(nIDEvent);
}

2、工作者线程的处理方式。
有时候,我们需要创建多线程来帮助我们提高程序效率,这种情况下又该怎样处理控件呢?
我们需要思考,这个工作者线程,只需要去计算我们的数据,算完一部分,就向我们的主程序发送消息,更新我们的UI就好。
2.1首先,添加新类,就叫做:CMyThread

#pragma once

class CMyThread
{
public:
	CMyThread();
	~CMyThread();
};

2.2 接着,添加功能函数,并实现:

	static void Calculate(void *p);

Ps: 至于这里为何需要使用static成员函数,就请移驾到参考里面看下吧,嘤嘤嘤~~

void CMyThread::Calculate(void *p)
{
	HWND hWnd = (HWND)p;
	
	const int MM = 100;			//为了做模拟,假定进行一百次数据计算
	const int NN = 1000000000;	//模拟工程计算量
	const   int nStep = 10;		//增量步长
	for (int i = 0; i < MM;i++)
	{
		long long sum = 0;
		for (int j = 0; j < NN;j++)
		{
			sum += j;
		}

		::PostMessage(hWnd, WM_MSG_UPDATE, WPARAM(nStep), LPARAM(0));
		Sleep(100);
	}

	_endthread();
}

2.3 然后,消息映射:

2.3.1 在CMyThread类头文件里面添加消息声明:

#define  WM_MSG_UPDATE WM_USER+119

2.3.2 在对话框类的头文件里面添加新的自定义消息:

	afx_msg LRESULT  OnMessageProgress(WPARAM w, LPARAM l);

2.3.3 在对话框类的源文件里面实现消息函数:

LRESULT   CdemoDlg::OnMessageProgress(WPARAM w, LPARAM l)
{
	m_nStep = w;
	UpdateProgress();
	return TRUE;
}

2.3.4 在对话框类的源文件的消息映射添加条目:

	ON_MESSAGE(WM_MSG_UPDATE, &CdemoDlg::OnMessageProgress)

2.4 然后,主程序调用:

void CdemoDlg::OnBnClickedButtonStart()
{
	_beginthread(&CMyThread::Calculate, 0, this->GetSafeHwnd());
}

3.总结。
3.1 对话框中启动新线程。把句柄做参数传递。
3.2 线程类中定义静态函数,也可以在对话框类中定义个全局函数。
3.3 线程回调函数必须是静态,且以void*为参数。
3.4 定义自定义消息,线程类发送,对话框类响应。

参考文章:http://greatverve.cnblogs.com/archive/2012/12/21/mfc-thread-CProgressCtrl.html

  • 0
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值