pw-embedded-c-style
Compare original and translation side by side
🇺🇸
Original
English🇨🇳
Translation
Chinesepw-embedded-c-style
pw-embedded-c-style
嵌入式 C 代码风格助手, 基于《手把手教你学51单片机》302 个 .c 文件和 66 个 .h 文件的代码风格分析。
默认命名风格: 蛇形命名 (snake_case)
Embedded C Code Style Assistant, based on the code style analysis of 302 .c files and 66 .h files from the book "Learn 51 MCU Step by Step".
Default Naming Style: snake_case
使用场景
Application Scenarios
适用情况
Applicable Cases
- 创建新的嵌入式 C 项目 (51 单片机、STM32 等)
- 优化现有嵌入式代码的命名和结构
- 生成硬件驱动模块 (定时器、串口、LCD、按键等)
- 规范化项目文件组织和代码风格
- 学习嵌入式 C 编程的最佳实践
- Create new embedded C projects (51 MCU, STM32, etc.)
- Optimize naming and structure of existing embedded code
- Generate hardware driver modules (timer, UART, LCD, key, etc.)
- Standardize project file organization and code style
- Learn best practices for embedded C programming
不适用情况
Inapplicable Cases
- 非嵌入式的通用 C/C++ 项目
- 需要 RTOS 或复杂架构的项目
- 纯算法实现 (无硬件交互)
- Non-embedded general C/C++ projects
- Projects requiring RTOS or complex architecture
- Pure algorithm implementation (no hardware interaction)
使用方式
Usage Methods
默认命名风格: 蛇形命名 (snake_case)
如需使用驼峰命名,请在指令中明确说明 "使用驼峰命名"。
Default Naming Style: snake_case
If you need to use camelCase, clearly specify "use camelCase" in the command.
创建项目
Create Project
bash
undefinedbash
undefined基础示例 (默认蛇形命名)
Basic example (default snake_case)
/pw-embedded-c-style 创建一个 LED 闪烁的项目
/pw-embedded-c-style Create an LED blinking project
带外设的项目
Project with peripherals
/pw-embedded-c-style 创建一个带 LCD1602 显示和按键输入的项目
/pw-embedded-c-style Create a project with LCD1602 display and key input
指定芯片
Specify chip
/pw-embedded-c-style 为 STM32F103 创建一个串口通信项目
/pw-embedded-c-style Create a UART communication project for STM32F103
明确指定使用驼峰命名
Explicitly specify using camelCase
/pw-embedded-c-style 使用驼峰命名创建一个定时器项目
undefined/pw-embedded-c-style Use camelCase to create a timer project
undefined优化代码
Optimize Code
bash
undefinedbash
undefined优化代码风格 (默认蛇形命名)
Optimize code style (default snake_case)
/pw-embedded-c-style 帮我优化这段按键扫描代码
/pw-embedded-c-style Help me optimize this key scanning code
重构模块
Refactor module
/pw-embedded-c-style 将这段代码重构为独立的驱动模块
/pw-embedded-c-style Refactor this code into an independent driver module
添加注释
Add comments
/pw-embedded-c-style 为这段定时器配置代码添加规范注释
/pw-embedded-c-style Add standardized comments to this timer configuration code
使用驼峰命名优化
Optimize using camelCase
/pw-embedded-c-style 使用驼峰命名优化这段代码
undefined/pw-embedded-c-style Use camelCase to optimize this code
undefined生成驱动模板
Generate Driver Template
bash
undefinedbash
undefined生成特定外设驱动
Generate specific peripheral driver
/pw-embedded-c-style 生成 DS1302 实时时钟驱动模板
/pw-embedded-c-style Generate DS1302 real-time clock driver template
生成通信协议
Generate communication protocol
/pw-embedded-c-style 生成串口通信的收发缓冲区实现
---/pw-embedded-c-style Generate UART communication receive/transmit buffer implementation
---核心规范
Core Specifications
命名规范
Naming Specification
默认风格: 蛇形命名 (snake_case)
c
// 函数: 蛇形命名
void config_timer0(unsigned int ms);
void lcd_write_cmd(unsigned char cmd);
void init_ds1302(void);
// 变量: 蛇形命名
unsigned char flag_500ms = 0;
unsigned char cnt_rxd = 0;
// 宏/常量: 全大写下划线
#define SYS_MCLK (11059200/12)
#define LCD1602_RS P1^0
// sbit: 全大写或蛇形
sbit LED = P0^0;
sbit EN_LED = P1^4;备选风格: 驼峰命名 (camelCase/PascalCase)
用户明确要求时使用此风格:
c
// 函数: 大驼峰
void ConfigTimer0(unsigned int ms);
void LcdWriteCmd(unsigned char cmd);
void InitDS1302(void);
// 变量: 小驼峰
unsigned char flag500ms = 0;
unsigned char cntRxd = 0;
// 宏/常量: 全大写下划线 (相同)
#define SYS_MCLK (11059200/12)
#define LCD1602_RS P1^0
// sbit: 全大写或大驼峰
sbit LED = P0^0;
sbit ENLED = P1^4;Default Style: snake_case
c
// Function: snake_case
void config_timer0(unsigned int ms);
void lcd_write_cmd(unsigned char cmd);
void init_ds1302(void);
// Variable: snake_case
unsigned char flag_500ms = 0;
unsigned char cnt_rxd = 0;
// Macro/Constant: All uppercase with underscores
#define SYS_MCLK (11059200/12)
#define LCD1602_RS P1^0
// sbit: All uppercase or snake_case
sbit LED = P0^0;
sbit EN_LED = P1^4;Alternative Style: camelCase/PascalCase
Use this style when explicitly requested by the user:
c
// Function: PascalCase
void ConfigTimer0(unsigned int ms);
void LcdWriteCmd(unsigned char cmd);
void InitDS1302(void);
// Variable: camelCase
unsigned char flag500ms = 0;
unsigned char cntRxd = 0;
// Macro/Constant: All uppercase with underscores (same as default)
#define SYS_MCLK (11059200/12)
#define LCD1602_RS P1^0
// sbit: All uppercase or PascalCase
sbit LED = P0^0;
sbit ENLED = P1^4;代码组织
Code Organization
单文件项目
c
#include <reg52.h>
// 宏定义
// sbit 定义
// 全局变量
// 函数声明
void main() { }
// 函数实现
// 中断函数多文件项目
config.h - 全局配置 (类型定义、系统参数、IO 定义)
module.h - 模块头文件 (结构体、extern 声明、函数声明)
module.c - 模块实现 (#define _MODULE_C)
main.c - 主程序Single-file Project
c
#include <reg52.h>
// Macro definitions
// sbit definitions
// Global variables
// Function declarations
void main() { }
// Function implementations
// Interrupt functionsMulti-file Project
config.h - Global configurations (type definitions, system parameters, IO definitions)
module.h - Module header file (structures, extern declarations, function declarations)
module.c - Module implementation (#define _MODULE_C)
main.c - Main program常用模式
Common Patterns
定时器配置
c
void config_timer0(unsigned int ms)
{
unsigned long tmp;
tmp = 11059200 / 12;
tmp = (tmp * ms) / 1000;
tmp = 65536 - tmp;
tmp = tmp + 12;
T0RH = (unsigned char)(tmp>>8);
T0RL = (unsigned char)tmp;
TMOD &= 0xF0;
TMOD |= 0x01;
TH0 = T0RH;
TL0 = T0RL;
ET0 = 1;
TR0 = 1;
}中断服务函数
c
void interrupt_timer0() interrupt 1
{
static unsigned char tmr_500ms = 0;
TH0 = T0RH;
TL0 = T0RL;
if (++tmr_500ms >= 50)
{
tmr_500ms = 0;
flag_500ms = 1;
}
}标志位驱动
c
bit flag_500ms = 0;
// 中断中设置
flag_500ms = 1;
// 主循环检测
while (1)
{
if (flag_500ms)
{
flag_500ms = 0;
// 执行任务
}
}按键消抖
c
void key_scan(void)
{
static unsigned char keybuf[4] = {0xFF, 0xFF, 0xFF, 0xFF};
keybuf[i] = (keybuf[i] << 1) | KEY_IN[i];
if ((keybuf[i] & 0x0F) == 0x00)
key_sta[i] = 0; // 稳定按下
}查表法
c
unsigned char code led_char[] = {
0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8,
0x80, 0x90, 0x88, 0x83, 0xC6, 0xA1, 0x86, 0x8E
};
P0 = led_char[num];Timer Configuration
c
void config_timer0(unsigned int ms)
{
unsigned long tmp;
tmp = 11059200 / 12;
tmp = (tmp * ms) / 1000;
tmp = 65536 - tmp;
tmp = tmp + 12;
T0RH = (unsigned char)(tmp>>8);
T0RL = (unsigned char)tmp;
TMOD &= 0xF0;
TMOD |= 0x01;
TH0 = T0RH;
TL0 = T0RL;
ET0 = 1;
TR0 = 1;
}Interrupt Service Function
c
void interrupt_timer0() interrupt 1
{
static unsigned char tmr_500ms = 0;
TH0 = T0RH;
TL0 = T0RL;
if (++tmr_500ms >= 50)
{
tmr_500ms = 0;
flag_500ms = 1;
}
}Flag-driven
c
bit flag_500ms = 0;
// Set in interrupt
flag_500ms = 1;
// Detect in main loop
while (1)
{
if (flag_500ms)
{
flag_500ms = 0;
// Execute task
}
}Key Debouncing
c
void key_scan(void)
{
static unsigned char keybuf[4] = {0xFF, 0xFF, 0xFF, 0xFF};
keybuf[i] = (keybuf[i] << 1) | KEY_IN[i];
if ((keybuf[i] & 0x0F) == 0x00)
key_sta[i] = 0; // Stable pressed
}Lookup Table Method
c
unsigned char code led_char[] = {
0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8,
0x80, 0x90, 0x88, 0x83, 0xC6, 0xA1, 0x86, 0x8E
};
P0 = led_char[num];注释风格
Comment Style
c
/* 文件头注释 */
/*
*******************************************************************************
* 文件名称: main.c
* 描 述: 功能描述
* 版本号: v1.0.0
*******************************************************************************
*/
/* 函数注释 */
void lcd_show_str(unsigned char x, unsigned char y,
unsigned char *str, unsigned char len)
// 行内注释
EA = 1; // 使能总中断
config_timer0(10); // 配置 T0 定时 10msc
/* File Header Comment */
/*
*******************************************************************************
* File Name: main.c
* Description: Function description
* Version: v1.0.0
*******************************************************************************
*/
/* Function Comment */
void lcd_show_str(unsigned char x, unsigned char y,
unsigned char *str, unsigned char len)
// In-line Comment
EA = 1; // Enable global interrupt
config_timer0(10); // Configure T0 for 10ms timing代码风格特点
Code Style Features
- 简洁直接, 避免过度抽象
- 直接操作硬件寄存器
- 使用 sbit 定义 IO 口
- 中断向量号直接指定 (interrupt 1)
- 存储区域修饰符 (code, pdata)
- 标志位驱动主循环
- 查表法处理数据映射
- Concise and straightforward, avoiding over-abstraction
- Directly operate hardware registers
- Use sbit to define IO ports
- Directly specify interrupt vector numbers (interrupt 1)
- Memory area modifiers (code, pdata)
- Flag-driven main loop
- Lookup table method for data mapping
最佳实践
Best Practices
命名风格选择
Naming Style Selection
默认使用蛇形命名 (snake_case):
- 符合 Linux 内核和大多数开源嵌入式项目的风格
- 单词分隔明显, 可读性好
- 适合生产项目和团队协作
使用驼峰命名 (camelCase/PascalCase) 的场景:
- 教学项目或参考教材使用驼峰命名
- 团队明确要求使用驼峰命名
- 需要与现有驼峰命名代码保持一致
重要原则: 整个项目必须统一命名风格
Default to snake_case:
- Complies with the style of Linux kernel and most open-source embedded projects
- Clear word separation, good readability
- Suitable for production projects and team collaboration
Scenarios for using camelCase:
- Teaching projects or reference materials use camelCase
- Team explicitly requires camelCase
- Need to maintain consistency with existing camelCase code
Important Principle: The naming style must be consistent throughout the entire project
项目创建建议
Project Creation Suggestions
- 单文件项目: 适用于简单功能 (LED 闪烁、按键检测等)
- 多文件项目: 适用于多外设协同 (LCD + 按键 + 串口等)
- 模块化驱动: 每个外设独立为 .h/.c 文件对
- Single-file project: Suitable for simple functions (LED blinking, key detection, etc.)
- Multi-file project: Suitable for multi-peripheral collaboration (LCD + key + UART, etc.)
- Modular driver: Each peripheral is an independent .h/.c file pair
代码优化建议
Code Optimization Suggestions
- 命名规范: 默认使用蛇形命名 (函数/变量), 宏用全大写下划线
- 中断精简: 中断函数只设置标志位, 主循环处理业务逻辑
- 硬件抽象: 使用宏定义隔离硬件相关代码, 便于移植
- 注释清晰: 关键寄存器配置必须添加行内注释
- Naming specification: Use snake_case by default (functions/variables), all uppercase with underscores for macros
- Interstreamlining: Interrupt functions only set flags, main loop handles business logic
- Hardware abstraction: Use macro definitions to isolate hardware-related code for easy porting
- Clear comments: In-line comments must be added for critical register configurations
常见错误处理
Common Error Handling
问题 1: 定时器不准确
c
// 错误: 未考虑指令周期补偿
tmp = 65536 - tmp;
// 正确: 加上补偿值
tmp = 65536 - tmp;
tmp = tmp + 12; // 补偿中断响应延迟问题 2: 按键抖动
c
// 错误: 直接读取按键状态
if (KEY == 0) { /* 处理 */ }
// 正确: 使用移位寄存器消抖
keybuf = (keybuf << 1) | KEY;
if ((keybuf & 0x0F) == 0x00) { /* 稳定按下 */ }问题 3: 全局变量冲突
c
// 错误: 多个模块使用相同变量名
unsigned char flag; // 在多个 .c 文件中定义
// 正确: 使用模块前缀或 static
static unsigned char uart_flag; // 仅本文件可见
extern unsigned char g_system_flag; // 全局共享Problem 1: Inaccurate timer
c
// Error: Instruction cycle compensation not considered
tmp = 65536 - tmp;
// Correct: Add compensation value
tmp = 65536 - tmp;
tmp = tmp + 12; // Compensate for interrupt response delayProblem 2: Key bouncing
c
// Error: Directly read key status
if (KEY == 0) { /* Process */ }
// Correct: Use shift register for debouncing
keybuf = (keybuf << 1) | KEY;
if ((keybuf & 0x0F) == 0x00) { /* Stable pressed */ }Problem 3: Global variable conflict
c
// Error: Multiple modules use the same variable name
unsigned char flag; // Defined in multiple .c files
// Correct: Use module prefix or static
static unsigned char uart_flag; // Visible only in this file
extern unsigned char g_system_flag; // Globally shared注意事项
Notes
- 中断函数必须尽快返回, 避免长时间占用
- 全局变量在中断和主循环共享时, 使用 volatile 修饰
- 大数组使用 code 修饰符存储在 ROM 中, 节省 RAM
- 延时函数不要在中断中调用
- 多文件项目必须使用 extern 正确声明全局变量
- Interrupt functions must return as soon as possible to avoid long-term occupation
- When global variables are shared between interrupts and main loop, use volatile modifier
- Large arrays use code modifier to store in ROM, saving RAM
- Do not call delay functions in interrupts
- Multi-file projects must use extern to correctly declare global variables
技术参数说明
Technical Parameter Description
支持的芯片平台
Supported Chip Platforms
- 51 系列: AT89C51, STC89C52, STC15 等
- STM32 系列: F103, F407 等 (需调整寄存器定义)
- 其他 8051 兼容芯片
- 51 series: AT89C51, STC89C52, STC15, etc.
- STM32 series: F103, F407, etc. (register definitions need to be adjusted)
- Other 8051-compatible chips
代码规范来源
Code Specification Source
基于《手把手教你学51单片机》实际教学项目:
- 302 个 .c 源文件
- 66 个 .h 头文件
- 涵盖 20+ 种常用外设驱动
- 默认采用蛇形命名, 更符合嵌入式 C 项目规范
Based on actual teaching projects from "Learn 51 MCU Step by Step":
- 302 .c source files
- 66 .h header files
- Covers more than 20 common peripheral drivers
- Uses snake_case by default, which is more in line with embedded C project specifications
输出格式
Output Format
- 单文件项目: 生成 main.c
- 多文件项目: 生成 config.h, module.h, module.c, main.c
- 驱动模块: 生成 driver.h, driver.c
- Single-file project: Generates main.c
- Multi-file project: Generates config.h, module.h, module.c, main.c
- Driver module: Generates driver.h, driver.c
示例场景
Example Scenarios
场景 1: 快速原型验证
Scenario 1: Rapid Prototype Verification
用户需求: "创建一个 LED 每秒闪烁的程序"
助手输出:
- 单个 main.c 文件
- 包含定时器配置和中断函数
- 使用标志位驱动 LED 翻转
User requirement: "Create a program that blinks an LED every second"
Assistant output:
- Single main.c file
- Includes timer configuration and interrupt function
- Uses flag-driven LED toggling
场景 2: 模块化项目
Scenario 2: Modular Project
用户需求: "创建一个带 LCD1602 显示温度和按键调节的项目"
助手输出:
- config.h (系统配置)
- lcd1602.h/c (LCD 驱动)
- key.h/c (按键驱动)
- ds18b20.h/c (温度传感器驱动)
- main.c (主程序)
User requirement: "Create a project with LCD1602 temperature display and key adjustment"
Assistant output:
- config.h (system configurations)
- lcd1602.h/c (LCD driver)
- key.h/c (key driver)
- ds18b20.h/c (temperature sensor driver)
- main.c (main program)
场景 3: 代码优化
Scenario 3: Code Optimization
用户需求: "优化这段串口接收代码"
助手操作:
- 检查命名规范
- 优化缓冲区管理
- 添加溢出保护
- 补充注释说明
基于实际教学项目总结, 注重实用性和可读性。默认使用蛇形命名风格, 符合嵌入式 C 项目规范。
User requirement: "Optimize this UART reception code"
Assistant actions:
- Check naming specifications
- Optimize buffer management
- Add overflow protection
- Add supplementary comments
Summarized based on actual teaching projects, focusing on practicality and readability. Uses snake_case by default, which complies with embedded C project specifications.