基于ADC电压采集的锂电池电量显示方法

应用场景

设备采用锂电池供电,可充电,MCU的ADC采集计算锂电池电压,电池电量根据锂电池放电特性,电池电量三段段码显示(分段式显示)。

电量显示策略

1.有充电器充电器插入的情况下

ADC采集电池电压,判断是否充满电(例如3.7V锂电池充满电的情况下大约是4.2V,满电电压根据实际测试选择,我们选择的是4.1V)。
注意:电池电压满电电压不宜选择过高,否则会出现一直充电的情况,也不宜选择过低,不然就会出现电池未充满但是显示充满的情况。
电池电压未到达满电电压:电池显示的三段循环递进显示(充电显示效果);
电池电压到达满电电压:电池显示的三段全部显示(充满电)。

2.无充电器充电器插入的情况下

ADC实时采集电池电压,判断电池电压值对应的显示段码,正常来讲三段式显示采用平均分配的方法,根据锂电池放电特性(电池容量与电池电压的关系,具体还要看电池厂家给的资料,这里的数据只作为参考):
在这里插入图片描述
电池电量显示设计为(不是很准确,举个栗子):
电池电压>3.9V,显示三格电量(满电);
3.7V<电池电压<=3.9V,显示两格;
3.3V<电池电压<=3.7V,显示一格;
电池电压<=3.3V,显示空,提示低电量,一段时间后自动断电关机。

关键问题处理

会出现的问题:显示跳动

按照上述的显示方式,由于ADC的采集误差,当电池电压处于显示临界值(例如3.9V左右时)的时候,会出现电量显示来回跳动的问题,在每个临界点都会出现显示跳动的问题。

解决方法

1.延长电量更新时间
我们之前是1秒钟更新一次,可以适当的加大更新时间(例如1min,这样的话随着电量的消耗,在一分钟的时间内,电池电压的下降的),这种方法的弊端就是你得确定你设置的更新时间间隔电池电压是必然下降的;
2.使用电量计
使用电量计能够精确的计算出消耗的电量,但是我们项目对于电量显示的要求并不高,而且会增加成本。
3.未充电情况下电池电压只允许下降
这种处理方法就是在未充电的情况下,每次采集的电压与上一次的采集电压相比较,如果本次电压大于或等于上次采集电压,那么本次采集的电压是无效的,显示电量采用的是上次电压值,如果本次电压小于上次采集电压,显示电量采用的是本次电压值。
以下是伪代码:

void displayBAT(void)
{
//V0为本次采集的电压值,V1为上次采集的电压值,Vdis为显示电量的电压值
	 if(V0<V1)
	 {
	  Vdis = V0;
	  display(Vdis);//更新电池电量显示
	  V1=V0;
	}
}

这种方式能够有效的解决临界值电量显示段码来回跳动的问题。

  • 11
    点赞
  • 58
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
基于STM32的ADC电压采集LCD1602显示-源文件1主要包含了ADC的初始化、读取电压值、LCD初始化以及显示电压值等功能。下面是一个简单的示例源代码: #include "stm32f10x.h" #include "delay.h" #include "lcd1602.h" void ADC_Init(void) { ADC_InitTypeDef ADC_InitStructure; GPIO_InitTypeDef GPIO_InitStructure; // 开启 ADC1 和 GPIOA 的时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOA, ENABLE); // 配置 ADC1 的模拟输入通道为 PA4 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; GPIO_Init(GPIOA, &GPIO_InitStructure); // ADC 初始化设置 ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; // 单通道独立模式 ADC_InitStructure.ADC_ScanConvMode = DISABLE; // 禁止扫描模式 ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; // 禁止连续转换模式 ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; // 右对齐 ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; // 不使用外部触发 ADC_InitStructure.ADC_NbrOfChannel = 1; // 只转换一个通道 ADC_Init(ADC1, &ADC_InitStructure); // 使能 ADC1 的 DMA 传输 ADC_DMACmd(ADC1, ENABLE); // 开启 ADC ADC_Cmd(ADC1, ENABLE); // 校准 ADC ADC_ResetCalibration(ADC1); while(ADC_GetResetCalibrationStatus(ADC1)); ADC_StartCalibration(ADC1); while(ADC_GetCalibrationStatus(ADC1)); } uint16_t ADC_GetValue(void) { // 设置要转换的通道 ADC_RegularChannelConfig(ADC1, ADC_Channel_4, 1, ADC_SampleTime_55Cycles5); // 开始转换 ADC_SoftwareStartConvCmd(ADC1, ENABLE); // 等待转换完成 while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC)); // 返回转换结果 return ADC_GetConversionValue(ADC1); } void LCD_Init(void) { // 初始化 LCD1602 LCD1602_Init(); } void displayVoltage(uint16_t voltage) { char str[16] = {0}; sprintf(str, "Voltage: %d.%02dV", voltage / 100, voltage % 100); LCD1602_SetCursor(0, 0); LCD1602_WriteString(str); } int main(void) { uint16_t voltage; // 初始化 ADC ADC_Init(); // 初始化 LCD LCD_Init(); while (1) { // 获取 ADC 转换结果 voltage = ADC_GetValue(); // 显示电压值 displayVoltage(voltage); // 延时一段时间 delay_ms(1000); } } 这段代码中,首先对ADC和LCD进行了初始化配置。然后在主循环中,不断获取ADC的转换结果,并通过LCD显示电压值。延时函数delay_ms()用于控制刷新频率。可以根据需要调整延时时间以达到合适的显示效果。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值