基于51单片机的智能热水器无线WiFi控制系统proteus仿真原理图程序

news2025/1/19 19:37:32

功能:
0.本项目采用STC89C52作为单片机系统的控制MCU
1.LCD1602液晶实时显示温度阈值、当前温度和定时时间
2.支持按键和红外遥控设置温度阈值和定时时间
3.通过传感器检测,判定当前值是否超过设定的阈值,然后对相关继电器进行控制
4.支持声光报警
5.支持WIFI上报温度、温度上下限和继电器状态,支持WIFI设置温度阈值和控制继电器
命令:
*HTO#:打开加热控制继电器
*HTC#:关闭加热控制继电器
*WTO#:打开加水控制继电器
*WTC#:关闭加水控制继电器
*TH+#:温度上限值+1
*TH-#:温度上限值-1
*TL+#:温度下限值+1
*TL-#:温度下限值-1
6.采用DC002作为电源接口可直接输入5V给整个系统供电

原理图:
在这里插入图片描述
在这里插入图片描述

PCB :
在这里插入图片描述

主程序:

//程序头函数
#include <reg52.h>
#include "infrared.h"
#include "delay.h"
#include "18b20.h"
#include "esp8266.h"
#include "lcd1602.h"
//宏定义
#define OFF 1
#define ON 0

//管脚声明
sbit RELAY_HEATING = P2^1; //加热控制继电器
sbit RELAY_WATER_ADDING = P2^0; //进水控制继电器
sbit BUZZER = P2^2;  //蜂鸣器
sbit KEY_SET = P3^3;  //设置键
sbit KEY_ADD = P3^4;  //加键
sbit KEY_SUB = P3^6;  //减键
sbit KEY_ENTER = P3^5;  //确认键
sbit WATER_UPPER_LIMIT_FLAG = P1^1; //水位上限传感器
sbit WATER_LOWER_LIMIT_FLAG = P1^2;  //水位下限传感器
sbit LED_WATER_LOWER = P1^3; //水位下限指示灯

//全局变量
xdata unsigned char dis0[16]; //显示缓存
int tempDataBuf = 0; //温度缓存值
xdata float tempData = 0; //温度值                
bit switchFlag = 0; //开关标志
bit alarmFlag; //报警标志
bit buzzerFlag; //蜂鸣器标志
bit sendFlag = 1; //发送标志
bit lackOfWaterFlag; //缺水标志
bit waterFlag1 = 0; //自动检测控制加水标志
bit waterFlag2 = 0; //WIFI控制加水标志
unsigned char tempLowerLimit = 29; //温度下限
unsigned char tempUpperLimit = 35; //温度上限
unsigned int setTime = 0; //倒计时
unsigned char setLocation = 0; //设定位置
bit irFlag = 0; //红外接收标志,收到一帧正确数据后置1
unsigned char irCode[4];
unsigned int timeCnt = 0; //计数

//函数声明
void CheckTemperature(void); //读取温度
void DispNormal(void); //正常显示
void DispSet(void); //设置显示
void SendData(void); //发送数据
void KeyProcess(void); //按键处理
void Alarm(void); //提醒
void InitTimer0(void); //初始化定时器0

