**这是本文档旧的修订版!**
测量温度
使用 Raspberry Pi Pico 的内置 ADC 转换模拟输入,并读取其内部温度传感器
在前面的章节中,您一直在使用 Raspberry Pi Pico 上的数字输入。数字输入打开或关闭,二进制状态。当按下按钮开关时,它会将引脚从低、关、高、开改变;当被动红外传感器检测到运动时, 它也一样。 不过,您的 Pico 可以接受另一种类型的输入信号:模拟输入。而数字是 只有打开或关闭,模拟信号可以是从完全关闭到完全打开的任何值——一系列可能的值。模拟输入用于从音量控制到气体、湿度和温度传感器的所有内容,它们通过称为模数转换器 (ADC) 的硬件工作。 在本章中,您将学习如何在 Pico 上使用 ADC - 以及如何利用其内部温度传感器来构建数据记录热量测量小工具。您还将学习一种创建类似模拟输出的技术。为此,您需要 Pico;任何颜色的 LED 和 330 Ω 电阻器;一个 10 kΩ 电位器;以及一系列公对公 (M2M) 跳线。您还需要一根微型 USB 电缆,并将 Pico 连接到 Raspberry Pi 或其他运行 Thonny MicroPython IDE 的计算机。
模数转换器 Raspberry Pi Pico 的 RP2040 微控制器是一种数字设备,就像所有主流微控制器一样:它由数千个晶体管、类似开关的微型设备组成,这些设备可以打开或关闭。因此,您的 Pico 无法真正理解模拟信号 - 可以是完全关闭和完全打开之间的任何频谱 - 不依赖额外的硬件:模数转换器(ADC )。 顾名思义,模数转换器将模拟信号转换为数字信号。无论您仔细观察,您都不会在 Pico 上看到 ADC:它内置于 RP2040 本身中。许多微控制器都有自己的 ADC,就像 RP2040 一样,而那些没有的可以使用连接到一个或多个数字输入的外部 ADC。 ADC 有两个关键特性:以数字位为单位的分辨率,以及它的通道,或者它可以同时接受和转换多少模拟信号。 Pico 中的 ADC 具有 12 位的分辨率,这意味着它可以将模拟信号转换为数字信号作为范围从 0 到 4095 的数字——尽管这是在 MicroPython 中处理的,转换为范围从 0 到 16 位的数字65,535,因此它的行为与其他 MicroPython 微控制器上的 ADC 相同。它具有三个连接到 GPIO 引脚的通道:GP26、GP27 和 GP28,对于模拟通道 0、1 和 2,它们也称为 GP26ADC0、GP27ADC1 和 GP28_ADC2。还有第四个 ADC 通道,它连接到RP2040内置温度传感器;您将在本章后面找到更多相关信息。
为什么是 65,535?
乍一看,数字“65,535”看起来很奇怪——为什么会这样,为什么不是简单的 0-100? 答案与您的 Pico 在二进制数系统上工作的事实有关,其中一个数字的唯一可能值是 0 或 1。一个 16 位二进制数被制作 最多 16 位,最大可能值是 16 个:1111111111111111。如果将其转换回十进制数,人类使用的 0-9 计数系统,您会得到 65,535。
读取电位器 连接到 Pico 模数转换器的每个引脚也可以用作简单的数字输入或输出; 要将其用作模拟输入,您需要一个模拟信号——您可以使用电位计轻松制作一个。 有多种类型的电位器可供使用:有些,例如您在第 7 章中使用的 HC-SR501 被动红外传感器上的电位器,设计为可以用螺丝刀调节; 其他的,通常用于音量控制和其他输入,有旋钮或滑块。 最常见的类型有一个小的,通常是塑料的,从顶部或前面伸出的旋钮:这被称为旋转电位器。
拿起你的电位器,把它翻过来:你会看到它有三个插在面包板上的针脚。 根据您如何连接这些引脚,电位计以两种不同的方式工作。 首先将电位计插入面包板,小心不要弯曲引脚。 使用公对公 (M2M) 跳线将中间引脚连接到 Pico 上的 GP26_ADC0 引脚,如图 8-1 所示——如果 Pico 插入最顶部的面包板,它会 在第 10 行。 最后,再取两根跳线,将电位计的一个外部引脚(无论哪个)连接到面包板的电源轨,将电源轨连接到 Pico 的 3V3 引脚。 打开 Thonny 并开始一个新程序: <code python> import machine import utime
与数字通用输入/输出 (GPIO) 引脚一样,模拟输入引脚由机器库处理——就像数字引脚一样,在使用之前需要设置它们。 继续你的程序: <code python> potentiometer = machine.ADC(26)
这将引脚 GP26ADC0 配置为模数转换器上的第一个通道 ADC0。 要从引脚读取,请设置一个循环: <code python> while True: print(potentiometer.readu16())
utime.sleep(2)
在这个循环中,读取引脚的值并将其打印在一行上:这是将值读入变量然后打印变量的更紧凑的替代方法,但仅当您不想这样做时才有效 除了打印之外的任何阅读 - 这正是该程序目前所需要的。 读取模拟输入几乎与读取数字输入相同,除了一件事:读取数字输入时使用 read(),但使用 read_u16() 读取该模拟输入。最后一部分 u16 只是警告您,您将收到一个 16 位无符号整数,即 0 到 65,535 之间的整数,而不是二进制 0 或 1 结果。 单击运行图标并将您的程序保存为 Potentiometer.py。观看 Shell:您会看到您的程序打印出大量数字,可能超过 60,000。尝试将电位计一直朝一个方向旋转:根据您转动旋钮的方向和您在电路中使用的外腿,数字会上升或下降。反过来:该值将向相反的方向变化。 但是,无论您朝哪个方向转动,它都不会接近 0。那是因为只有两条腿连接,电位计充当称为可变电阻器或变阻器的组件。压敏电阻是一个可以改变值的电阻器——在 10 kΩ 电位计的情况下,介于 0 Ω 和 10,000 Ω 之间。电阻越高,来自 3V3 引脚的电压到达您的模拟输入端的电压就越低——因此数字会下降。电阻越低,到达您的模拟输入的电压就越大——因此数字会增加。 电位计的工作原理是在内部有一个导电条,连接到两个外部引脚,以及一个雨刷或刷子连接到内部引脚(图 8-2)。当您转动旋钮时,雨刷器会靠近条带的一端并远离另一端。雨刮器离您连接到 Pico 3V3 引脚的条带末端越远,电阻越高;离得越近,阻力越小。 压敏电阻是非常有用的组件,但有一个缺点:您会注意到,无论您将旋钮向任一方向旋转多远,您都无法获得 0 值——或接近它的任何值。这是因为 10 kΩ 电阻器的强度不足以将 3V3 引脚的输出降至 0 V。您可以寻找具有更高最大电阻的更大电位器,或者您可以简单地将现有电位器连接起来作为分压器。 作为分压器的电位器 电位器上未使用的引脚不是用来展示的:在电路中添加到该引脚的连接完全改变了电位器的工作方式。单击停止图标以停止您的程序,并抓住两根公对公 (M2M) 跳线。如图 8-3 所示,使用一个将电位计未使用的引脚连接到面包板的接地轨。取另一个并将接地轨连接到 Pico 上的 GND 引脚。
单击运行图标以重新启动程序。 再次转动电位计旋钮,从一个方向一直旋转到另一个方向。 观察打印到 Shell 区域的值:与以前不同,它们现在从接近零变为接近完整的 65,535——但为什么呢? 将接地连接添加到电位计导电条的另一端创建了一个分压器:而在电位计之前只是充当 3V3 引脚和模拟输入引脚之间的电阻器,它现在将 3.3 V 输出之间的电压分压为 3V3 引脚和 GND 引脚的 0 V。 将旋钮完全旋转一个方向,您将获得 3.3V 的 100%; 完全反过来,0%。
零是最难的数字
如果您无法让 Pico 的模拟输入准确读取 0 或 65,535,请不要担心 - 您没有做错任何事情! 所有电子元件都带有公差,这意味着任何声称的价值都不会准确。 就电位器而言,它可能永远不会精确地达到其输入的 0% 或 100%——但它会让你非常接近!
您在 Shell 上看到的数字是模数转换器原始输出的十进制表示——但这并不是最友好的查看方式,尤其是当您忘记 65,535 表示“全电压”时。 不过,有一个简单的方法可以解决这个问题:一个简单的数学方程。 返回您的程序,并在循环上方添加以下内容: <code python> conversion_factor = 3.3 / (65535)
这建立了一种数学方法,可以将模数转换器提供给您的数字转换为它所代表的实际电压的近似值。 第一个数字是引脚可以预期的最大可能电压:3.3 V,来自 Pico 的 3V3 引脚; 第二个数字是模拟输入读数的最大值,65,535。 总而言之,转换因子是由“3.3 除以 65,535”创建的数字 - 最大可能电压除以模数转换器报告的值范围,这反过来又是其分辨率的特征(以位为单位)。 设置转换系数后,您只需在程序中使用它即可。 回到你的循环,编辑它以阅读: <code python> while True:
voltage = potentiometer.read_u16() * conversion_factor print(voltage) utime.sleep(2)
循环内的第一行通过模拟输入引脚从电位计读取读数,并将其(* 符号)乘以您之前在程序中设置的转换因子,将结果存储为可变电压。 然后将该变量打印到 Shell,代替您之前使用的原始读数。 您完成的程序将如下所示: <code python> import machine import utime potentiometer = machine.ADC(26) conversionfactor = 3.3 / (65535) while True: voltage = potentiometer.readu16() * conversion_factor
print(voltage) utime.sleep(2)
线性 VS 对数
如果您发现在一个极限和另一个极限之间缓慢转动电位计会使数字开始缓慢变化,然后开始更快地变化,或者反过来,您几乎可以肯定使用的是对数或对数电位计。 而线性电位器在 在它的整个范围内,对数电位计开始进行微小的变化,然后迅速提高变化的速度。 对数电位器通常用于放大器的音量控制,而线性电位器更常见于基于微控制器的设备,例如 Pico。
单击运行图标。将电位计一直朝一个方向转动,然后再朝另一个方向转动。观察打印到 Shell 区域的数字:您会看到,当电位器一直处于单向位置时,数字非常接近于零;当完全相反时,它们非常接近 3.3。这些数字代表引脚读取的实际电压——当你转动电位器的旋钮时,你正在最小和最大之间平滑地划分电压,0 V 到 3.3 V。 恭喜:您现在知道如何将电位计连接为压敏电阻和分压器,以及如何将模拟输入读取为原始值和电压! 测量温度 Raspberry Pi Pico 的 RP2040 微控制器有一个内部温度传感器,可在第四个模数转换器通道上读取。与电位计一样,传感器的输出是可变电压:随着温度的变化,电压也会发生变化。 启动一个新程序,并导入 machine 和 utime 库: <code python> import machine import utime
再次设置模数转换器,但这次不是使用引脚编号,而是使用连接到温度传感器的通道编号: <code python> sensor_temp = machine.ADC(4)
您将再次需要转换系数,将传感器的原始读数更改为电压值,因此添加: <code python> conversion_factor = 3.3 / (65535) 然后设置一个循环从模拟输入读取读数,应用转换因子,然后 将它们存储在一个变量中:
<code python> while True:
reading = sensor_temp.read_u16() * conversion_factor
但是,您不需要直接打印读数,而是需要进行第二次转换 - 将模数转换器报告的电压转换为摄氏度: <code python> temperature = 27 - (reading - 0.706)/0.001721
这是另一个数学方程,并且特定于 RP2040 中的温度传感器。 这些值取自称为数据表或数据手册的技术文件:所有电子元件都有数据表,通常应制造商的要求提供。 您可以在 rptl.io/rp2040-get-started 的 Pico 文档中查看 RP2040 的数据表——它包含了关于微控制器如何工作的完整信息,尽管它是针对工程师的,因此具有很强的技术性。 最后,完成你的循环: <code python> print(temperature) utime.sleep(2)
Your program will now look like this: <code python> import machine import utime sensortemp = machine.ADC(4) conversionfactor = 3.3 / (65535) while True:
reading = sensor_temp.read_u16() * conversion_factor temperature = 27 - (reading - 0.706)/0.001721 print(temperature) utime.sleep(2)
单击运行图标并将您的程序保存为 Temperature.py。 观察外壳区域:您会看到打印的数字代表传感器报告的温度(以摄氏度为单位)。
热和 RP2040
如果你有一个传统的温度计,你可能会看到你的 Pico 报告的数字要高一点:那是因为温度传感器位于 Pico 的 RP2040 芯片内部,它正在忙于运行你的程序。 当微控制器通电时,它会自行产生热量——而这些热量足以扭曲结果。 对于这样一个简单的程序,偏斜可能不会太高; 如果您的程序进行了大量复杂的计算,则偏斜可能会更高。
尝试将指尖轻轻按在 Pico 中间最大的黑色芯片 RP2040 上,然后将其握在那里:手指的温暖应该会使芯片变暖,温度会升高。将手指从芯片上移开,温度会再次下降。 恭喜 - 您已经将 Pico 变成了温度计!
用 PWM 使 LED 衰减 Pico 中的模数转换器仅以一种方式工作:它将模拟信号转换为微控制器可以理解的数字信号。如果你想走另一条路,让你的数字微控制器创建一个模拟输出,你通常需要一个数模转换器 (DAC)——但是有一种方法可以“伪造”模拟信号,使用一种叫做脉宽调制或PWM。 微控制器的数字输出只能打开或关闭,0 或 1。打开和关闭数字输出称为脉冲,通过改变引脚打开和关闭的速度,您可以改变或调制这些输出的宽度脉冲——因此称为“脉宽调制”。 Pico 上的每个 GPIO 引脚都能够进行脉宽调制,但微控制器的脉宽调制块由八个切片组成,每个切片有两个输出。看图 8-4:你会看到每个引脚都有一个字母和一个数字。数字代表连接到该引脚的 PWM 片;字母表示使用切片的哪个输出。
PWM 冲突
如果您不小心使用了两次相同的 PWM 输出,您就会知道,因为每次更改一个引脚上的 PWM 值时,它也会影响冲突的引脚。 如果发生这种情况,请查看图 8-4 中的引脚排列图和您的电路,并找到您没有的 PWM 输出还没用。
如果这听起来令人困惑,请不要担心:这意味着您需要确保跟踪正在使用的 PWM 切片和输出,确保仅连接到带有字母和数字组合的引脚” t 已经使用。 如果您在 GP0 针脚上使用 PWMA[0] 并在针脚 GP1 上使用 PWMB[0],则一切正常,如果您在针脚 GP2 上添加 PWMA[1],则将继续工作; 但是,如果您尝试在引脚 GP0 和引脚 GP16 上使用 PWM 通道,则会遇到问题,因为它们都连接到 PWMA[0]。 取一个 LED 和一个 330 Ω 的限流电阻,将它们放入面包板中,如图 8-5 所示。 将 LED 的较长脚(阳极)通过 330 Ω 电阻连接到 GP15 引脚,并将较短的脚连接到 Pico 的接地引脚。 单击 Thonny 工具栏下方的选项卡,返回您的第一个程序; 如果您已经关闭它,请单击“打开”图标并从您的 Pico 加载 Potentiometer.py。 在您将电位计设置为模数输入的位置下方,键入: <code python> led = machine.PWM(machine.Pin(15))
这会在引脚 GP15 上创建一个 LED 对象,但有一个区别:它激活引脚上的脉宽调制输出,通道 B[7]——第八个切片的第二个输出,从零开始计数。 您还需要设置频率,这是您可以更改以控制或调制脉冲宽度的两个值之一。 在读数下方立即添加另一行: <code python> led.freq(1000)
这将设置频率为 1000 赫兹 - 每秒一千个周期。 接下来,转到程序的底部并在添加以下内容之前删除 print(voltage) 和 utime.sleep(2) 行,记住将其缩进四个空格,使其成为循环中嵌套代码的一部分: <code python> led.dutyu16(potentiometer.readu16())
这条线从连接到电位计的模拟输入中获取原始读数,然后将其用作脉宽调制的第二个方面:占空比。占空比控制引脚的输出:0% 的占空比使引脚在每秒 1000 个脉冲时关闭,并有效地关闭引脚; 100% 的占空比使引脚在每秒 1000 个脉冲时都处于开启状态,并且在功能上等同于仅将引脚作为固定数字输出开启;占空比为 50% 时,引脚打开一半脉冲,关闭一半脉冲。 为了能够正确控制 LED 的亮度,您需要将模拟输入的值映射到 PWM 片可以理解的范围。执行此操作的最佳方法是告诉 MicroPython 您将占空比值作为无符号 16 位整数传递,与您从 Pico 的模拟输入引脚接收到的数字格式相同。这是通过使用 led.dutyu16 实现的。 您完成的程序将如下所示: <code python> import machine import utime potentiometer = machine.ADC(26) led = machine.PWM(machine.Pin(15)) led.freq(1000) while True: led.dutyu16(potentiometer.read_u16())
单击“运行”图标并尝试将电位计一直转动到一个方向,然后再转动到另一个方向。 观察 LED:这一次,除非您使用对数电位器,否则您会看到 LED 的亮度从电位器旋钮限制的一端完全关闭到另一端完全点亮。 恭喜:您不仅掌握了模拟输入,而且现在可以使用脉宽调制创建等效于模拟输出的内容!
挑战:定制
您能否结合您的两个程序,并通过车载温度传感器的温度读数来控制 LED 的亮度? 您还记得您的 Pico 有多少个模拟输入吗? PWM输出呢? 尝试在 Pico 中添加另一个模拟传感器——比如光敏电阻器 (LDR)、气体传感器或气压计——并让你的程序读取它而不是电位计。