✅作者简介:热爱科研的嵌入式开发者,修心和技术同步精进

❤欢迎关注我的知乎:对error视而不见

代码获取、问题探讨及文章转载可私信。

☁ 愿你的生命中有够多的云翳,来造就一个美丽的黄昏。

🍎获取更多嵌入式资料可点击链接进群领取,谢谢支持!👇

点击领取更多详细资料

一、引言

光敏电阻传感器是一种能够将光信号转换为电信号的传感器,其阻值会随着光照强度的变化而改变。在许多实际应用中,如自动照明系统、环境光检测等,光敏电阻传感器发挥着重要作用。STM32作为一款高性能的微控制器,具有丰富的外设资源,能够方便地与光敏电阻传感器结合使用,实现对光照强度的检测和处理。本文将详细介绍光敏电阻传感器的工作原理、硬件连接以及基于STM32的软件实现。

二、光敏电阻传感器工作原理

光敏电阻通常由半导体材料制成,其工作原理基于内光电效应。当光线照射到光敏电阻上时,半导体材料中的电子会吸收光子的能量,从价带跃迁到导带,从而使材料的导电能力增强,电阻值降低。反之,当光照强度减弱时,电阻值会增大。

光敏电阻的特性曲线通常呈现出非线性关系,即光照强度与电阻值之间不是简单的线性比例关系。在实际应用中,需要根据具体的需求和特性曲线来进行校准和处理。

三、硬件连接

3.1 电路原理

为了将光敏电阻的阻值变化转换为电压信号,通常会采用分压电路。将光敏电阻与一个固定电阻串联,然后连接到电源和地之间。通过测量光敏电阻两端的电压,就可以间接得到光照强度的信息。

3.2 STM32硬件连接

以STM32F103C8T6为例,将光敏电阻与固定电阻组成的分压电路的输出端连接到STM32的一个模拟输入引脚(如PA0)。同时,确保电源和地的连接稳定。

以下是一个简单的硬件连接示意图:

+3.3V --- [光敏电阻] --- [固定电阻] --- GND
                      |
                      +--- PA0 (STM32模拟输入引脚)

四、软件实现

4.1 初始化ADC(模拟数字转换器)

在STM32中,使用ADC来将模拟电压信号转换为数字值。以下是使用STM32 HAL库初始化ADC的代码示例:

#include "stm32f1xx_hal.h"

ADC_HandleTypeDef hadc1;

void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_ADC1_Init(void);

int main(void)
{
  HAL_Init();
  SystemClock_Config();
  MX_GPIO_Init();
  MX_ADC1_Init();

  while (1)
  {
    // 主循环
  }
}

void SystemClock_Config(void)
{
  RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};

  /** 初始化RCC振荡器 
  */
  RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI;
  RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT;
  RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;
  if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  {
    Error_Handler();
  }
  /** 初始化RCC时钟 
  */
  RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
                              |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_HSI;
  RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1;
  RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;

  if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_0) != HAL_OK)
  {
    Error_Handler();
  }
}

static void MX_ADC1_Init(void)
{
  ADC_ChannelConfTypeDef sConfig = {0};

  /** 初始化ADC 
  */
  hadc1.Instance = ADC1;
  hadc1.Init.ScanConvMode = DISABLE;
  hadc1.Init.ContinuousConvMode = DISABLE;
  hadc1.Init.DiscontinuousConvMode = DISABLE;
  hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
  hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
  hadc1.Init.NbrOfConversion = 1;
  if (HAL_ADC_Init(&hadc1) != HAL_OK)
  {
    Error_Handler();
  }
  /** 配置ADC通道 
  */
  sConfig.Channel = ADC_CHANNEL_0;
  sConfig.Rank = ADC_REGULAR_RANK_1;
  sConfig.SamplingTime = ADC_SAMPLETIME_1CYCLE_5;
  if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
  {
    Error_Handler();
  }
}

static void MX_GPIO_Init(void)
{
  GPIO_InitTypeDef GPIO_InitStruct = {0};

  __HAL_RCC_GPIOA_CLK_ENABLE();
  // 无需特殊GPIO配置,因为使用模拟输入
}

void Error_Handler(void)
{
  while(1)
  {
  }
}

4.2 读取ADC值

在主循环中,通过调用HAL_ADC_StartHAL_ADC_PollForConversion函数来启动ADC转换并等待转换完成,然后使用HAL_ADC_GetValue函数读取转换结果。

int main(void)
{
  HAL_Init();
  SystemClock_Config();
  MX_GPIO_Init();
  MX_ADC1_Init();

  uint32_t adc_value;

  while (1)
  {
    // 启动ADC转换
    HAL_ADC_Start(&hadc1);

    // 等待转换完成
    if (HAL_ADC_PollForConversion(&hadc1, 100) == HAL_OK)
    {
      // 读取ADC值
      adc_value = HAL_ADC_GetValue(&hadc1);

      // 可以在这里对ADC值进行处理,如显示或进一步计算
    }

    HAL_Delay(100);  // 延时100ms
  }
}

4.3 光照强度计算

根据分压电路的原理,可以通过ADC值计算出光敏电阻两端的电压,进而根据已知的固定电阻值和电源电压计算出光敏电阻的阻值。最后,根据光敏电阻的特性曲线,将阻值转换为光照强度。

// 假设电源电压为3.3V,固定电阻为10K
#define VREF 3.3
#define FIXED_RESISTANCE 10000

float calculate_light_intensity(uint32_t adc_value)
{
  // 计算光敏电阻两端的电压
  float voltage = (float)adc_value * VREF / 4095;

  // 计算光敏电阻的阻值
  float resistance = FIXED_RESISTANCE * voltage / (VREF - voltage);

  // 这里需要根据光敏电阻的特性曲线进行光照强度计算
  // 简单示例:假设光照强度与阻值成反比
  float light_intensity = 1000 / resistance;

  return light_intensity;
}

int main(void)
{
  HAL_Init();
  SystemClock_Config();
  MX_GPIO_Init();
  MX_ADC1_Init();

  uint32_t adc_value;
  float light_intensity;

  while (1)
  {
    // 启动ADC转换
    HAL_ADC_Start(&hadc1);

    // 等待转换完成
    if (HAL_ADC_PollForConversion(&hadc1, 100) == HAL_OK)
    {
      // 读取ADC值
      adc_value = HAL_ADC_GetValue(&hadc1);

      // 计算光照强度
      light_intensity = calculate_light_intensity(adc_value);

      // 可以在这里对光照强度进行处理,如显示或进一步控制
    }

    HAL_Delay(100);  // 延时100ms
  }
}

五、总结

本文详细介绍了光敏电阻传感器的工作原理、硬件连接以及基于STM32的软件实现。通过使用STM32的ADC功能,可以方便地将光敏电阻的阻值变化转换为数字值,并进一步计算出光照强度。在实际应用中,需要根据具体的光敏电阻特性曲线进行校准和优化,以提高光照强度检测的准确性。同时,还可以根据光照强度的检测结果实现各种自动控制功能,如自动调光、光控开关等。

Logo

智能硬件社区聚焦AI智能硬件技术生态,汇聚嵌入式AI、物联网硬件开发者,打造交流分享平台,同步全国赛事资讯、开展 OPC 核心人才招募,助力技术落地与开发者成长。

更多推荐