void main()
{
    IR_INPUT = 1;

    LCD_Init(); //LCD初始化
    Start18B20(); //启动18B20
    CheckTemperature(); //检测温度
    LCD_DispStr(4, 0, "Welcome!"); //显示

    InitTimer0();  //初始化定时器0
    IR_Init(); //初始化红外配置
    DelayMs(250);
    DelayMs(250);
    ESP8266_Init(); //初始化WIFI模块

    Start18B20(); //启动18B20
    CheckTemperature(); //检测温度
    DispNormal();//显示

    while (1)
    {
        Start18B20(); //启动18B20
        CheckTemperature(); //检测温度
        if (sendFlag == 1) //WIFI上报数据
        {
            sendFlag = 0;
            SendData();
        }
        if (setLocation == 0) //非设置状态
        {
            DispNormal();//正常显示   
        }
        
        if ((WATER_LOWER_LIMIT_FLAG == 0) && (WATER_UPPER_LIMIT_FLAG == 0)) //低于下限
        {
            lackOfWaterFlag = 1; //缺水标志
            waterFlag1 = 1; //加水
            RELAY_HEATING = OFF; //关闭加热继电器
            LED_WATER_LOWER = ON; //缺水指示灯
            BUZZER = OFF; //蜂鸣器关闭
        }
        else if ((WATER_UPPER_LIMIT_FLAG == 1) && (WATER_LOWER_LIMIT_FLAG == 1)) //高于上限
        {
            lackOfWaterFlag = 0;
            waterFlag1 = 0;
            waterFlag2 = 0; //WIFI控制加水标志关
            LED_WATER_LOWER = OFF; //关闭指示灯
            if (alarmFlag == 0)
                alarmFlag = 1;
            BUZZER = OFF;
        }
        else if ((WATER_UPPER_LIMIT_FLAG == 1) && (WATER_LOWER_LIMIT_FLAG == 0)) //错误
        {
            waterFlag1 = 0;
            waterFlag2 = 0; //WIFI控制加水标志关
            RELAY_HEATING = OFF; //关闭加热继电器
            LED_WATER_LOWER = OFF; //关闭指示灯
            BUZZER = ON; //蜂鸣器响
            lackOfWaterFlag = 1; //缺水标志
        }
        else
        {
            LED_WATER_LOWER = OFF;//关闭指示灯
            lackOfWaterFlag = 0;
            BUZZER = OFF;
        }

        if (waterFlag1 || waterFlag2)
        {
            RELAY_WATER_ADDING = ON; //打开加水继电器
        }
        else
        {
            RELAY_WATER_ADDING = OFF; //关闭加水继电器
        } 
        
        if (switchFlag == 1) //加热开关打开
        {
            if (tempData >= tempUpperLimit) //高于温度上限
            {
                RELAY_HEATING = OFF; //关闭加热继电器
                switchFlag = 0;
                if (alarmFlag == 0)
                    alarmFlag = 1;
            }
            else if ((tempData < tempLowerLimit) && (lackOfWaterFlag == 0)) //低于温度下限且有水
            {
                RELAY_HEATING = ON; //打开加热继电器
                if (alarmFlag == 0)
                    alarmFlag = 1;
            }
            else
            {
                alarmFlag = 0;
                buzzerFlag = 0;
            }
        }
        else
        {
            RELAY_HEATING = OFF; //关闭加热继电器
            alarmFlag = 0;
            buzzerFlag = 0;
        }
        
        Alarm(); //提醒处理
        KeyProcess(); //按键处理
    }
}

/************************* 温度检测 *************************/
void CheckTemperature(void)
{
    bit ack;

    ack = Get18B20Temp(&tempDataBuf);
    DelayMs(20);
    if (ack)
    {
        tempData = (float)tempDataBuf * 0.0625; //实际温度转换
    }

    if (tempData < 0)
    {
        tempData = 0;
    }
    if (tempData > 99)
    {
        tempData = 99;
    }
}

/************************* 正常显示函数 *************************/
void DispNormal(void)
{
    sprintf(dis0, "Tp:%4.1f", tempData); //显示温度
    LCD_DispStr(0, 0, dis0);
    LCD_DispOneChar(7, 0, 0xdf);
    LCD_DispOneChar(8, 0, 'C');

    sprintf(dis0, " Ti:%03d", (int)setTime); //显示定时
    LCD_DispStr(9, 0, dis0);

    sprintf(dis0, "TH:%02d TL:%02d ", (int)tempUpperLimit, (int)tempLowerLimit); //显示温度上下限
    LCD_DispStr(0, 1, dis0);

    //显示水位
    LCD_DispOneChar(12, 1, 'H');
    if (WATER_UPPER_LIMIT_FLAG == 0)
    {
        LCD_DispOneChar(13, 1, ' ');
    }
    else
    {
        LCD_DispOneChar(13, 1, '*');
    }

    LCD_DispOneChar(14, 1, 'L');
    if (WATER_LOWER_LIMIT_FLAG == 0)
    {
        LCD_DispOneChar(15, 1, ' ');
    }
    else
    {
        LCD_DispOneChar(15, 1, '*');
    }
}

