初级会员

- 积分
- 188
- 金钱
- 188
- 注册时间
- 2019-10-28
- 在线时间
- 37 小时
|
5金钱
求助,stm32f103ADC单次采集数据没问题,在把数据求平均值后数值降低了好多,如单次采样电压为2.2V,10次采样取均值就变为1.9V,255次采样取均值直接变为0.09V了,样本数越多,数值降低得越厉害。以下是ADC相关的代码,希望大佬们能解答
void ADC_GPIO_Init(void)
{ //GPIO初始化设置
GPIO_InitTypeDef GPIO_InitStructure;
// RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);
// RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);//使能DMA1时钟(用于ADC的数据传送)
// RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);//使能ADC1时钟
GPIO_InitStructure.GPIO_Pin = AD_1_TEST | AD_2_TEST | AD_3_TEST; //!!!选择端口
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //选择IO接口工作方式
GPIO_Init(ADCPORT, &GPIO_InitStructure);
}
void Adc_Init(void) //ADC通道初始化
{
ADC_InitTypeDef ADC_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC |RCC_APB2Periph_ADC1, ENABLE ); //使能GPIOA,ADC1通道时钟
ADC_GPIO_Init();
RCC_ADCCLKConfig(RCC_PCLK2_Div8); //分频因子6时钟为72M/8=9MHz
ADC_DeInit(ADC1); //将外设 ADC1 的全部寄存器重设为缺省值
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式:ADC1和ADC2工作在独立模式
ADC_InitStructure.ADC_ScanConvMode = DISABLE; //模数转换工作在单通道模式
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //模数转换工作在单次转换模式
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //转换由软件而不是外部触发启动
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC数据右对齐
ADC_InitStructure.ADC_NbrOfChannel = 1; //顺序进行规则转换的ADC通道的数目
ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器
// ADC_TempSensorVrefintCmd(ENABLE); //开启内部温度感器和内部参考电压
ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1
ADC_ResetCalibration(ADC1); //重置指定的ADC1的复位寄存器
while(ADC_GetResetCalibrationStatus(ADC1)); //获取ADC1重置校准寄存器的状态,设置状态则等待
ADC_StartCalibration(ADC1);
while(ADC_GetCalibrationStatus(ADC1)); //获取指定ADC1的校准程序,设置状态则等待
}
/************获取单通道采集的AD值********************/
u16 V_Get_Adc(u8 ch)
{
ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5); //ADC1,ADC内部通道,第一个转换,采样时间为239.5周期
ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能
while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束
return ADC_GetConversionValue(ADC1); //返回最近一次ADC1规则组的转换结果
}
//获取通道ch的转换值
//取times次,然后平均
u16 V_Get_Adc_Average(u8 ch,u8 times)
{
u16 temp_val=0;
u8 i, j ,t;
unsigned int value[255] = {0};
for(t=0;t<times;t++)
{
value[t] = V_Get_Adc(ch);
// temp_val+=V_Get_Adc(ch);
delay_us(200);
}
//冒泡排序去头去尾
for(i = 0; i < times-1; i ++)
for(j = i; j < times - i - 1; j ++)
{
if(value[j] < value[j+1])
{
u8 tem =0;
tem = value[j];
value[j] = value[j+1];
value[j+1] = tem;
}
}
//采样次数大于2
if(times > 2)
{
for(i = 1; i < times-1 ; i ++)
{
temp_val += value[i];
}
//返回去掉最大最小值的采样平均值
return temp_val/(times - 2);
}
else
{
for(i = 0; i < times; i++)
{
temp_val += value[i];
}
return temp_val/times;
}
}
//获取相应通道电压值
//Vchx=Vrefint*(ADchx/ADrefint)
u16 Get_V(uint8_t ch, u8 t)
{
u16 ADchx ,Vchx;
ADchx = V_Get_Adc_Average(ch,t);
Vchx = VREF * ADchx / 4096;
return Vchx;
}
|
|