ESP32实现简单频率计(1hz-120khz左右)

本文介绍了如何使用Arduino库(如LiquidCrystal_I2C)测量PWM信号的频率,并利用中断处理实时更新频率值。代码展示了如何在setup()和loop()函数中设置中断以及在changeISR()中计算频率。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

/*-------结合他人代码改写------*/

#include <Arduino.h>
#include <LiquidCrystal_I2C.h>

volatile unsigned long raiseTime = 0; //前一次上升沿时间
volatile unsigned long fallTime = 0;  //前一次下降沿时间
//volatile double duty = 0;  //占空比
volatile double fre = 0;   //频率

int pwmPin = 27;   //信号输入管脚为27

LiquidCrystal_I2C lcd(0x27, 16, 2);

portMUX_TYPE mux = portMUX_INITIALIZER_UNLOCKED;  //自旋锁

void changeISR()
{
  auto now = micros();//(以微秒为单位记录当前时间)
  if(digitalRead(pwmPin))  //现在是高电平
  {
    portENTER_CRITICAL_ISR(&mux);   //进入临界区域
    auto total = now - raiseTime;
    fre = 1e6/(double)total;
    //auto h = fallTime - raiseTime;
    //duty = h/(double)total;
    portEXIT_CRITICAL_ISR(&mux);   //离开临界区域
    raiseTime = now;
  }
  else
  {
    fallTime = now;
  }
}

void setup() {
  // lcd.init();
  // lcd.backlight();
  // lcd.setCursor(0,0);
  // lcd.print("fre: ");
  // lcd.setCursor(0,1);
  // lcd.print("duty: ");

  Serial.begin(115200);
  pinMode(pwmPin, INPUT);
  attachInterrupt(digitalPinToInterrupt(pwmPin), changeISR, CHANGE);
}

void loop() {
  delay(1000);

  portENTER_CRITICAL(&mux);
  double f = fre;
  //double d = duty;
  portEXIT_CRITICAL(&mux);
  
   Serial.print("信号频率为:");
   Serial.print(f);
   Serial.println("hz");

  // lcd.setCursor(5,0);
  // lcd.print(f);
  // lcd.setCursor(6,1);
  // lcd.print(d);
}

### ESP32 ADC 采样率配置 对于ESP32而言,ADC模块支持不同的采样频率设置。然而,具体的采样速率取决于所使用的硬件特性以及软件中的配置方式。 #### 硬件限制 每个芯片的可测量输入电压范围有所不同,这间接影响了ADC的最大采样速度,因为更高的分辨率通常意味着更慢的速度[^1]。这意味着,在设定较高的采样率之前,应当确认所需的精度是否能够满足应用需求。 #### 软件配置 当使用Arduino框架开发时,可以通过调整`adc_set_width()` 和 `adc_set_attenuation()` 函数来改变转换位宽和衰减等级,从而影响最终的有效采样率。但是这些并不是直接控制采样率的方法;它们更多地是为了适应不同量程下的精确度要求。 为了实现更高频次的数据获取,可以考虑减少每次读数之间的延时时间或者优化代码逻辑以提高效率。例如,在给定的例子中有一个5毫秒的任务延迟(`vTaskDelay(5)`), 这显然会降低整体吞吐量[^3]: ```c++ uint32_t adc_get_result_average(uint32_t ch, uint32_t times) { uint32_t temp_val = 0; for (uint8_t t = 0; t < times; ++t){ temp_val += adc1_get_raw(ch); vTaskDelay(5); // 可能需要根据实际应用场景调整或移除此行 } return temp_val / times; } ``` 如果希望进一步提升性能,则可能需要深入研究官方文档提供的API接口,并探索其他高级选项如DMA传输等机制来加速数据处理过程。 另外值得注意的是,默认情况下FreeRTOS调度器也会对程序执行造成一定影响,因此合理规划任务优先级同样重要。 #### MicroPython环境下的注意事项 在MicroPython环境下工作的话,安装好固件之后就可以利用其内置库来进行简单的初始化操作[^2]: ```python from machine import Pin, ADC import time adc = ADC(Pin(34)) # 创建一个连接到指定引脚上的ADC对象 adc.atten(ADC.ATTN_11DB) # 设置最大输入电平为大约3.9V start_time = time.ticks_ms() for i in range(100): # 测试循环次数可以根据实际情况修改 value = adc.read() # 执行一次AD转换并返回结果 end_time = time.ticks_ms() print("Time taken:", end_time - start_time, "ms") # 输出花费的时间 ``` 通过上述测试可以帮助理解当前系统的极限在哪里,并据此做出相应的参数调整决策。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值