主要选项说明(绝大部分工程设置下面参数即可):
Clock Prescaler:ADC时钟;时钟不超过36M(从ABP2时钟分出)。
Resolution:分辨率;例如我们选12bits,即使把3.3V电压分成2^12。不可测量超过3.3V的电压,必须分压。
Data Alignment:数据对齐方式;选右对齐就行了。(不详细讲,需要看手册)
Scan Conversion Mode:连续扫描模式;用于多通道,单通道不选。用于多通道时,会按照Rank设置的顺序扫描。
Continuous Conversion Mode和Discontinuous Conversion Mode:连续转化还是单次转换,互斥关系,二选一。
DMA Continuous Requests:DMA接收转换数据;
Number Of Conversion:转换通道数。
Rank:转化顺序;数据排列与其一致。
Channel:通道。
Sampling Time:采集时间。
-
#include
-
unsigned
int temp[
100] = {
0}, i;
-
unsigned
long ad =
0;
-
-
int fputc(
int ch, FILE* file)
-
{
-
return HAL_UART_Transmit(&huart1, (uint8_t*)&ch,
1,
1000);
-
}
-
-
int main(
void)
-
{
-
HAL_Init();
-
SystemClock_Config();
-
MX_GPIO_Init();
-
MX_DMA_Init();
-
MX_ADC1_Init();
-
MX_USART1_UART_Init();
-
-
HAL_ADC_Start_DMA(&hadc1, temp,
100);
-
-
while (
1)
-
{
-
for(i =
0; i <
100; i++)
-
ad += temp[i];
-
ad /=
100;
-
printf(
"ad = %f\r\n", ad*
3.3f/
4096);
-
HAL_Delay(
1000);
-
}
-
}
多通道ADC采集的时候,DMA是会按照通道配置的顺序采集及排列数据。
比如通道顺序是A1 A2,那么对应的DMA缓存区的数据即 A1 A2 A1 A2....
-
#include
-
unsigned
int temp[
100] = {
0}, i;
-
unsigned
long ad1 =
0, ad2 =
0;
-
-
int fputc(
int ch, FILE* file)
-
{
-
return HAL_UART_Transmit(&huart1, (uint8_t*)&ch,
1,
1000);
-
}
-
-
int main(
void)
-
{
-
HAL_Init();
-
SystemClock_Config();
-
MX_GPIO_Init();
-
MX_DMA_Init();
-
MX_ADC1_Init();
-
MX_USART1_UART_Init();
-
-
HAL_ADC_Start_DMA(&hadc1, temp,
100);
-
-
while (
1)
-
{
-
/* USER CODE END WHILE */
-
-
/* USER CODE BEGIN 3 */
-
for(i =
0; i <
100;)
-
{
-
ad1 += temp[i++];
-
ad2 += temp[i++];
-
}
-
-
ad1 /=
50;
-
ad2 /=
50;
-
printf(
"ad1 = %fV\r\n", ad1 *
3.3f /
4096);
-
printf(
"ad2 = %fV\r\n", ad2 *
3.3f /
4096);
-
HAL_Delay(
1000);
-
}
-
}
最后:在IO悬空的时候,ADC会串扰, 采集转换的数据会一端正常一端串扰,故IO不能悬空。
全篇完。
本人是一个嵌入式未入门小白,博客仅仅代表我个人主观见解,记录成长笔记。
笔记是以最简单的方式,只展示最核心的原理。
若有与 大神大大 见解有歧义,我绝对坚信 大神大大 见解是对的,我的是错的。
若无积分等无法下载源码,可加入QQ群657407920下载交流经验。感谢~!