前言:在前面的章节中,我们已经学习了 STM32 的 ADC 模块以及 DMA 模块的基础使用。
这一节,我们将两者结合,实现一个高效、连续、自动化的多通道 ADC 采集系统。
目录
一、接线图

四个 ADC 通道分别接入了模拟信号源:
PA0 → ADC_Channel_0
PA1 → ADC_Channel_1
PA2 → ADC_Channel_2
PA3 → ADC_Channel_3
所有模拟输入端都接入 0~3.3V 的电压信号范围。
二、为什么要将 ADC 与 DMA 结合?
在单独使用 ADC 采集时,我们通常需要:
软件启动一次 ADC;
等待转换完成;
读取结果;
再次启动下一次采集。
当采样通道多、采样频率高时,CPU 就需要频繁介入,效率低、实时性差。
而 DMA(Direct Memory Access,直接存储器访问)可以在 外设和存储器之间自动搬运数据,不经过 CPU 干预。
因此,ADC + DMA 结合的最大优点就是:
自动化:ADC 每转换完一次数据,DMA 会自动把结果搬运到指定数组;
高效率:CPU 只需读取最终结果,无需参与每次搬运;
实时性强:可持续连续采样,不丢数据。
三、系统设计思路
配置 GPIO 为模拟输入模式;
初始化 ADC,设置为 扫描模式 + 连续转换模式;
使用 DMA 将 ADC 结果自动搬运到
AD_Value[]数组;OLED 实时显示 4 路 ADC 转换结果。
四、代码实现
1.ADC初始化
void AD_Init(void)
{
/* 1. 时钟使能 */
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
/* 2. 设置ADC时钟 */
RCC_ADCCLKConfig(RCC_PCLK2_Div6); // ADC时钟 = 72MHz / 6 = 12MHz
/* 3. GPIO配置为模拟输入 */
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3;
GPIO_Init(GPIOA, &GPIO_InitStructure);
/* 4. ADC规则组通道配置(通道顺序) */
ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 4, ADC_SampleTime_55Cycles5);
/* 5. ADC初始化 */
ADC_InitTypeDef ADC_InitStructure;
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
ADC_InitStructure.ADC_ScanConvMode = ENABLE;
ADC_InitStructure.ADC_NbrOfChannel = 4;
ADC_Init(ADC1, &ADC_InitStructure);
扫描模式 (Scan Mode):依次采集多个通道;
连续转换 (Continuous Mode):采完一轮立即开始下一轮;
右对齐 (Right Align):数据位右对齐存放;
不使用外部触发:通过软件启动。
2.DMA初始化
DMA_InitTypeDef DMA_InitStructure;
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR;
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)AD_Value;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_BufferSize = 4;
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;
DMA_Init(DMA1_Channel1, &DMA_InitStructure);
DMA_PeripheralBaseAddr:ADC 数据寄存器
ADC1->DR;DMA_MemoryBaseAddr:目标数组
AD_Value;Circular 模式:DMA 到尾后自动回到头部循环存放;
HalfWord 模式:每次传输 16 位 ADC 数据;
方向 (PeripheralSRC):数据来源为外设(ADC)。
3.启动流程
DMA_Cmd(DMA1_Channel1, ENABLE); // 启动DMA
ADC_DMACmd(ADC1, ENABLE); // 允许ADC触发DMA
ADC_Cmd(ADC1, ENABLE); // 启动ADC
// 校准ADC
ADC_ResetCalibration(ADC1);
while (ADC_GetResetCalibrationStatus(ADC1) == SET);
ADC_StartCalibration(ADC1);
while (ADC_GetCalibrationStatus(ADC1) == SET);
// 软件触发ADC开始转换
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
}
至此,DMA 已与 ADC 完全绑定:
当 ADC 完成一次转换后,DMA 自动把结果送入AD_Value[]。
ADC 连续模式下会不断循环,DMA 也在不断覆盖数组。
4.主函数
int main(void)
{
OLED_Init();
AD_Init();
OLED_ShowString(1, 1, "AD0:");
OLED_ShowString(2, 1, "AD1:");
OLED_ShowString(3, 1, "AD2:");
OLED_ShowString(4, 1, "AD3:");
while (1)
{
OLED_ShowNum(1, 5, AD_Value[0], 4);
OLED_ShowNum(2, 5, AD_Value[1], 4);
OLED_ShowNum(3, 5, AD_Value[2], 4);
OLED_ShowNum(4, 5, AD_Value[3], 4);
Delay_ms(100);
}
}
此时,OLED 会实时刷新显示四个通道的电压采样值,
而 CPU 几乎不需要干预采集过程,DMA 和 ADC 自主运行。
五、程序现象
OLED 屏幕上持续刷新四个通道的 ADC 数值;
当模拟输入电压变化时,OLED 上对应通道的数值会实时变化;
CPU 主循环只负责显示,不再处理采样逻辑,系统响应更快。
浙公网安备 33010602011771号