基于CircuitPython与CRICKIT的仿生机械手制作:从PWM控制到交互实现
1. 项目概述:从零打造一个会“听话”的机械手
如果你对机器人、自动化或者仅仅是让东西“动起来”感兴趣,那么用微控制器控制伺服电机绝对是一个绕不开的经典课题。这不仅仅是让一个舵机转来转去那么简单,它背后是一整套关于信号控制、机械传动和交互逻辑的学问。今天,我想分享一个我最近完成的、特别有意思的项目:一个基于CircuitPython和CRICKIT扩展板的仿生机械手。这个项目的魅力在于,它用最朴素的材料——纸板、吸管和橡皮筋,结合现代的开源硬件和编程环境,实现了一个既直观又富有教育意义的交互装置。你可以通过触摸电容板来远程操控每一根手指的弯曲,看着自己制作的机械手随着你的触摸做出响应,那种成就感是无可比拟的。
这个项目非常适合创客、教育工作者以及对嵌入式系统和机器人入门感兴趣的爱好者。无论你是想学习如何用CircuitPython编程控制多个伺服电机,还是想了解如何将电容触摸这种自然的交互方式集成到硬件项目中,亦或是单纯想做一个酷炫的桌面玩具,它都能提供一条清晰、有趣的实践路径。整个制作过程融合了机械结构设计、电子电路连接和软件编程,是一个典型的“机电一体化”微型项目。接下来,我会拆解每一个步骤,从材料准备、机械组装、电路连接到代码编写,并分享我在实操中踩过的坑和总结出的技巧,希望能让你在复现时少走弯路。
2. 核心硬件选型与原理深度解析
在动手之前,搞清楚我们手中的“武器”至关重要。这个项目的核心硬件是Circuit Playground Express(简称CPX)和CRICKIT扩展板。选择它们,并非偶然,而是基于其在教育、创客领域的独特优势。
2.1 为什么是CircuitPython与CRICKIT?
CircuitPython是Adafruit主导开发的一款基于Python的开源微控制器编程语言。对于初学者和快速原型开发者来说,它最大的优势是“简单”。你不需要复杂的IDE和烧录工具,只需将开发板通过USB连接到电脑,它就会显示为一个名为CIRCUITPY的U盘。你把写好的.py代码文件拖进去,它就能自动运行。这种“即写即用”的特性,极大地降低了嵌入式开发的门槛,让你能更专注于逻辑本身,而不是环境配置。
CRICKIT(Creative Robotics & Interactive Construction Kit)则是一个为CPX量身定做的“力量扩展板”。CPX本身功能强大,集成了加速度计、光线传感器、电容触摸、LED灯环等,但其驱动能力有限,特别是对于需要较大电流的伺服电机或直流电机。CRICKIT完美地弥补了这一短板。它通过I2C接口与CPX通信,自身集成了4路大电流伺服电机驱动、2路直流电机/步进电机驱动、4路大功率“驱动链”输出、4路电容触摸输入、一个音频功率放大器以及多个 Grove/Qwiic 接口。这意味着,CPX可以专注于“思考”(运行逻辑代码),而CRICKIT负责“出力”(驱动各种执行器),分工明确,稳定可靠。
伺服电机我们选用的是常见的TowerPro SG-5010标准舵机。它的工作电压通常在4.8V-6V,扭矩适中,非常适合驱动我们这种轻量级的纸板手指。舵机内部包含一个小型直流电机、减速齿轮组和一个控制电路。它不像普通电机那样通电就转,而是通过接收来自控制器的PWM(脉冲宽度调制)信号来精确控制输出轴的角度。这是整个项目运动控制的核心。
2.2 PWM控制舵机:不仅仅是“给个信号”
很多人知道用PWM控制舵机,但未必清楚其底层原理。这里我多解释几句,理解了它,你就能举一反三,控制更多类型的执行器。
舵机接收的PWM信号,其核心参数是脉冲的高电平持续时间。对于一个周期为20毫秒(即频率50Hz)的标准PWM信号:
- 当高电平持续时间为1.5毫秒时,舵机输出轴会转到中间位置(例如180度行程的舵机转到90度)。
- 当高电平持续时间缩短到1.0毫秒时,输出轴转向最小角度(如0度)。
- 当高电平持续时间延长到2.0毫秒时,输出轴转向最大角度(如180度)。
这个1.0ms到2.0ms的脉冲宽度,对应着0到180度的角度范围,是一种线性映射关系。CRICKIT板载的伺服驱动芯片,正是根据我们通过代码(如servo.angle = 90)设定的角度,自动生成对应宽度的PWM信号,省去了我们手动计算和生成脉冲的麻烦。
注意:不同品牌、型号的舵机,其中位脉冲宽度和角度范围可能略有差异。SG-5010是标准舵机,兼容性很好。如果你使用其他舵机,发现角度不准,可能需要在代码中调整脉宽范围。幸运的是,CircuitPython的
adafruit_motor.servo库通常能很好地处理标准舵机。
电容触摸是另一个亮点。CRICKIT上的4个电容触摸焊盘,本质上是利用人体(导体)接近时引起的微小电容变化来检测触摸。相比物理按钮,它无需按压,交互更自然、更有“科技感”。在代码中,我们通过读取crickit.touch_X.value的真假值来判断是否被触摸,逻辑清晰直观。
3. 机械结构制作:从纸板到灵巧的“手”
机械部分是整个项目的基础,也是最需要耐心和巧思的地方。一个结构合理、传动顺畅的机械手,是后续稳定控制的前提。
3.1 材料准备与手部模板制作
材料清单(除电子件外):
- 瓦楞纸板:建议使用较厚的快递箱纸板,它强度足够且易于切割。关键点在于纹理方向。纸板内部的波浪形夹层(瓦楞)是有方向的。制作时,务必让瓦楞的纹路垂直于手指方向(即竖直于手指)。这样,手指在关节处弯曲时,纸板会顺着瓦楞纹路产生折痕,弯曲更顺滑、不易断裂。如果纹路平行于手指,弯曲会非常困难且容易损坏。
- 吸管:用作“肌腱”导管。普通塑料吸管即可,它的作用是让牵引线(棉绳)在内部顺畅滑动,减少摩擦。
- 牵引线:棉绳、风筝线或结实的缝纫线都可以。要求是表面光滑、耐磨、不易拉伸。我实测下来,蜡棉绳效果很好,表面有蜡层,非常顺滑。
- 橡皮筋:用作“回位肌腱”。当舵机放松牵引线时,依靠橡皮筋的弹力将手指拉回伸直状态。需要选择弹力适中、长度足够的橡皮筋。
- 热熔胶枪与胶棒:用于固定吸管、牵引线头等。热熔胶固化快,粘接纸板效果不错。
- 美工刀或剪刀:精细切割用。
制作手型模板:
- 将你的左手(或右手)平放在纸板上,用铅笔仔细描出手掌和五指的轮廓。注意:大拇指的机械结构相对复杂(多一个关节且运动平面不同),为了简化首个项目,原设计只控制食指、中指、无名指和小指四指。你可以选择不描画拇指,或者描出后将其固定不动。
- 沿着轮廓线,用美工刀仔细切割。切割时下面垫一块切割垫或废旧木板,保护桌面和刀尖。线条尽量平滑,转角处可以多划几刀,避免撕扯纸板。
3.2 关节铰链与传动系统搭建
这是让纸板手“活”起来的关键步骤。
1. 制作活动关节:
- 在刚刚切好的纸板手上,用铅笔在每个手指的指关节处(两个)和手掌与手指的连接处做好标记。
- 用美工刀的刀尖,沿着标记线轻轻划出刻痕。注意,不是切透,只是划破最表层的纸皮。这个刻痕将成为关节的“铰链”,引导手指在固定位置弯曲。划好后,轻轻弯折每个关节,让其预先形成一定的弯曲趋势,这样后续被拉线时动作会更自然。
2. 安装“肌腱”导管(吸管):
- 将吸管剪成小段,每段长度略短于每个指节(从关节到指尖)。原则是:当手指弯曲时,相邻指节上的吸管段不能相互碰撞,否则会卡住。
- 用热熔胶将吸管段固定在每个指节的背面(即手心那一面)。胶要涂在吸管两侧,确保粘牢。确保吸管通道畅通,棉绳能轻松穿过。
3. 穿入“肌腱”(棉绳)与设置“回位肌腱”(橡皮筋):
- 剪出4根长约45厘米的棉绳,分别对应四根手指。
- 将棉绳从指尖端的吸管穿入,从手掌根部的吸管穿出。在指尖背面,将棉绳头折回一小段,用热熔胶牢牢固定。固定后,轻轻拉扯棉绳,测试手指是否能顺畅弯曲。技巧:穿线前,可以用打火机轻微灼烧一下棉绳头,使其硬化,更容易穿过吸管。
- 安装橡皮筋:这是实现手指自动伸直的核心。将一根橡皮筋剪开成一条。一端用热熔胶固定在指尖背面。然后,在手掌心对应手指根部的位置,用铅笔尖戳一个小孔。将橡皮筋的另一端穿过小孔,拉到手掌背面。此时,轻轻拉动棉绳使手指弯曲,然后调整橡皮筋在手掌背面的长度,并打结固定。调整的目标是:当松开棉绳时,橡皮筋的拉力能刚好将手指拉回完全伸直状态;当拉紧棉绳时,又能克服橡皮筋拉力使手指弯曲。这个需要一点耐心微调,是影响手部动作是否自然的关键。
实操心得:橡皮筋的拉力不是越强越好。拉力太强,舵机可能拉不动或者需要很大电流;拉力太弱,手指回位无力,显得松垮。一个简单的测试方法是:固定好橡皮筋后,手动拉动棉绳,感觉手指弯曲和回弹的力度是否适中、顺滑。理想状态是有一点阻力,但动作清晰果断。
4. 整体安装与舵机布局:
- 找一个大小合适的纸盒作为底座。将制作好的手部用螺丝或强力胶固定在纸盒一侧。
- 将4个舵机在纸盒内布局。核心原则是:为每个舵机的摆臂(舵盘)留出足够的旋转空间,确保它们彼此之间以及和纸盒壁之间不会发生干涉。规划好位置后,在纸盒上开孔,将舵机塞进去卡紧,舵机轴朝上。
- 将所有舵机摆臂拆下,将舵机轴手动旋转至逆时针最大角度(这通常是0度位置)。然后,将摆臂以大约钟表1点钟的方向重新安装到舵机上。这个初始角度很重要,它决定了棉绳的初始张紧度。1点钟方向能提供一个适中的预紧张力。
5. 连接“肌腱”到舵机:
- 将棉绳的另一端穿过舵机摆臂末端的小孔。
- 关键步骤:调整棉绳长度并固定。轻轻拉紧棉绳,直到对应的手指刚刚开始有轻微的弯曲(但还未完全弯曲)。保持这个张力,将棉绳在舵机摆臂上缠绕固定。我推荐使用“八字结”或类似的方法:在摆臂上绕一个“8”字形,最后将绳尾压在最下面一圈的绳子下面拉紧,这样摩擦力很大,不易松脱。务必确保缠绕牢固,否则运行中松脱会很麻烦。
4. 电路连接与系统集成
机械部分完成后,我们来连接电子部分。这部分相对简单,但务必仔细,避免接错。
4.1 硬件连接步骤
- CPX与CRICKIT连接:将Circuit Playground Express严丝合缝地扣在CRICKIT扩展板中央的插座上。它们之间通过边缘连接器通信,无需焊接。
- 舵机连接:将4个舵机的三线接口(信号-黄/白线,电源-红线,地线-棕/黑线)分别连接到CRICKIT上标有
Servo 1至Servo 4的端口。注意方向:确保舵机插头的信号线(通常是黄色或白色)朝向CRICKIT板子的外侧边缘。这是标准接法。 - 电源连接:这是保证系统稳定运行的重中之重。你需要一个5V/4A以上的直流电源适配器(中心正极,2.1mm接口)。将其插入CRICKIT的
DC IN口。绝对不要试图通过CPX的USB口来为整个系统供电,USB口无法提供舵机同时运动时所需的大电流,会导致CPX重启或舵机抖动无力。 - 扬声器连接(用于二进制计数项目):如果你要做带语音的二进制计数功能,需要一个4Ω或8Ω的扬声器。将其两根线不分正负地拧在CRICKIT的
Speaker端子座上。
4.2 电源与接地的关键考量
伺服电机是“电老虎”,尤其在启动和堵转(被卡住)时,瞬时电流可以轻松超过1A。四个舵机同时动作,对电源是个考验。
- 电源规格:务必使用稳压、足功率的5V电源。标称5V/4A的开关电源是安全的选择。劣质或功率不足的电源在负载变大时电压会下降,导致舵机工作异常(抖动、角度不准)甚至CPX重启。
- 接地回路:确保所有设备共地。在这个项目中,舵机的电源地和信号地、CPX的地、CRICKIT的地以及外部电源的地,都通过CRICKIT板子内部连接在了一起,形成了一个共同的接地参考点,这很重要,能避免信号干扰。
- 上电顺序:建议先连接好所有信号线,最后再接通外部电源。断电时,先断外部电源。
注意事项:在调试代码时,如果频繁地、快速地让所有舵机同时从0度转到180度,可能会触发电源的过流保护或导致电压骤降。在代码中为舵机动作添加适当的延时(例如
time.sleep(0.2)),错开它们的启动时间,是一个好习惯。原版代码中在循环检测触摸时没有加延时,在实际操作中,如果非常快速地连续触摸多个电容板,偶尔会观察到CPX指示灯闪烁(复位迹象)。我建议在主循环while True:中增加一个很小的延时,如time.sleep(0.01),既能降低CPU占用,也能给电源一点缓冲时间。
5. 软件编程:从交互操控到二进制计数
硬件就绪后,就到了赋予它“灵魂”的编程环节。我们将使用CircuitPython,分两个模式来编程:交互式木偶模式和有语音的二进制计数模式。
5.1 环境准备与基础代码结构
首先,确保你的CPX已经刷入了支持CRICKIT的特殊版本CircuitPython固件,并且将必要的库文件(主要是adafruit_crickit)复制到了CPX的lib文件夹中。这些准备工作在Adafruit的官方指南中有详细步骤。
我们以交互式木偶模式的代码为例,进行深度解析。理解了这个,二进制计数模式就很容易举一反三。
# SPDX-FileCopyrightText: 2018 John Edgar Park for Adafruit Industries # SPDX-License-Identifier: MIT import board from digitalio import DigitalInOut, Direction, Pull from adafruit_crickit import crickit # 1. 设置CPX上的滑动开关 switch = DigitalInOut(board.SLIDE_SWITCH) switch.direction = Direction.INPUT switch.pull = Pull.UP代码解读1:滑动开关设置这里初始化了CPX板载的滑动开关。Pull.UP启用了内部上拉电阻。当开关拨到右侧(靠近复位按钮)时,物理上连接到GND,switch.value读取为False(0);拨到左侧时,switch.value为True(1)。我们用它作为总开关,非常方便。
# 2. 设置4个舵机 servos = (crickit.servo_1, crickit.servo_2, crickit.servo_3, crickit.servo_4) for servo in servos: servo.angle = 180 # 初始角度,张开手代码解读2:舵机初始化创建一个包含4个舵机对象的元组。初始化时将所有舵机角度设为180度(根据你的机械安装,这对应手指完全伸直的状态)。crickit.servo_X对象已经封装了PWM生成的细节,我们直接操作角度即可。
# 3. 设置4个电容触摸 touches = (crickit.touch_1, crickit.touch_2, crickit.touch_3, crickit.touch_4) cap_state = [False, False, False, False] cap_justtouched = [False, False, False, False] cap_justreleased = [False, False, False, False] curl_finger = [False, False, False, False] finger_name = ['Index', 'Middle', 'Ring', 'Pinky']代码解读3:电容触摸与状态变量
touches: 电容触摸对象元组。cap_state: 记录每个触摸垫当前是否被触摸的状态。cap_justtouched/cap_justreleased: 这两个是边缘检测的关键。它们记录的是“刚刚被触摸”和“刚刚被释放”的瞬间事件,用于触发一次性的动作(如弯曲手指),而不是持续按住时的重复触发。curl_finger: 可以用于实现“点按切换”模式(本例中未完全使用)。finger_name: 用于调试打印的友好名称。
5.2 主循环逻辑与状态机
核心逻辑都在while True:循环中,这是一个典型的状态机应用。
while True: if not switch.value: # 如果开关关闭(拨到右边) continue # 跳过本次循环,什么都不做 # 检查电容触摸 for i in range(4): # 每次循环先重置“刚刚”状态 cap_justtouched[i] = False cap_justreleased[i] = False if touches[i].value: # 如果检测到触摸 if not cap_state[i]: # 并且之前的状态是“未触摸” cap_justtouched[i] = True # 标记为“刚刚触摸” print(f"{finger_name[i]} finger bent.") servos[i].angle = 0 # 弯曲手指(0度) cap_state[i] = True # 更新当前状态为“触摸中” else: # 如果未检测到触摸 if cap_state[i]: # 但之前的状态是“触摸中” cap_justreleased[i] = True # 标记为“刚刚释放” print(f"{finger_name[i]} finger straightened.") servos[i].angle = 180 # 伸直手指(180度) cap_state[i] = False # 更新当前状态为“未触摸”逻辑精髓:这段代码实现了“按下-动作,松开-复位”的直观交互。关键在于cap_state这个状态变量。它像一个记忆单元,记住了每个触摸垫上一帧的状态。通过对比当前触摸值touches[i].value和上一帧的状态cap_state[i],我们就能精确判断出是“从无到有”(触摸开始)还是“从有到无”(触摸结束)的边缘事件,并只在边缘事件发生时驱动舵机动作一次。这避免了在持续触摸期间舵机不断收到指令而抖动。
5.3 二进制计数与语音播放模式
第二个程序在第一个的基础上,增加了语音播放和自动序列控制的功能。其核心是一个预定义的列表counting,它定义了数字1到15分别对应要抬起哪几根手指(用舵机索引表示)。
counting = ( [3], # 数字1: 食指 (索引3,因为列表从0开始,对应servo_4? 这里需要根据你的接线确认映射) [2], # 数字2: 中指 [3, 2], # 数字3: 食指+中指 [1], # 数字4: 无名指 [1, 3], # 数字5: 无名指+食指 ... # 以此类推至15 )重要提示:原代码中的索引映射需要根据你的实际接线来理解。servos = (crickit.servo_1, crickit.servo_2, crickit.servo_3, crickit.servo_4),那么servos[0]对应servo_1。在counting列表中,[3]表示操作servos[3],即servo_4。你需要确保servo_4连接的是你定义为“食指”的舵机。建议在代码中通过打印语句或单独测试,确认每个索引对应的物理手指。
语音播放部分使用了audioio库。代码定义了一个play_file函数,它打开.wav文件并播放,同时通过while cpx_audio.playing:循环阻塞,直到播放完毕才执行下一步。这使得动作和语音可以同步。
6. 调试、优化与问题排查实录
即使按照步骤操作,你也可能会遇到一些问题。这里记录了我遇到的一些典型情况及其解决方法。
6.1 机械结构常见问题
| 问题现象 | 可能原因 | 解决方案 |
|---|---|---|
| 手指弯曲不顺畅,有卡顿 | 1. 吸管段过长,弯曲时相互碰撞。 2. 关节刻痕不够深或太浅。 3. 棉绳在吸管内摩擦阻力大。 | 1. 修剪吸管,确保手指弯曲到最大角度时,相邻吸管不接触。 2. 重新用刀尖加深刻痕,并反复弯折几次使其灵活。 3. 更换更光滑的线材(如蜡绳),或在穿线前在吸管内滴一滴润滑油(慎用,避免污染)。 |
| 手指无法完全回位(伸直) | 1. 橡皮筋拉力不足或老化。 2. 棉绳缠绕在舵机摆臂上过紧,有残余张力。 3. 关节处纸板因潮湿或多次弯曲变软,产生塑性变形。 | 1. 更换弹力更强的橡皮筋,或并联一根橡皮筋增加拉力。 2. 松开棉绳,在舵机初始位置(手指自然伸直)时重新固定,确保无预紧力。 3. 在关节处背面用热熔胶或胶带粘贴一根牙签或细竹签作为“加强筋”,只允许其在刻痕处弯曲。 |
| 舵机有“滋滋”声且发热 | 1. 舵机在到达目标角度后仍被持续施加力(“堵转”),如橡皮筋拉力与舵机保持力对抗。 2. 机械结构卡死,舵机无法转到指令角度。 | 1. 这是正常现象,舵机在维持位置时需要持续供电。但应检查机械结构是否顺滑,减少不必要的阻力。 2. 立即断电!手动检查手指和传动机构是否被异物卡住。调整机械结构,确保全程运动无阻碍。 |
6.2 电气与软件问题
| 问题现象 | 可能原因 | 解决方案 |
|---|---|---|
| 触摸电容板无反应 | 1. CPX与CRICKIT连接不牢。 2. 代码未正确上传或文件名错误。 3. 电源未接通或功率不足。 | 1. 重新拔插CPX,确保完全扣紧。 2. 确认代码文件已保存为 code.py或main.py到CPX根目录。通过Mu编辑器的串行REPL查看是否有错误输出。3. 检查5V/4A电源是否已接入CRICKIT的DC口,并打开开关。 |
| 单个舵机不动作或动作相反 | 1. 舵机接线错误(信号线接反)。 2. 舵机损坏。 3. 代码中舵机索引与实际手指映射错误。 | 1. 检查舵机三线接口是否完全插入CRICKIT,黄线在外侧。 2. 将该舵机换到其他已知正常的端口测试。 3. 在代码中单独测试每个舵机,例如在循环中让 servo_1从0度转到180度,确认其控制的是哪根手指。 |
| 同时触摸多个电容板时系统复位(CPX重启) | 多个舵机同时启动导致瞬时电流过大,电源电压被拉低,造成CPX欠压复位。 | 这是最常见的问题。优化方案: 1.升级电源:使用额定电流更大的5V电源(如5V/6A)。 2.软件错峰:在主循环 while True:末尾增加一个短暂延时,如time.sleep(0.01)。或者在驱动舵机动作的函数中,为每个舵机动作之间添加微小延时,错开其启动瞬间。3.增加电容:在CRICKIT的电源输入端并联一个大容量(如1000uF)的电解电容,可以缓冲瞬时电流需求。 |
| 二进制计数模式语音播放卡顿或无声音 | 1. 音频文件格式或位置不正确。 2. 扬声器阻抗不匹配或接触不良。 3. 内存不足。 | 1. 确保.wav文件是单声道、16-bit、22050Hz采样率的格式,并已直接放在CPX的根目录下。 2. 检查扬声器线是否牢固连接在CRICKIT的Speaker端子上。尝试更换一个扬声器。 3. CircuitPython内存有限,过大的音频文件或复杂的代码可能造成问题。确保只保留了项目必要的文件。 |
6.3 性能优化与扩展思路
在项目稳定运行后,你可以尝试以下优化和扩展:
- 增加手势记忆与回放:修改代码,记录一段时间内电容触摸的状态序列,并保存下来。然后可以添加一个模式,让机械手自动复现这套动作,实现简单的“编程”与“回放”。
- 引入传感器反馈:在手指指尖内部粘贴一个弯曲传感器或微型压力传感器,通过额外的模拟输入引脚连接到CPX。这样代码不仅能控制手指弯曲,还能读取手指的弯曲程度或是否接触到物体,实现简单的闭环控制或更复杂的交互。
- 无线控制:为CPX添加蓝牙或Wi-Fi模块,通过手机App或电脑上的图形化界面来远程控制机械手,摆脱电容触摸板的物理限制。
- 结构强化与美化:使用激光切割的亚克力板或3D打印部件替换纸板,制作更坚固、更美观的机械手结构。可以设计更符合人体工学的指关节和更高效的传动机构。
这个项目从看似简单的材料开始,却完整地串联了机械设计、电子硬件和软件编程的知识点。最重要的是,它充满了动手的乐趣和即刻可见的成果。当你第一次触摸电容板,看到自己制作的机械手指随之而动时,那种连接数字世界与物理世界的奇妙感觉,正是创客精神的精髓所在。希望这份详细的指南和心得,能帮助你顺利打造出自己的仿生机械手,并在此基础上探索出更多可能。
