首页 热点资讯 义务教育 高等教育 出国留学 考研考公
您的当前位置:首页正文

树莓派三极管+温度传感器实现温差控制风扇+代码

2024-12-20 来源:华佗小知识

入手了树莓派之后,买了个树莓派的亚克力外壳,送了一个风扇。
但是送的这个风扇,只有正负极,不能自主控制。
风扇如果一直转,噪音让人有点受不了,如果没有风扇,树莓派散热效果又要受到影响。
然后在实验室找了个三极管,准备用树莓派的GPIO口来根据CPU的温度来智能控制风扇是否转动。

安装三极管控制风扇教程

温控风扇的不足之处

我设置的温度是温度高于48度就转动,低于42度就停止转动。
在冬天用起来是非常舒服的,确实是树莓派高负荷的时候才会转动,但是到了第二年5月左右,就发现风扇转动的频率变高了,考虑到了是室温的变化导致了这种情况。
到6月左右,风扇几乎是大半天都在转动了,于是手动去调节了代码的设置,也在想解决方案:怎么才能让树莓派在冬天和夏天都只在高负载的时候运转呢?

想到了网上有很多树莓派教程,获取室温,如果可以获取到室温,设置的温度根据室温浮动在一个区间,这样就可以保持风扇转动的灵活性了,树莓派也多了个检测室温的功能了。然后找到朋友弄了个DS18B20的温度传感器模块。

安装温度传感器教程

安装遇到的问题

最后焊接了之后,接到树莓派上,但是树莓派一直都检测不到温度。单独检测又是没问题的。晚上回到寝室,又弄了半天,发现这个模块只有在3.3V的供电下才能使用,在5V的接口下没办法使用(不知道是不是我操作有问题)。

树莓派引脚图片

树莓派引脚图片,来源见水印

温控风扇代码介绍

网上找的代码,要么不全,要么不容易懂,所以我结合网上的代码,自己也写了个万能代码,主要特点:

  • 兼容没有温度传感器的情况
  • 自动寻找温度传感器的位置,不需要手动设置
  • 生成日志文件,方便查看记录与可视化
  • 一次设置,以后基本上都不用管了

代码:

# -*- coding:utf-8 -*-
# date: 2018-10-08 23:54
"""
控制树莓派风扇
"""
import os
import time
import commands
import RPi.GPIO as GPIO
from datetime import datetime

T_HIGH = 48  # 温度>48度开始转动,在温度传感器失效时使用
T_LOW = 42  # 温度<42度停止转动,在温度传感器失效时使用
T_DIFF_HIGH = 24  # 温差>24度开始转动
T_DIFF_LOW = 18  # 温差<18度停止转动
T_SENOR_DIFF = 0  # 温度传感器和真实环境温度矫正值
fan_pin = 12  # 风扇IO针脚BOARD编号
NPN = True  # 控制风扇用的是NPN三极管
IS_LOG_FILE = True  # 是否输出温度信息到文件
IS_LOG_CONSOLE = True  # 是否输出温度信息到控制台
time_interval = 5  # 检测温度间隔时间单位秒
log_file_duration = 12  # 日志记录保留时间长度,单位小时

base_dir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
log_file_path = os.path.join(base_dir, 'temperature_log')
GPIO.setwarnings(False)
GPIO.setmode(GPIO.BOARD)
GPIO.setup(fan_pin, GPIO.OUT)
temp_path = None

if NPN:
    START = 1
    STOP = 0
else:
    START = 0
    STOP = 1

# 自动获取温度传感器温度信息所在的文件位置
try:
    for each_dir in os.listdir('/sys/bus/w1/devices/'):
        if '28-' in each_dir:
            temp_path = '/sys/bus/w1/devices/'+each_dir+'/w1_slave'
except FileNotFoundError:
    print('Warning!: 没有检测到温度传感器!')


def get_gpu_temp():
    gpu_temp = commands.getoutput('/opt/vc/bin/vcgencmd measure_temp').replace('temp=', '').replace('\'C', '')
    return float(gpu_temp)


def get_env_temp():
    with open(temp_path, 'r') as temp_file:
        # 读取文件所有内容
        text = temp_file.read()
        # a7 01 4b 46 7f ff 09 10 e0 : crc=e0 YES
        # a7 01 4b 46 7f ff 09 10 e0 t=26437

        temp_line = text.split("\n")[1]
        # a7 01 4b 46 7f ff 09 10 e0 t=26437

        temperature_data = temp_line.split(" ")[9]
        # 't=26437'
        env_temperature = float(temperature_data[2:])
        # 26437
        env_temperature = env_temperature / 1000
        # 26.437
        # print(temperature)
        return env_temperature


def main():
    count = 0
    env_temp = None
    temp_diff = None
    count_per_min = 60/time_interval
    while True:
        gpu_temp_loop = get_gpu_temp()
        if temp_path is not None:
            env_temp = get_env_temp()
            temp_diff = gpu_temp_loop - env_temp

        if IS_LOG_FILE:
            try:
                if count <= count_per_min * 60 * log_file_duration:
                    fp = open(log_file_path, 'a')
                    if temp_diff is None:
                        fp.write('{} CPU temp: {}C\n'.format(datetime.now().strftime("%Y-%m-%d %H:%M:%S"), gpu_temp_loop))
                    else:
                        fp.write('{} CPUtemp: {}C, ENVtemp: {}C, DIFF: {}C\n'.format(datetime.now().strftime("%Y-%m-%d %H:%M:%S"),
                                                                                     gpu_temp_loop, env_temp, temp_diff))
                    count += 1
                    fp.close()
                else:
                    fp = open(log_file_path, 'w')
                    fp.write('{} CPU temp: {}C\n'.format(datetime.now().strftime("%Y-%m-%d %H:%M:%S"), gpu_temp_loop))
                    count = 0
                    fp.close()
            except Exception as e:
                print('Error: {}-{}'.format(datetime.now().strftime("%Y-%m-%d %H:%M:%S"), e))

        if IS_LOG_CONSOLE:
            if temp_diff is None:
                print('{} CPU temp: {}C\n'.format(datetime.now().strftime("%Y-%m-%d %H:%M:%S"), gpu_temp_loop))
            else:
                print('{} CPUtemp: {}C, ENVtemp: {}C, DIFF: {}C\n'.format(datetime.now().strftime("%Y-%m-%d %H:%M:%S"),
                                                                          gpu_temp_loop, env_temp, temp_diff))
        if temp_diff is None:
            if gpu_temp_loop > T_HIGH:
                GPIO.output(fan_pin, START)
            elif gpu_temp_loop < T_LOW:
                GPIO.output(fan_pin, STOP)
        else:
            if temp_diff > T_DIFF_HIGH:
                GPIO.output(fan_pin, START)
            elif temp_diff < T_DIFF_LOW:
                GPIO.output(fan_pin, STOP)
        time.sleep(time_interval)


if __name__ == '__main__':
    main()

显示全文