/************************* 设置显示函数 *************************/
void DispSet(void)
{
    LCD_DispStr(0, 0, "Set state");

    sprintf(dis0, " Ti:%03d", (int)setTime);
    LCD_DispStr(9, 0, dis0);

    sprintf(dis0, "TH:%02d TL:%02d ", (int)tempUpperLimit, (int)tempLowerLimit);
    LCD_DispStr(0, 1, dis0);

    LCD_DispOneChar(12, 1, 'H');
    if (WATER_UPPER_LIMIT_FLAG == 0)
    {
        LCD_DispOneChar(13, 1, ' ');
    }
    else
    {
        LCD_DispOneChar(13, 1, '*');
    }

    LCD_DispOneChar(14, 1, 'L');
    if (WATER_LOWER_LIMIT_FLAG == 0)
    {
        LCD_DispOneChar(15, 1, ' ');
    }
    else
    {
        LCD_DispOneChar(15, 1, '*');
    }

    switch (setLocation)
    {
        case 1: LCD_SetCursor(15, 0, 1); break;
        case 2: LCD_SetCursor(4, 1, 1); break;
        case 3: LCD_SetCursor(10, 1, 1); break;
        default:;
    }
}

/************************* WIFI上报信息 *************************/
void SendData(void)
{
    unsigned char dat[16];

    UART_SendStr("AT+CIPSEND=0,38\r\n", 17); //发送AT命令
    DelayMs(50);
    sprintf(dat, "Tp:%4.1f'C ", tempData);
    UART_SendStr(dat, 10); //发送内容
    DelayMs(50);
    sprintf(dat, "TH:%02d TL:%02d\r\n", (int)tempUpperLimit, (int)tempLowerLimit);
    UART_SendStr(dat, 13); //发送内容

    //发送继电器控制状态
    if (RELAY_WATER_ADDING == OFF)
    {
        UART_SendStr("WATER:C ", 8);
    }
    else
    {
        UART_SendStr("WATER:O ", 8);
    }
    
    if (RELAY_HEATING == OFF)
    {
        UART_SendStr("HEAT:C\r\n", 8);
    }
    else
    {
        UART_SendStr("HEAT:O\r\n", 8);
    }
}

/************************* 按键检测 *************************/
void KeyProcess(void)
{
    
    if (KEY_SET == 0) //设置键按下
    {
        DelayMs(20); //延时去抖
        if (KEY_SET == 0) //再次确认设置键按下
        {
            setLocation++; //设定位置切换
            if (setLocation == 4)
            {
                setLocation = 1;
            }
            DispSet(); //设置界面显示
        }
        while (KEY_SET == 0);
    }

    if (KEY_ADD == 0 && setLocation != 0) //加键按下
    {
        DelayMs(180);
        if (KEY_ADD == 0 && setLocation != 0)
        {
            switch (setLocation)
            {
                case 1:
                {
                    if (setTime < 999)
                    {
                        setTime++;
                    }
                    break;
                }
                case 2:
                {
                    if (tempUpperLimit < 99 - 1)
                    {
                        tempUpperLimit++;
                    }
                    break;
                }
                case 3:
                {
                    if (tempLowerLimit < tempUpperLimit - 1)
                    {
                        tempLowerLimit++;
                    }
                    break;
                }
            }
            DispSet();
        }
    }

    if (KEY_SUB == 0 && setLocation != 0) //减键按下
    {
        DelayMs(180);
        if (KEY_SUB == 0 && setLocation != 0)
        {
            switch (setLocation)
            {
                case 1:
                {
                    if (setTime > 0)
                    {
                        setTime--;
                    }
                    break;
                }
                case 2:
                {
                    if (tempUpperLimit > tempLowerLimit + 1)
                    {
                        tempUpperLimit--;
                    }
                    break;
                }
                case 3:
                {
                    if (tempLowerLimit > 0)
                    {
                        tempLowerLimit--;
                    }
                    break;
                }
            }
            DispSet();
        }
        
    }
    if (KEY_ENTER == 0) //确认键按下
    {
        DelayMs(20);
        if (KEY_ENTER == 0)
        {
            LCD_WriteCommand(0x0C, 0); //关闭光标闪烁
            DispNormal();
            if (setTime > 0)
            {
                timeCnt = 0;
                switchFlag = 0;
                RELAY_HEATING = OFF;
                TR0 = 1;
            }
            else
            {
                switchFlag = ~switchFlag; //手动启停
            }
            setLocation = 0;
        }
        while (KEY_ENTER == 0);

    }

    //红外按键处理
    if (irFlag == 1)
    {
        if (irCode[2] == IRCodeMap[8][0]) //遥控设置键
        {
            setLocation++;
            if (setLocation == 4)
            {
                setLocation = 1;
            }
            DispSet();
        }

        else if (irCode[2] == IRCodeMap[7][0]) //+键
        {
            if (setLocation != 0)
            {
                switch (setLocation)
                {
                    case 1:
                    {
                        if (setTime < 999)
                        {
                            setTime++;
                        }
                        break;
                    }
                    case 2:
                    {
                        if (tempUpperLimit < 99 - 1)
                        {
                            tempUpperLimit++;
                        }
                        break;
                    }
                    case 3:
                    {
                        if (tempLowerLimit < tempUpperLimit - 1)
                        {
                            tempLowerLimit++;
                        }
                        break;
                    }
                }
            }
            DispSet();
        }

        else if (irCode[2] == IRCodeMap[6][0]) //-键
        {
            if (setLocation != 0)
            {
                switch (setLocation)
                {
                    case 1:
                    {
                        if (setTime > 0)
                        {
                            setTime--;
                        }
                        break;
                    }
                    case 2:
                    {
                        if (tempUpperLimit > tempLowerLimit + 1)
                        {
                            tempUpperLimit--;
                        }
                        break;
                    }
                    case 3:
                    {
                        if (tempLowerLimit > 0)
                        {
                            tempLowerLimit--;
                        }
                        break;
                    }
                }
            }
            DispSet();
        }

        else if (irCode[2] == IRCodeMap[5][0]) //确定键
        {
            
            LCD_WriteCommand(0x0C, 0); //关闭光标闪烁
            DispNormal();
            if (setTime > 0)
            {
                timeCnt = 0;
                switchFlag = 0;
                RELAY_HEATING = OFF;
                TR0 = 1;
            }
            else
            {
                switchFlag = ~switchFlag; //手动启停
            }
            setLocation = 0;
        }
        irFlag = 0;
    }
}

