
本文介绍如何使用 micropython 在树莓派 pico 上实现按钮单击一次即循环切换预设的 11 个城市/国家选项,并说明如何优化按键检测逻辑、避免误触发,同时提供兼容无 `itertools` 的轻量级循环迭代器实现。
在 MicroPython 环境(如 Raspberry Pi Pico)中,实现“按钮单击 → 切换下一个选项”的核心难点不在循环本身,而在于可靠检测一次有效按下事件(即消抖 + 边沿触发),并配合一个可无限遍历的选项序列。你当前的代码通过连续两次读取电平并比较来判断状态变化,但存在逻辑缺陷:它试图在 10ms 内捕捉从高到低或低到高的跳变,而机械按钮的抖动时间通常为 5–20ms,且 button.value() 返回的是当前电平(PULL_UP 下:未按下为 1,按下为 0),因此 first and not second 实际检测的是「释放」而非「按下」——这与直觉相反,也极易漏判。
✅ 正确做法是:等待按钮被按下(电平变低)→ 等待其释放(电平恢复高)→ 视为一次有效点击。以下是一个简洁健壮的阻塞式按键等待函数:
import machine
import time
def wait_for_press(button, debounce_ms=20):
"""等待按钮被按下并完全释放,内置硬件消抖"""
# 等待按下(电平由 1→0)
while button.value(): # PULL_UP:未按下时为 True(1)
time.sleep_ms(5)
# 消抖:等待稳定低电平
time.sleep_ms(debounce_ms)
# 等待释放(电平由 0→1)
while not button.value():
time.sleep_ms(5)
# 再次消抖确保已释放
time.sleep_ms(debounce_ms)接下来,定义你的 11 个城市/国家选项(例如用于调用天气或时区 API):
CITIES = [
"Beijing", "Tokyo", "Seoul", "Singapore", "Sydney",
"Los Angeles", "New York", "London", "Paris", "Cairo", "Dubai"
]由于 MicroPython 默认固件不包含 itertools 模块(尤其在 Pico 的 minimal firmware 中),直接使用 itertools.cycle 会报 ImportError。因此我们采用官方文档推荐的手写等效生成器:
立即学习“Python免费学习笔记(深入)”;
def cycle(iterable):
"""轻量级 cycle 实现,兼容所有 MicroPython 版本"""
saved = []
for item in iterable:
yield item
saved.append(item)
while saved:
for item in saved:
yield item
# 创建循环迭代器
city_cycle = cycle(CITIES)
current_city = next(city_cycle) # 初始化为第一个城市完整主循环示例(整合显示与逻辑):
button = machine.Pin(12, machine.Pin.IN, machine.Pin.PULL_UP)
while True:
# 显示当前选项(此处用 print 模拟,实际可驱动 OLED 或串口发送)
print(f"Current city: {current_city}")
# 等待一次有效按钮按下+释放
wait_for_press(button)
# 切换到下一个选项
current_city = next(city_cycle)
# 可选:添加短延时防止连击(若 UI 响应过快)
time.sleep_ms(300)⚠️ 注意事项:
- 引脚配置:确保按钮一端接 GPIO12,另一端接地(GND),启用内部上拉(PULL_UP),这是最简硬件方案;
- 非阻塞扩展:若后续需同时处理网络请求或传感器读取,应将按键检测改为非阻塞轮询(记录上一次状态+时间戳),避免 while 循环阻塞主线程;
- 内存优化:cycle() 中 saved 列表会完整缓存所有选项,11 个字符串开销极小,但若选项极大(如含图片路径),建议改用索引计数器 idx = (idx + 1) % len(CITIES);
- API 调用时机:实际项目中,应在切换 current_city 后立即触发对应 API 请求(注意 MicroPython 的 urequests 限制及 WiFi 连接状态)。
通过以上结构,你即可稳定实现「按一次,换一城」的交互逻辑,为后续集成实时天气、时区转换等功能打下坚实基础。










