
esp32集成了两个12位逐次逼近寄存器(sar)adc,分别为adc1和adc2,共支持18个模拟测量通道。然而,adc2的使用存在一个重要的限制:它与wi-fi驱动程序共享硬件资源。这意味着当wi-fi功能被激活并运行时,应用程序将无法正常使用adc2进行模拟值读取。试图在wi-fi活跃时访问adc2引脚,会导致oserror: [errno 116] etimedout错误,表明adc读取操作超时。
ESP32 ADC通道分配:
因此,如果您的模拟传感器连接到了ADC2的任何一个引脚(例如GPIO 4),并且您的MicroPython程序同时需要连接Wi-Fi,就会出现上述冲突。
以下代码片段展示了当使用ADC2引脚(GPIO 4)并在Wi-Fi连接后尝试读取模拟值时,会遇到的典型问题:
from machine import ADC, Pin
import network
import time
# 使用ADC2引脚,例如GPIO 4
adc_pin_num = 4
adc = ADC(Pin(adc_pin_num))
# Wi-Fi凭据(请替换为您的实际凭据)
WIFI_SSID = "YOUR_SSID"
WIFI_PASSWORD = "YOUR_PASSWORD"
def read_water_sensor():
"""尝试读取模拟传感器值"""
try:
value = adc.read() # 此处可能引发OSError
return value
except OSError as e:
print(f"读取ADC失败: {e}")
return -1 # 返回一个错误指示值
def connect_wifi():
"""连接到Wi-Fi网络"""
sta_if = network.WLAN(network.STA_IF)
if not sta_if.isconnected():
print("Connecting to Wi-Fi...")
sta_if.active(True)
sta_if.connect(WIFI_SSID, WIFI_PASSWORD)
while not sta_if.isconnected():
time.sleep(0.5)
print("Connected to Wi-Fi")
# 连接Wi-Fi
connect_wifi()
# 循环读取传感器值
while True:
water_value = read_water_sensor()
if water_value != -1:
print("Water sensor value:", water_value)
time.sleep(1)运行上述代码,当Wi-Fi成功连接后,read_water_sensor()函数中的adc.read()调用将抛出OSError: [Errno 116] ETIMEDOUT错误。
立即学习“Python免费学习笔记(深入)”;
针对ESP32 ADC2与Wi-Fi的冲突,有以下几种解决方案:
这是最直接且推荐的解决方案。如果您的应用需要同时使用Wi-Fi和ADC功能,应优先将模拟传感器连接到ADC1的可用引脚上。ADC1的引脚(GPIO 32-39)与Wi-Fi驱动没有冲突。
示例代码(使用ADC1引脚,例如GPIO 34):
from machine import ADC, Pin
import network
import time
# 将传感器连接到ADC1引脚,例如GPIO 34
adc_pin_num = 34
adc = ADC(Pin(adc_pin_num))
# Wi-Fi凭据
WIFI_SSID = "YOUR_SSID"
WIFI_PASSWORD = "YOUR_PASSWORD"
def read_water_sensor():
"""读取模拟传感器值"""
value = adc.read()
return value
def connect_wifi():
"""连接到Wi-Fi网络"""
sta_if = network.WLAN(network.STA_IF)
if not sta_if.isconnected():
print("Connecting to Wi-Fi...")
sta_if.active(True)
sta_if.connect(WIFI_SSID, WIFI_PASSWORD)
while not sta_if.isconnected():
time.sleep(0.5)
print("Connected to Wi-Fi")
# 连接Wi-Fi
connect_wifi()
# 循环读取传感器值
while True:
water_value = read_water_sensor()
print("Water sensor value:", water_value)
time.sleep(1)通过将adc_pin_num更改为ADC1的可用引脚(如34),程序将能够同时进行Wi-Fi通信和模拟值读取。
如果由于硬件限制,您必须使用ADC2引脚,那么一种可能的(但不推荐用于连续操作)方法是在需要读取ADC2值时暂时禁用Wi-Fi,读取完毕后再重新启用Wi-Fi。这种方法会引入网络连接的延迟和中断。
示例代码(概念性):
from machine import ADC, Pin
import network
import time
adc_pin_num = 4 # 假设必须使用ADC2引脚
adc = ADC(Pin(adc_pin_num))
sta_if = network.WLAN(network.STA_IF)
WIFI_SSID = "YOUR_SSID"
WIFI_PASSWORD = "YOUR_PASSWORD"
def connect_wifi_if_needed():
"""检查并连接Wi-Fi"""
if not sta_if.isconnected():
print("Connecting to Wi-Fi...")
sta_if.active(True)
sta_if.connect(WIFI_SSID, WIFI_PASSWORD)
while not sta_if.isconnected():
time.sleep(0.5)
print("Connected to Wi-Fi")
def disconnect_wifi():
"""断开Wi-Fi连接"""
if sta_if.isconnected():
print("Disconnecting Wi-Fi...")
sta_if.disconnect()
sta_if.active(False)
print("Wi-Fi disconnected")
def read_adc2_safely():
"""安全读取ADC2值"""
if sta_if.isconnected(): # 如果Wi-Fi连接中,先断开
disconnect_wifi()
time.sleep(0.1) # 留一点时间让硬件资源释放
value = adc.read()
print(f"Read ADC2 value: {value}")
connect_wifi_if_needed() # 读取后重新连接Wi-Fi
return value
# 初始连接Wi-Fi
connect_wifi_if_needed()
while True:
# 模拟需要读取ADC2值并上传到MQTT的场景
water_value = read_adc2_safely()
# ... 在此处执行MQTT上传操作 ...
time.sleep(5) # 每隔5秒读取一次并重新连接Wi-Fi这种方法会频繁地断开和重新连接Wi-Fi,导致网络延迟和不稳定,通常只适用于对实时性要求不高且可以容忍网络中断的特定应用场景。
如果ADC1的引脚数量不足,或者您的应用对模拟精度、采样率有更高要求,并且无法接受Wi-Fi与ADC2的软件切换方案,那么使用外部ADC模块是一个可靠的硬件解决方案。通过I2C或SPI接口连接外部ADC芯片(如ADS1115、MCP3008等),可以完全规避ESP32内部ADC的限制。
优点:
实现方式: 通过MicroPython的machine.I2C或machine.SPI模块与外部ADC芯片通信。
ESP32的ADC2与Wi-Fi驱动共享硬件资源是一个固有的设计特性,导致在Wi-Fi活跃时无法使用ADC2引脚。解决此问题的核心在于理解这一限制并采取相应的策略。最简单有效的方案是优先利用ADC1的引脚进行模拟量采集。若ADC1引脚不足或有特殊需求,可以考虑在严格控制下交替使用Wi-Fi和ADC2,或者采用外部ADC模块来彻底规避内部冲突,从而确保您的ESP32项目能够稳定地进行数据采集和网络通信。
以上就是ESP32 MicroPython ADC2与Wi-Fi共存问题及解决方案的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号