关于STM32定时器学习的一些心得和笔记

本文详细解析了STM32定时器的工作原理,包括预装载寄存器的使能、定时器分频、输出比较通道及编码器模式等关键特性。通过探讨不同配置下定时器的行为,为读者提供了深入理解STM32定时器机制的指南。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.关于定时器TIMX->CR1寄存器的位7 APRE

  • 当APRE=1时,此时使能预装载寄存器。此时在对ARR寄存器写入新的值得时候,预装入了TIMx_ARR,先到值先到自动加载寄存器,然后当产生更新事件时,进入影子寄存器。

  • 当APRE=0时,此时没有缓冲。此时在对ARR寄存器写入新的值的时候,值直接在自动加载寄存器就可以更新。

2.定时器的分频

  • 更改定时器的分频是更改预分频计数器中的数,进而改变定时器时钟CK_CNT。

3.关于预装载寄存器的使能

  • TIMx-CCMR1寄存器中OC1PE=0,禁止TIMx_CCR1寄存器的预装载功能,不用进入预装载寄存器,可随时写入TIMx_CCR1寄存器,并且新写入的数值立即起作用。
  • TIMx-CCMR1寄存器中OC1PE=1,开启TIMx_CCR1寄存器的预装载功能,读写操作仅对预装载寄存器操作,TIMx_CCR1的预装载值在更新事件到来时被传送至当前寄存器中。

4.输出比较通道

  • 关于TIM_OCMode_Toggle,也就是定时器的翻转模式,当TIMx_CCR1=TIMx_CNT时,翻转OC1REF的电平。 capture1 =TIM_GetCapture1(TIM4); TIM_SetCompare1(TIM4, capture1+1000),此时一般TIM_Period = 65535,不用考虑周期,计数器TIM4_CNT一直在增加,让CCR1每经过一个中断+1000,这样就可以利用CCR1寄存器自动溢出产生pwm波。所以改变其翻转时间,可以使一个定时器产生不同频率的波形(用于控制步进电机)。

5.关于APB1和APB2的时钟

  • F1中,APB1的最大频率是36M,APB2的最大频率是72M
  • F427中,APB1的最大频率是180M/4=45M,APB2是180M/2=90M。

6.定时器的编码器模式

  • 使用定时器的TI1和TI2,有三种编码器模式,TI1计数,TI2计数,TI1和TI2都计数。

有什么不对的,可以一起探讨。

在ClassWizard中响应ID为~Dlg中的WM_TIMER消息。 使用SetTimer(nIDEvent,time,NULL)来建立一个定时器,关闭定时器用KillTimer(nIDEvent)函数。 然后可以响应ON_WM_TIMER消息来响应一个定时器完成一次记时后的程序。 响应方式如下: void CTimeDlg::OnTimer(UINT nIDEvent) { if(nIDEvent==1000)//间隔为5秒 { //处理事件 } elseif(nIDEvent==1001)//间隔为10秒 { //处理事件 } CDialog::OnTimer(nIDEvent); } 以下是给出一个串口通信定时检查接收数据的部分代码 void CMyDlg::OnOpenCom() { // TODO: Add your control notification handler code here if( f_open_com==true ) { f_open_com = false; GetDlgItem(IDC_OPEN_COM)->SetWindowText("打开通信端口"); CloseHandle(hComm); KillTimer(1000); /// 关闭定时器 return ; } SetTimer(1000, 1000, NULL); ///nIDEvent==1000,time=5000ms const char *ComNo; DCB dcb; string temp("COM1"); ComNo = temp.c_str(); hComm = CreateFile( ComNo , GENERIC_READ|GENERIC_WRITE , 0 , NULL , OPEN_EXISTING , 0 , 0); if( hComm==INVALID_HANDLE_VALUE ) /// 如果端口未打开 { MessageBox("打开通信端口出错!" , "Comm Error" , MB_OK); return ; } /// 将dcb地址传入,以取得通信参数 GetCommState(hComm,&dcb); /// 得知目前通信状态 dcb.BaudRate = CBR_9600; dcb.ByteSize = 8; /// 字节为8 dcb.Parity = NOPARITY; /// Parity为None dcb.StopBits = ONESTOPBIT; /// 1个停止位 if( !SetCommState( hComm , &dcb)){ MessageBox("通信端口设置出错!" , "Set Error" , MB_OK ); CloseHandle(hComm); return; } GetDlgItem(IDC_OPEN_COM)->SetWindowText("关闭通信端口"); f_open_com = true; } void CMyDlg::OnTimer(UINT nIDEvent) { // TODO: Add your message handler code here and/or call default char inbuff[1024]; DWORD nBytesRead , dwError; COMSTAT cs; /// 取得状态 ClearCommError( hComm , &dwError , &cs); /// 数据是否大于所准备的缓冲区 if( cs.cbInQue > sizeof(inbuff) ) { PurgeComm(hComm , PURGE_RXCLEAR ); /// 清除通信端口数据 return ; } ReadFile(hComm , inbuff , cs.cbInQue , &nBytesRead , NULL ); //接收通信端口的数据 inbuff[cs.cbInQue] = '\0'; MessageBox("打开通信端口出错!" , "Comm Error" , MB_OK); m_Receive.Format("%s",inbuff); UpdateData(false); CDialog::OnTimer(nIDEvent); } 李杨: for(int i=0; ;i++ ) { ... Sleep(5); if(i>...) {AfxMessageBox("错误XXX"); return;} }//跳出后记得停止一些机器动作
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值