/************************* 提醒函数 *************************/
void Alarm(void)
{
    if (alarmFlag == 1 && buzzerFlag == 0)
    {
        BUZZER = ON;
        DelayMs(250);
        DelayMs(250);
        BUZZER = OFF;

        buzzerFlag = 1;
    }
}

/************************* 初始化定时器0 *************************/
void InitTimer0(void)
{
    TMOD &= 0xF0;
    TMOD |= 0x01; //设置工作方式 16位自动重载

	TL0 = 0x00;		//设置定时初始值
	TH0 = 0x4C;		//设置定时初始值  50ms

    ET0 = 1;     //打开允许开关
    TR0 = 1; //t0开始计时
    EA = 1;      //打开中断总开关
}

/************************* 定时器0中断 *************************/
void Timer0_Interrupt() interrupt 1 //定时器函数
{
    static unsigned char cnt;

	TL0 = 0x00;		//设置定时初始值
	TH0 = 0x4C;		//设置定时初始值  50ms

    timeCnt++;
    if (timeCnt >= 1200) //timeCnt=20为1s钟  1200为1分钟
    {
        timeCnt = 0;
        if (setTime != 0)
        {
            setTime--;
            if (setTime == 0) //定时时间到
            {
                switchFlag  = 1; //开启加热
            }
        }
    }

    cnt++;
    if (cnt >= 20) //1s发送一次数据
    {
        cnt = 0;
        sendFlag = 1;
    }
    
}

/************************* 红外解码定时器程序 *************************/
void Ext0_Interrupt(void) interrupt 0 //外部中断解码程序_外部中断0
{
    unsigned char i, j;
    unsigned char byt;
    unsigned int time;

    time = IR_GetLowTime();

    if ((time < 7833) || (time > 8755))
    {
        IE0 = 0;
        return;
    } //找到启始码

    time = IR_GetHighTime();

    if ((time < 3686) || (time > 4608)) //时间判定范围为4.0~5.0ms,
    {                                   //超过此范围则说明为误码,直接退出
        IE0 = 0;
        return;
    }
    //接收并判定后续的4 字节数据
    for (i = 0; i < 4; i++) //循环接收4 个字节
    {
        for (j = 0; j < 8; j++) //循环接收判定每字节的8 个bit
        {
            //接收判定每bit 的560us 低电平
            time = IR_GetLowTime();
            if ((time < 313) || (time > 718)) //时间判定范围为340~780us,
            {                                 //超过此范围则说明为误码,直接退出
                IE1 = 0;
                return;
            }
            //接收每bit 高电平时间,判定该bit 的值
            time = IR_GetHighTime();
            if ((time > 313) && (time < 718)) //时间判定范围为340~780us,
            {                                 //在此范围内说明该bit 值为0
                byt >>= 1;                    //因低位在先,所以数据右移,高位为0
            }
            else if ((time > 1345) && (time < 1751)) //时间判定范围为1460~1900us,
            {                                        //在此范围内说明该bit 值为1
                byt >>= 1;                           //因低位在先,所以数据右移,
                byt |= 0x80;                         //高位置1
            }
            else //不在上述范围内则说明为误码,直接退出
            {
                IE0 = 0;
                return;
            }
        }
        irCode[i] = byt; //接收完一个字节后保存到缓冲区
    }
    irFlag = 1; //接收完毕后设置标志
    IE0 = 0;    //退出前清零INT0 中断标志
}


