STM32简介
STM32是意法半导体(ST)推出的基于ARM Cortex-M内核的32位微控制器系列,性能强大、功耗低、外设丰富。
为什么选择STM32?
- 性能强劲:主频可达400MHz+
- 外设丰富:定时器、ADC、DAC、通信接口一应俱全
- 生态完善:HAL库、CubeMX图形化配置
- 价格亲民:从几元到几十元不等
- 资料丰富:中文社区活跃
开发环境搭建
硬件准备
- 开发板:STM32F103C8T6(蓝色药丸)
- 调试器:ST-Link V2
- USB转串口:CH340
- 面包板和杜邦线
软件安装
Keil MDK
1 2 3
| 1. 下载Keil MDK-ARM 2. 安装STM32芯片包(PACK) 3. 注册激活(学习使用可用评估版)
|
STM32CubeMX
图形化配置工具,自动生成初始化代码:
1 2 3
| 1. 官网下载CubeMX 2. 安装Java运行环境 3. 下载对应芯片的固件库
|
第一个程序:点亮LED
CubeMX配置
- 新建项目,选择芯片STM32F103C8T6
- 配置时钟:外部8MHz晶振,PLL倍频到72MHz
- 配置GPIO:PC13设置为GPIO_Output(LED引脚)
- 生成代码
代码实现
1 2 3 4 5 6 7 8 9 10 11 12 13
| #include "main.h"
int main(void) { HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); while (1) { HAL_GPIO_TogglePin(GPIOC, GPIO_PIN_13); HAL_Delay(500); } }
|
编译下载
1 2 3 4
| 1. Keil中编译项目 2. 连接ST-Link到开发板 3. 点击"Download"下载程序 4. 复位开发板,看到LED闪烁
|
GPIO编程
寄存器操作
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| void LED_Init(void) { RCC->APB2ENR |= RCC_APB2ENR_IOPCEN; GPIOC->CRH &= 0xFF0FFFFF; GPIOC->CRH |= 0x00300000; }
void LED_On(void) { GPIOC->BSRR = GPIO_BSRR_BR13; }
void LED_Off(void) { GPIOC->BSRR = GPIO_BSRR_BS13; }
|
HAL库操作
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| void LED_Init(void) { __HAL_RCC_GPIOC_CLK_ENABLE(); GPIO_InitTypeDef GPIO_InitStruct = {0}; GPIO_InitStruct.Pin = GPIO_PIN_13; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(GPIOC, &GPIO_InitStruct); }
uint8_t Key_Scan(void) { if (HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_0) == GPIO_PIN_SET) { HAL_Delay(20); if (HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_0) == GPIO_PIN_SET) { while (HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_0) == GPIO_PIN_SET); return 1; } } return 0; }
|
中断系统
外部中断
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| void EXTI_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; __HAL_RCC_GPIOA_CLK_ENABLE(); GPIO_InitStruct.Pin = GPIO_PIN_0; GPIO_InitStruct.Mode = GPIO_MODE_IT_RISING; GPIO_InitStruct.Pull = GPIO_PULLDOWN; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); HAL_NVIC_SetPriority(EXTI0_IRQn, 0, 0); HAL_NVIC_EnableIRQ(EXTI0_IRQn); }
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin) { if (GPIO_Pin == GPIO_PIN_0) { HAL_GPIO_TogglePin(GPIOC, GPIO_PIN_13); } }
|
定时器应用
基础定时器
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
| void TIM_Init(void) { TIM_HandleTypeDef htim2; __HAL_RCC_TIM2_CLK_ENABLE(); htim2.Instance = TIM2; htim2.Init.Prescaler = 71; htim2.Init.CounterMode = TIM_COUNTERMODE_UP; htim2.Init.Period = 999; htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; HAL_TIM_Base_Init(&htim2); HAL_TIM_Base_Start_IT(&htim2); HAL_NVIC_EnableIRQ(TIM2_IRQn); }
void TIM2_IRQHandler(void) { HAL_TIM_IRQHandler(&htim2); }
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim) { if (htim->Instance == TIM2) { static uint16_t count = 0; count++; if (count >= 1000) { count = 0; HAL_GPIO_TogglePin(GPIOC, GPIO_PIN_13); } } }
|
PWM输出
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45
| void PWM_Init(void) { TIM_HandleTypeDef htim3; TIM_OC_InitTypeDef sConfigOC = {0}; __HAL_RCC_TIM3_CLK_ENABLE(); __HAL_RCC_GPIOA_CLK_ENABLE(); GPIO_InitTypeDef GPIO_InitStruct = {0}; GPIO_InitStruct.Pin = GPIO_PIN_6; GPIO_InitStruct.Mode = GPIO_MODE_AF_PP; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); htim3.Instance = TIM3; htim3.Init.Prescaler = 0; htim3.Init.CounterMode = TIM_COUNTERMODE_UP; htim3.Init.Period = 999; HAL_TIM_PWM_Init(&htim3); sConfigOC.OCMode = TIM_OCMODE_PWM1; sConfigOC.Pulse = 0; HAL_TIM_PWM_ConfigChannel(&htim3, &sConfigOC, TIM_CHANNEL_1); HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_1); }
void Set_PWM_Duty(uint16_t duty) { __HAL_TIM_SET_COMPARE(&htim3, TIM_CHANNEL_1, duty); }
void Breathing_LED(void) { static uint16_t duty = 0; static int8_t direction = 1; duty += direction * 10; if (duty >= 1000) direction = -1; if (duty <= 0) direction = 1; Set_PWM_Duty(duty); HAL_Delay(10); }
|
串口通信
基础串口
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36
| UART_HandleTypeDef huart1;
void USART_Init(void) { __HAL_RCC_USART1_CLK_ENABLE(); __HAL_RCC_GPIOA_CLK_ENABLE(); GPIO_InitTypeDef GPIO_InitStruct = {0}; GPIO_InitStruct.Pin = GPIO_PIN_9; GPIO_InitStruct.Mode = GPIO_MODE_AF_PP; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); GPIO_InitStruct.Pin = GPIO_PIN_10; GPIO_InitStruct.Mode = GPIO_MODE_INPUT; GPIO_InitStruct.Pull = GPIO_NOPULL; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); huart1.Instance = USART1; huart1.Init.BaudRate = 115200; huart1.Init.WordLength = UART_WORDLENGTH_8B; huart1.Init.StopBits = UART_STOPBITS_1; huart1.Init.Parity = UART_PARITY_NONE; huart1.Init.Mode = UART_MODE_TX_RX; HAL_UART_Init(&huart1); }
int fputc(int ch, FILE *f) { HAL_UART_Transmit(&huart1, (uint8_t *)&ch, 1, HAL_MAX_DELAY); return ch; }
printf("Hello STM32! Value=%d\r\n", value);
|
DMA串口
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| uint8_t rx_buffer[100];
void USART_DMA_Init(void) { HAL_UART_Receive_IT(&huart1, rx_buffer, 1); }
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) { if (huart->Instance == USART1) { HAL_UART_Transmit(&huart1, rx_buffer, 1, 100); HAL_UART_Receive_IT(&huart1, rx_buffer, 1); } }
|
ADC采集
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35
| ADC_HandleTypeDef hadc1;
void ADC_Init(void) { ADC_ChannelConfTypeDef sConfig = {0}; __HAL_RCC_ADC1_CLK_ENABLE(); hadc1.Instance = ADC1; hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE; hadc1.Init.ContinuousConvMode = DISABLE; hadc1.Init.DiscontinuousConvMode = DISABLE; hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START; hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT; hadc1.Init.NbrOfConversion = 1; HAL_ADC_Init(&hadc1); sConfig.Channel = ADC_CHANNEL_0; sConfig.Rank = ADC_REGULAR_RANK_1; sConfig.SamplingTime = ADC_SAMPLETIME_55CYCLES_5; HAL_ADC_ConfigChannel(&hadc1, &sConfig); }
uint16_t Read_ADC(void) { HAL_ADC_Start(&hadc1); HAL_ADC_PollForConversion(&hadc1, HAL_MAX_DELAY); return HAL_ADC_GetValue(&hadc1); }
float Get_Voltage(void) { uint16_t adc_value = Read_ADC(); return (adc_value * 3.3f) / 4096.0f; }
|
I2C通信
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
| I2C_HandleTypeDef hi2c1;
void I2C_Init(void) { __HAL_RCC_I2C1_CLK_ENABLE(); hi2c1.Instance = I2C1; hi2c1.Init.ClockSpeed = 100000; hi2c1.Init.DutyCycle = I2C_DUTYCYCLE_2; hi2c1.Init.OwnAddress1 = 0; hi2c1.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT; HAL_I2C_Init(&hi2c1); }
void EEPROM_Write(uint16_t addr, uint8_t *data, uint16_t len) { HAL_I2C_Mem_Write(&hi2c1, 0xA0, addr, I2C_MEMADD_SIZE_16BIT, data, len, HAL_MAX_DELAY); HAL_Delay(5); }
void EEPROM_Read(uint16_t addr, uint8_t *data, uint16_t len) { HAL_I2C_Mem_Read(&hi2c1, 0xA0, addr, I2C_MEMADD_SIZE_16BIT, data, len, HAL_MAX_DELAY); }
|
FreeRTOS移植
添加RTOS
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
| void StartDefaultTask(void const * argument); void LED_Task(void const * argument);
void MX_FREERTOS_Init(void) { osThreadDef(defaultTask, StartDefaultTask, osPriorityNormal, 0, 128); defaultTaskHandle = osThreadCreate(osThread(defaultTask), NULL); osThreadDef(ledTask, LED_Task, osPriorityLow, 0, 128); osThreadCreate(osThread(ledTask), NULL); }
void LED_Task(void const * argument) { for(;;) { HAL_GPIO_TogglePin(GPIOC, GPIO_PIN_13); osDelay(500); } }
void UART_Task(void const * argument) { for(;;) { printf("Running...\r\n"); osDelay(1000); } }
|
调试技巧
SWD调试
1 2 3 4 5 6 7
| #include "stdio.h"
int _write(int file, char *ptr, int len) { ITM_SendChar(*ptr); return len; }
|
硬件断点
Keil中设置断点,单步调试查看变量。
逻辑分析仪
使用逻辑分析仪查看I2C、SPI等通信波形。
常见问题
1. 程序不运行
2. 串口乱码
3. 定时器不准
项目实战:智能温控系统
综合应用ADC、PWM、串口:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
| #define TEMP_TARGET 25.0f #define KP 10.0f
float current_temp = 0; uint16_t fan_speed = 0;
void Control_Loop(void) { float voltage = Get_Voltage(); current_temp = voltage * 100.0f; float error = TEMP_TARGET - current_temp; fan_speed = (uint16_t)(KP * error); if (fan_speed > 1000) fan_speed = 1000; if (fan_speed < 0) fan_speed = 0; Set_PWM_Duty(fan_speed); printf("Temp:%.1f Fan:%d\r\n", current_temp, fan_speed); }
|
总结
STM32功能强大,学习路径:
- 掌握寄存器和HAL库
- 熟悉常用外设(GPIO、定时器、串口)
- 学习通信协议(I2C、SPI、CAN)
- 移植RTOS,开发复杂应用
推荐资源: