stm32 产生精准脉冲数PWM

stm32 产生精准脉冲数PWM

PWM输出TIM高级定时器、通用定时器自带PWM输出功能,难点在于指定脉冲的个数。

方法1:

产生一个周期的PWM触发一次中断,中断计数实现指定个数PWM,则N个PWM波形,触发N次中断。次类方法看起来看不错,实际应用的时候,当PWM的频率高的时候,程序就会频繁进入中断,导致整个程序的实时性变差。低频的时候可以用,一般频率不超过1KHz,1mS中断一次。

方法2:定时器中断法

输出PWM的同时,采用高级定时器的重复次数计数器,将脉冲数放在计数器里面,到达个数的时候溢出中断。
在这里插入图片描述
在这里插入图片描述

小结:

这个方法比较简单,注意计数器是8位,所以只能计数256个脉冲,需要增加个数的,可以在256个溢出中断里面计算,每次进入中断就减去256,再重新给重复计数器剩下的数值,每次最多给256,最后实现超过n个256脉冲+小于m个脉冲的组合。

公示: 目标脉冲y=n*256+m;

n=y/256
m=y%256

这个方法不错,但是只有高级定时器才有重复计数器。

还可以开启另外一个定时器

比如:

  1. 定时器1输出PWM。周期是1ms,占空比可以自己设定。
  2. 定时器2用来定时,想要输出100个脉冲,定时器2就定时100*1ms=100ms。
  3. 同时启动定时器1、2。定时时间到在TIM2中断中关闭PWM输出。
    小结:设定好,两个定时器的周期数,相同的最好计算,由于计数器是16位的,所以最大计算范围是65536。

方法3

采用两个定时器,从模式定时器产生PWM,主模式定时器控制PWM的产生时间。
如下图:从定时器TIM3,ITR1(TS=001)主定时器则为TIM2。


在这里插入图片描述

设置主定时器为单脉冲模式

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

TIM_SelectOutputTrigger(TIM2,TIM_TRGOSource_OC1Ref); //OC1REF signal is used as the trigger output (TRGO).单一模式不需要设置

输出引脚

在这里插入图片描述

在这里插入图片描述

void TIM_Init(void)
{	
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
	TIM_OCInitTypeDef TIM_OCInitStruct;
	GPIO_InitTypeDef  GPIO_InitStruct;

	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOC, ENABLE);//使能PC,PA端口时钟
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2|RCC_APB1Periph_TIM3,ENABLE);   //使能定时器23的时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);						  //使能复用IO时钟	

	//GPIO_PinRemapConfig(GPIO_FullRemap_TIM3, ENABLE);	

	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出
	GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0;			
	GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStruct);	
	
	GPIO_InitStruct.GPIO_Pin = GPIO_Pin_6;
	GPIO_Init(GPIOA,&GPIO_InitStruct); 

	//TIM2工作在单脉冲下
	TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;//向上计数模式
	TIM_TimeBaseInitStruct.TIM_Prescaler = 7199;				//预分频值,每100us计数一次
	TIM_TimeBaseInitStruct.TIM_Period = 99; 					//重装值,决定单脉冲周期
	TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStruct);	

	TIM_SelectOnePulseMode(TIM2,TIM_OPMode_Single);		 //设置TIM2在单脉冲模式,且是单一的脉冲,在下一个更新事件停止	  
	TIM_SelectOutputTrigger(TIM2,TIM_TRGOSource_OC1Ref); //OC1REF signal is used as the trigger output (TRGO).单一模式不需要设置

	TIM_OC1PreloadConfig(TIM2,TIM_OCPreload_Enable);	 //使能定时器2的通道1预装载寄存器
	TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_PWM2;		 //在向上计数时,一旦TIMx_CNT>TIMx_CCR1时通道1为有效电平
	//TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable;	//OC1输出使能
	TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High;//TIM_OCPolarity_High;		//有效电平为高TIM_OCPolarity_Low
	//没什么用
	
	TIM_OCInitStruct.TIM_Pulse = 20;		//比较捕获1的预装载值,决定单脉冲占空比,这个20就是低电平延续的来源
	TIM_OC1Init(TIM2,&TIM_OCInitStruct);		
	TIM_Cmd(TIM2,DISABLE); //Disable the TIM Counter

	TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;//向上计数模式
	TIM_TimeBaseInitStruct.TIM_Prescaler = 719; //预分频值,10us计数一次
	TIM_TimeBaseInitStruct.TIM_Period = 29;		//重装值,决定PWM周期
	TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStruct);

	TIM_SelectSlaveMode(TIM3, TIM_SlaveMode_Gated);//TIM3为门控模式	
	TIM_SelectMasterSlaveMode(TIM3,TIM_MasterSlaveMode_Enable);//使能TIM3的主从模式	
	TIM_SelectInputTrigger(TIM3,TIM_TS_ITR1);//内部触发,TIM_TS_ITR1对应TIM2
	

	TIM_OC1PreloadConfig(TIM3,TIM_OCPreload_Enable);//使能定时器3的通道1预装载寄存器
	TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_PWM2;//在向上计数时,一旦TIMx_CNT>TIMx_CCR1时通道1为有效电平
	TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable;//OC1输出使能
	TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High;//有效电平为高
	TIM_OCInitStruct.TIM_Pulse = 15;//比较捕获1的预装载值,占空比为50%
	TIM_OC1Init(TIM3,&TIM_OCInitStruct);
	TIM_Cmd(TIM3,ENABLE);//使能TIM3
}