仿真演示视频:
https://www.bilibili.com/video/BV1dP411c7k3/

实物演示视频:
https://www.bilibili.com/video/BV1id4y1b7mL/

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/84773.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

使用DevExpress WPF主题设计器轻松创建Office 2019绿色主题(二)

DevExpress WPF拥有120个控件和库&#xff0c;将帮助您交付满足甚至超出企业需求的高性能业务应用程序。通过DevExpress WPF能创建有着强大互动功能的XAML基础应用程序&#xff0c;这些应用程序专注于当代客户的需求和构建未来新一代支持触摸的解决方案。 DevExpress WPF的The…

数据结构C语言版 —— 时间复杂度空间复杂度概念和计算

文章目录时间复杂度&空间复杂度1. 算法效率2. 时间复杂度1) 时间复杂度的概念2) 大O的渐近表示法3) 时间复杂度案例举例3. 空间复杂度1) 空间复杂度概念2) 计算实例时间复杂度&空间复杂度 1. 算法效率 算法效率分析一般分为两种&#xff0c;一种是时间效率&#xff0…

Unity初学者Shader Graph教程

Unity初学者Shader Graph教程 了解面向非程序员的 Unity 引擎可视化着色器编程工具的来龙去脉 课程英文名&#xff1a;Your Ultimate Guide to Shader Graph for Beginners 此视频教程共28.0小时&#xff0c;中英双语字幕&#xff0c;画质清晰无水印&#xff0c;源码附件全 …

基于C++实现(WinForm)家谱管理系统【100010033】

⼀、需求分析 《家谱管理系统》程序的设计⽬的&#xff0c;是为了解决中国传统家谱不易保存、不易修改、不易统计的缺陷。利⽤计算机程序&#xff0c;可以实现在计算机上存储、管理、查看家谱的相关信息。 ⽬标功能&#xff1a; 建⽴家谱&#xff1a;在计算机上建⽴树状家谱结…

二叉树算法

写在前面 树的定义 typedef struct Node {int data;struct Node *lchild,*rchild; }Bnode,*Btree;最近公共祖先 已知一棵二叉树按顺序存储结构进行存储&#xff0c;设计一个算法&#xff0c;求编号分别为i和j的两个节点的最近公共祖先节点的值。 算法思想&#xff1a; 顺序…

牛客竞赛每日俩题 - Day9

目录 日期推算 分解因数 日期推算 美国节日__牛客网 思路&#xff1a; 首先&#xff0c;我们要想找到一个月第N个星期W&#xff0c;一定需要一个参照物&#xff0c;最好的目标当然是这个月的第一天。拿到参照物后&#xff0c;我要能得 到参照物的星期数&#xff0c;然后就能…

马来酰亚胺聚乙二醇叠氮,MAL-PEG-N3,Maleimide-PEG-Azide

马来酰亚胺聚乙二醇叠氮&#xff08;MAL-PEG-N3&#xff09;是MeloPEG的硫醇反应性“点击化学”PEG交联剂之一。马来酰亚胺与pH 6.5-7.5的游离巯基/巯基迅速反应&#xff0c;形成稳定的&#xff0c;不可裂解的硫醚键。马来酰亚胺和硫醇的反应使得叠氮化物功能性PEG能够快速有效…

[附源码]Python计算机毕业设计Django的家政服务平台

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

设计模式-Acyclic Visitor(非循环访问者模式)

目的&#xff1a;允许将新功能添加到现有的类层次结构中&#xff0c;而不会影响这些层次结构&#xff0c;也不会有四人帮访客模式中那样循环依赖的问题。 类图&#xff1a; 使用场景&#xff1a; 需要在现有层次结构中添加新功能而无需更改或影响该层次结构时。 当某些功能在层…