/*********************************************************
名称:Pulse_Control(u16 cycle, u16 pulse_num)
说明:生成指定个数脉冲,每个脉冲周期为cycle微秒,脉冲个数生成的个数
      和单脉冲高电平时间有关系,脉冲个数就由高电平时间来确定
参数cycle:为TIM3一个脉冲周期,单位(us)
参数pulse_num:为脉冲个数,决定步进电机步数
**65516=65536 -20**
要求:cycle * pulse_num < 6551600  us
返回值:无
*********************************************************/
void Pulse_Control(u16 cycle, u16 pulse_num)
{
	u16 arr3 = 0;
	u32 time = 0;
	if(pulse_num)
	{ 
		time = cycle * pulse_num / 100;//预分频为720,100us计数一次
		arr3 = cycle / 10;             //预分频为72,10us计数一次 
		TIM_SetAutoreload(TIM2, time+19);//低电平周期加高电平周期
		
		TIM_SetAutoreload(TIM3, arr3 - 1);
		TIM_SetCompare1(TIM3,arr3 / 2); //设置PWM占空比为50%
		TIM_Cmd(TIM2,ENABLE);//使能TIM2
	}
}

单脉冲主从模式



``
单次测量
![在这里插入图片描述](https://img-blog.csdnimg.cn/485e54a16d90427ca88919fdbd901843.png)
循环发送

```c
int main()
{  	
	TIM_Init(); 
	delay_init();
	
	while(1)
	{	
		
		Pulse_Control(500,20); //500us,10个脉冲
		delay_ms(1);		
		
	}
}

** 周期15.057ms ,占空比参数20**
在这里插入图片描述
** 周期15.057ms ,占空比参数20**
在这里插入图片描述


#include "stm32f10x.h"
#include "delay.h"
#include "timer.h"

int main()
{  	
	TIM_Init(); 
	delay_init();
	
	while(1)
	{	
		
		Pulse_Control(500,20); //500us,10个脉冲
		delay_ms(1);		
		
	}
}

在这里插入图片描述

连续输出波形的探讨

去掉主函数的延时函数,加多脉冲的输出
一开始以为会在没执行完脉冲的输出个数后会覆盖输出,导致输出混乱,实际测量才发现并没有

int main()
{  	
	TIM_Init(); 
	delay_init();
	
	while(1)
	{	
		
		Pulse_Control(500,500); //500us,10个脉冲
		//delay_ms(1);		
		
	}
}

在这里插入图片描述

意味着如何执行输出指令,它都等待下一个质量执行完才重新输出,出现这样的情况原因如下:

在这里插入图片描述
硬件仿真
确实没有使能该位
在这里插入图片描述
寄存器详情
在这里插入图片描述
** 溢出中断才会更新ARR值**

福利彩票双色球号码预测可以使用线性回归算法来进行推测。具体步骤如下: 1. 保存历史数据:使用Python内置的csv模块,将历史中奖数据保存到csv文件中。 2. 获取历史数据:使用pandas库读取csv文件中的历史中奖数据。 3. 数据预处理:将历史中奖数据中的红球和蓝球分别提取出来,并将其转换为数字类型。 4. 特征工程:将历史中奖数据中的红球和蓝球分别作为特征,将下一期的中奖号码作为标签。 5. 模型训练:使用sklearn库中的线性回归算法对特征和标签进行训练。 6. 预测中奖号码:使用训练好的模型对下一期的中奖号码进行预测。 具体代码实现可以参考以下范例: ```python import pandas as pd from sklearn.linear_model import LinearRegression # 保存历史数据 import csv f = open('双色球.csv', mode='a', encoding='utf-8', newline='') csv_writer = csv.DictWriter(f, fieldnames=['期号', '开奖日期', '红球', '蓝球', '一等奖中奖注数', '一等奖中奖金额', '二等奖中奖注数', '二等奖中奖金额', '三等奖中奖注数', '三等奖中奖金额', '四等奖中奖注数', '四等奖中奖金额', '五等奖中奖注数', '五等奖中奖金额', '六等奖中奖注数', '六等奖中奖金额', '一等奖中奖地区', '奖池金额']) csv_writer.writeheader() # 写入表头 csv_writer.writerow(dit) # 写入数据 # 获取历史数据 data = pd.read_csv('双色球.csv') # 数据预处理 data['红球'] = data['红球'].apply(lambda x: [int(i) for i in x.split()]) data['蓝球'] = data['蓝球'].apply(lambda x: int(x)) # 特征工程 X = data[['红球', '蓝球']][:-1] Y = data[['红球', '蓝球']][1:] # 模型训练 model = LinearRegression() model.fit(X, Y) # 预测中奖号码 predict_value = [[33]] # 下一期的中奖号码 result = model.predict(predict_value) if data.shape[0] < 7: print(f'中奖第{data.shape[0]}个红球为:', result[0][0].astype('int64'), '号球') else: print('中奖蓝球为:', result[0][1].astype('int64'), '号球') ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值