Linux服务器远程访问通过Tomcat部署的静态资源

一、安装Java和Tomcat 1.1 安装Java 下载jdk8切换到root用户&#xff0c;创建文件夹/usr/local/java&#xff0c;将下载的jdk压缩包上传到该目录下&#xff0c;解压 mkdir /usr/local/java cd /usr/local/java rz tar -zxvf jdk-8u351-linux-x64.tar.gz编辑配置文件&#xf…

数据技术篇之数据服务

第6章 数据服务 1.服务架构演进 演进过程 DWSOA &#xff08;1&#xff09;实施原理   将业务方对数据的需求通过SOA服务的方式暴露出去。有需求驱动&#xff0c;一个需求开发一个或则几个接口&#xff0c;编写接口文档&#xff0c;开放给业务方调用。 &#xff08;2&…

[附源码]Python计算机毕业设计SSM基于WEB的心理测评系统(程序+LW)

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

【手把手教你】使用qstock进行量化回测

01qstock简介qstock由“Python金融量化”公众号开发&#xff0c;试图打造成个人量化投研分析开源库&#xff0c;目前包括数据获取&#xff08;data&#xff09;、可视化(plot)、选股(stock)和量化回测&#xff08;backtest&#xff09;四个模块。其中数据模块&#xff08;data&…

SAS中用单因素ANOVA研究不同疗法对焦虑症的有效性

本教程将介绍如何使用SAS进行单因素方差分析。 最近我们被客户要求撰写关于单因素ANOVA的研究报告&#xff0c;包括一些图形和统计输出。我们想研究不同疗法对焦虑症的有效性。我们收集了以下类别的75个主题的样本&#xff1a; 无处理&#xff08;1个n1 27&#xff09;。生物…

5W3H法与SMART原则的结合使用

5W3H 5W3H分析法&#xff0c;又称“八何分析法”。在实际工作中&#xff0c;人们常常运用5W3H分析法&#xff0c;进行顾客分析、市场需求分析&#xff0c;解决计划编制的结构问题、方向问题、执行力问题。 5W3H是描述问题的手段&#xff0c;其具体指的是&#xff1a;What&…

【图像处理】深入解析LBP算法

问题 LBP是一种常见的特征描述算法&#xff0c;用来提取局部的纹理特征&#xff0c;其原理其实很简单&#xff0c;下面我们就来看看它是怎么一回事吧。 LBP简介 LBP&#xff08;Local Binary Patterns&#xff0c;局部二值模式&#xff09;是一种很简单但很高效的局部纹理特…

2025年DMS前装搭载或突破750万辆!多方势力搅局「融合集成」

在智能驾驶功能和舱内人机交互体验不断融合演进的过程中&#xff0c;基于摄像头的DMS/OMS功能正在进入新的发展周期。而在欧洲等部分国家及地区&#xff0c;DMS的标配&#xff08;主要涉及驾驶安全的监控&#xff0c;比如&#xff0c;驾驶员疲劳、注意力分散、安全带使用、吸烟…

java工厂策略模式的开发应用

java工厂策略模式的开发应用前言准备工作具体实现测试仰天大笑出门去&#xff0c;我辈岂是蓬蒿人前言 大概内容&#xff1a; 假设我有五个车间&#xff0c;每一个车间生产不同的车子&#xff0c;想要统一管理&#xff0c;不关心哪个车间生产什么车&#xff1b;只需找一个代理…

Ajax(一)

1.客户端与服务器 1.1 服务器 上网过程中&#xff0c;负责存放和对外提供资源的电脑。 1.2 客户端 上网过程中&#xff0c;负责获取和消费资源的电脑。 2.URL地址 2.1 URL地址的概念 2.2 URL地址的组成部分 3. 分析网页的打开过程 3.1 图解客户端与服务器的通信过程 三个…

NLP学习笔记(一) RNN基本介绍

大家好&#xff0c;我是半虹&#xff0c;这篇文章来讲循环神经网络 (Recurrent Neural Network, RNN) 文章行文思路如下&#xff1a; 首先通过前馈神经网络引出为啥需要循环神经网络然后介绍循环神经网络的核心思想与运作方式最后拓展两个循环神经网络常见且常用的变体 在讲循…