基于单片机设计的电子柜锁

一、前言

随着现代社会的不断发展,电子柜锁的应用越来越广泛。传统的机械柜锁存在一些不便之处,例如钥匙容易丢失、密码容易泄露等问题。设计一款基于单片机的电子柜锁系统成为了一个有趣而有意义的项目。

该电子柜锁系统通过电磁锁作为柜锁的开关,通过继电器控制电磁锁的开关状态。用户可以通过矩阵键盘输入密码进行开锁,并且密码数据会通过LCD1602液晶显示屏进行显示。同时,系统还支持输入密码验证开锁和修改密码的功能。当用户成功输入正确的密码并开锁时,系统会通过蜂鸣器发出提示音。

这款电子柜锁系统的设计为了提高柜锁的安全性和便利性。相比传统的机械柜锁,电子柜锁具有以下优势:

【1】密码安全性:电子柜锁采用密码作为开锁方式,相比传统钥匙更加安全可靠,用户可以根据需要设置较复杂的密码,有效防止密码泄露和非法开锁。

【2】方便易用:用户只需要通过矩阵键盘输入密码即可开锁,无需携带钥匙或记忆复杂的机械操作步骤,操作简单方便。

【3】修改密码功能:用户可以根据需要随时修改密码,提高了柜锁的灵活性和可维护性。

【4】提示音提示:系统通过蜂鸣器发出提示音,让用户在输入密码和开锁成功时得到明确的反馈,提升了用户体验。

电子柜锁系统的设计不仅具有实用性,而且可以为学习嵌入式系统设计和单片机编程的初学者提供一个非常好的实践项目。通过这个项目,可以学习和掌握单片机的输入输出控制、按键扫描、LCD显示、蜂鸣器控制等相关知识和技术。还涉及到密码输入和验证的算法设计和实现,锻炼了逻辑思维和程序设计能力。

通过这个电子柜锁系统项目,可以体验到现代电子技术的魅力,提高柜锁的安全性和便利性,为用户提供更好的使用体验。image-20230823171105869

image-20230823170732930

image-20230823170655846

二、硬件选型介绍

硬件选型方面,根据需求,下面是电子柜锁的最终硬件选型:

【1】主控芯片:STC89C52单片机是一款常用的8位单片机,具有丰富的外设资源和较大的存储容量,适合作为电子柜锁的主控芯片。

【2】电磁锁:选择适合的电磁锁作为柜锁的开关,确保其能提供足够的安全性和可靠性。考虑使用12V电磁锁,满足电源和控制信号要求。

【3】继电器:使用继电器来控制电磁锁的通断,确保信号隔离和电流放大。

【4】矩阵键盘:选择适用的矩阵键盘用于输入密码。选择4x4矩阵键盘,具有16个按键,支持数字和功能键。

【5】LCD1602液晶显示屏:作为密码输入和状态显示的界面,LCD1602具有两行16列的字符显示,能够清晰显示输入的密码和相关提示信息。

【6】蜂鸣器:用于发出开锁成功、密码输入错误等提示音。

【7】电源模块:有稳定可靠的电源供应非常重要,选择使用AC/DC 5/12V适配器供电。

三、整体设计思路

软件设计逻辑和思路如下:

【1】初始化:在程序开始时,进行系统初始化设置,包括配置IO口、定时器和外设等。同时,需要初始化密码存储区、LCD1602显示屏和蜂鸣器等。

【2】密码输入和验证:通过矩阵键盘读取用户输入的密码。可以采用一个固定长度的密码,例如4位。用户每按下一个数字键,将其添加到密码缓冲区中,并在LCD1602上显示相应的“*”字符表示已输入。当输入的密码长度达到预设长度时,即可触发密码验证操作。

【3】密码验证:将密码缓冲区中的数字转换为字符串形式,与预先设置好的正确密码进行比较。如果密码输入正确,则进行开锁操作;否则,进行密码错误提示处理。

【4】开锁操作:当密码验证成功后,控制继电器通断,打开或关闭电磁锁。同时,通过蜂鸣器发出开锁成功的提示音,并在LCD1602上显示开锁成功信息。

【5】修改密码:提供修改密码的功能。在成功验证密码后,用户可以输入新密码进行修改。修改完成后,将新密码存储起来,供下次验证使用。

【6】状态显示:将相关的状态信息实时显示在LCD1602上,例如输入密码错误提示、修改密码成功提示等。

【7】系统保护:为了保护系统安全,可以设置安全策略,例如密码输入错误次数限制、锁定时间等。当达到错误次数上限或锁定时间到达时,系统会自动进行相应的保护处理。

【8】中断服务:使用定时器中断等方式进行按键检测和LCD1602刷新等操作,提高系统的实时性。

【9】循环检测:设计一个主循环函数,不断检测矩阵键盘的按键输入、执行开锁、密码验证、密码修改以及状态显示等功能。

四、项目代码

#include <reg51.h>
#include <intrins.h>#define PASSWORD_LENGTH 4    // 密码长度
#define MAX_ATTEMPTS 3    // 最大尝试次数sbit Buzzer = P1^0;    // 蜂鸣器控制引脚
sbit ElectromagneticLock = P1^1;    // 电磁锁控制引脚unsigned char password[PASSWORD_LENGTH] = {1, 2, 3, 4};    // 初始密码
unsigned char enteredPassword[PASSWORD_LENGTH];    // 输入的密码
unsigned char attempts = 0;    // 尝试次数// 延时函数
void delay(unsigned int ms) {unsigned int i, j;for (i = 0; i < ms; i++)for (j = 0; j < 110; j++);
}// 初始化LCD1602
void LCD_Init() {LCD_WriteCommand(0x38);    // 设置8位数据总线,2行显示,5x8点阵字符LCD_WriteCommand(0x0c);    // 显示器开,光标关闭LCD_WriteCommand(0x06);    // 光标右移,字符不移动LCD_WriteCommand(0x01);    // 清屏
}// 写入命令到LCD1602
void LCD_WriteCommand(unsigned char cmd) {LCD_RS = 0;LCD_RW = 0;LCD_EN = 1;P0 = cmd;_nop_();_nop_();LCD_EN = 0;
}// 写入数据到LCD1602
void LCD_WriteData(unsigned char dat) {LCD_RS = 1;LCD_RW = 0;LCD_EN = 1;P0 = dat;_nop_();_nop_();LCD_EN = 0;
}// 在LCD1602上显示字符串
void LCD_ShowString(unsigned char x, unsigned char y, unsigned char *str) {unsigned char i = 0;if (x < 16) {if (y == 0)LCD_WriteCommand(0x80 + x);else if (y == 1)LCD_WriteCommand(0xc0 + x);while (str[i] != '\0') {LCD_WriteData(str[i]);i++;}}
}// 初始化矩阵键盘
void Keypad_Init() {Keypad_Row1 = 1;Keypad_Row2 = 1;Keypad_Row3 = 1;Keypad_Row4 = 1;
}// 读取矩阵键盘按键值
unsigned char Keypad_Read() {unsigned char row, col;unsigned char keyVal;for (col = 0; col < 4; col++) {Keypad_Col1 = 1;Keypad_Col2 = 1;Keypad_Col3 = 1;Keypad_Col4 = 1;switch (col) {case 0:Keypad_Col1 = 0;break;case 1:Keypad_Col2 = 0;break;case 2:Keypad_Col3 = 0;break;case 3:Keypad_Col4 = 0;break;}for (row = 0; row < 4; row++) {if (Keypad_Row1 == 0) {delay(5);if (Keypad_Row1 == 0) {while (Keypad_Row1 == 0);keyVal = row * 4 + col + 1;return keyVal;}}if (Keypad_Row2 == 0) {delay(5);if (Keypad_Row2 == 0) {while (Keypad_Row2 == 0);keyVal = row * 4 + col + 5;return keyVal;}}if (Keypad_Row3 == 0) {delay(5);if (Keypad_Row3 == 0) {while (Keypad_Row3 == 0);keyVal= row * 4 + col + 9;return keyVal;}}if (Keypad_Row4 == 0) {delay(5);if (Keypad_Row4 == 0) {while (Keypad_Row4 == 0);keyVal = row * 4 + col + 13;return keyVal;}}}}return 0xFF;    // 返回0xFF表示没有按键按下
}// 检查输入的密码是否与设定密码一致
bit CheckPassword() {unsigned char i;for (i = 0; i < PASSWORD_LENGTH; i++) {if (enteredPassword[i] != password[i])return 0;    // 密码不一致}return 1;    // 密码一致
}// 输入密码
bit EnterPassword() {unsigned char i;unsigned char key;for (i = 0; i < PASSWORD_LENGTH; i++) {while ((key = Keypad_Read()) == 0xFF);enteredPassword[i] = key;LCD_WriteData('*');delay(300);}return CheckPassword();
}// 修改密码
void ChangePassword() {unsigned char i;LCD_ShowString(0, 1, "Enter New Password");for (i = 0; i < PASSWORD_LENGTH; i++) {while ((enteredPassword[i] = Keypad_Read()) == 0xFF);LCD_WriteData('*');delay(300);}for (i = 0; i < PASSWORD_LENGTH; i++)password[i] = enteredPassword[i];LCD_ShowString(0, 1, "Password Changed ");delay(1000);LCD_ShowString(0, 1, "Enter Password: ");
}// 开锁
void Unlock() {LCD_ShowString(0, 1, "Unlocking...");Buzzer = 1;    // 发出提示音ElectromagneticLock = 0;    // 解锁状态delay(2000);Buzzer = 0;    // 关闭提示音ElectromagneticLock = 1;    // 上锁状态LCD_ShowString(0, 1, "Enter Password: ");
}// 主函数
void main() {LCD_Init();    // 初始化LCD1602Keypad_Init();    // 初始化矩阵键盘LCD_ShowString(0, 0, "Electronic Lock");LCD_ShowString(0, 1, "Enter Password: ");while (1) {if (EnterPassword()) {Unlock();    // 密码正确,开锁attempts = 0;    // 尝试次数清零} else {attempts++;    // 尝试次数加一if (attempts >= MAX_ATTEMPTS) {LCD_ShowString(0, 1, "Max Attempts Exceeded");Buzzer = 1;    // 发出警报音delay(2000);Buzzer = 0;    // 关闭警报音attempts = 0;    // 尝试次数清零} else {LCD_ShowString(0, 1, "Wrong Password     ");delay(1000);LCD_ShowString(0, 1, "Enter Password: ");}}while ((Keypad_Read()) != 0xFF);    // 等待按键释放if (Keypad_Read() == '#') {ChangePassword();    // 输入'#'进入修改密码模式}}
}

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

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

相关文章

【JMeter参数化】上一个接口返回作为下一个接口入参

前言: 实际工作场景当中,比如获取到商品列表,并查看商品详情。如果将商品id写死,就很笨拙。所以我们可以进行参数化动态去更新商品id 目录 【同一个线程组内的】 场景1:接口A仅取一个值,作为接口B的入参 场景:接口A是获取教师列表中某个教室的id,接口B是查看该教师的详…

debounce and throtlle

debounce // 核心&#xff1a;单位时间内触发>1 则只执行最后一次。//excutioner 可以认为是执行器。执行器存在则清空&#xff0c;再赋值新的执行器。function debounce(fn, delay 500) {let excutioner null;return function () {let context this;let args arguments…

建筑能源管理(8)——合同能源管理

1、简介 合同能源管理是20世纪70年代中期在发达国家逐步发展起来的一种节能服务机制在国外简称EPC(Energy Performance Contracting)&#xff0c;在国内广泛地被称为EMC (Energy Management Contracting)&#xff0c;它由专门的节能服务公司(Energy Service Company,ESCO)在为…

STM32F4X SDIO(六) 例程讲解-SD_PowerON

STM32F4X SDIO&#xff08;六&#xff09; 例程讲解-SD_PowerON 例程讲解-SD_PowerONSDIO引脚初始化和时钟初始化SDIO初始化(单线模式)CMD0:GO_IDLE_STATE命令发送程序命令响应程序 CMD8:SEND_IF_CONDCMD8参数命令发送程序命令响应程序 CMD55:APP_CMDCMD55命令参数命令发送命令…

【Acwing170】加成序列(dfs+迭代加深+剪枝)题解和一点感想

本思路来自acwing算法提高课 题目描述 看本文需要准备的知识 1.dfs算法基本思想 2.对剪枝这个词有个简单的认识 迭代加深思想和此题分析 首先&#xff0c;什么是迭代加深呢&#xff1f;当一个问题的解有很大概率出现在递归树很浅的层&#xff0c;但是这个问题的解本身存在…

音视频开发:音频编码原理+采集+编码实战

原理&#xff1a; 消除冗余信息&#xff0c;压缩量最大&#xff0c;也叫有损压缩 剔除人耳听觉范围外的音频信号20Hz以下和20000Hz以上&#xff1b;去除被掩蔽的音频信号&#xff0c;信号的遮蔽可以分为频域遮蔽和时域遮蔽&#xff1b;频域遮蔽效应 屏蔽70分贝以下&#xff0…

汽车标定技术(一):XCP概述

目录 1.汽车标定概述 2.XCP协议由来及版本介绍 3.XCP技术通览 3.1 XCP上下机通信模型 3.2 XCP指令集 3.2.1 XCP帧结构定义 3.2.2 标准指令集 3.2.3 标定指令集 3.2.4 页切换指令集 3.2.5 数据采集指令集 3.2.6 刷写指令集 3.3 ECU描述文件(A2L)概述 3.3.1 标定上位…

C++中何时及如何使用析构函数

C中何时及如何使用析构函数 析构函数不返回任何值&#xff0c;没有返回类型&#xff0c;也没有函数参数。由于没 有函数参数&#xff0c;因此它不能被重载。换言之&#xff0c;一个类可以有多个构造函数&#xff0c;但是只能有一个析构函数。 何时调用析构函数&#xff1a; &…

有方N58 HTTP POST 请求连接 TDengine

串口调试软件&#xff1a;格西调试精灵 第一步先注册网络获取IP地址 建立PPP连接 ATXIIC1\r PPP链路建立成功&#xff0c;查询IP地址 ATXIIC?\r 设置网络APN ATCREG?\r 运行结果&#xff0c;红线处是获…

js:可选链运算符(?.)和空值合并运算符(??)

文档&#xff1a; 可选链运算符&#xff08;?.&#xff09;https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Operators/Optional_chaining空值合并运算符&#xff08;??&#xff09;https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Referenc…

【算法与数据结构】--算法和数据结构的进阶主题--并行算法和分布式数据结构

一、并行算法 1.1 并行计算概述 并行计算是一种计算方法&#xff0c;旨在通过同时执行多个计算任务来提高计算性能和效率。与传统的串行计算不同&#xff0c;其中每个任务按顺序执行&#xff0c;并行计算允许多个任务同时执行。这种并行性通常通过将计算任务分解为较小的子任…

算法:Java构建二叉树并迭代实现二叉树的前序、中序、后序遍历

先自定义一下二叉树的类&#xff1a; // Definition for a binary tree node. public class TreeNode {int val;TreeNode left;TreeNode right;TreeNode() {}TreeNode(int val) { this.val val; }TreeNode(int val, TreeNode left, TreeNode right) {this.val val;this.left…

MongoDB安装及开发系例全教程

一、系列文章目录 一、MongoDB安装教程—官方原版 二、MongoDB 使用教程(配置、管理、监控)_linux mongodb 监控 三、MongoDB 基于角色的访问控制 四、MongoDB用户管理 五、MongoDB基础知识详解 六、MongoDB—Indexs 七、MongoDB事务详解 八、MongoDB分片教程 九、Mo…

ATE新能源汽车充电桩自动负载测试系统

随着新能源汽车的普及&#xff0c;充电桩的需求也在不断增加&#xff0c;为了确保充电桩的性能和安全性&#xff0c;对其进行负载测试是非常重要的。ATE新能源汽车充电桩自动负载测试系统是一种专门用于检测充电桩性能的设备&#xff0c;它可以模拟各种实际使用场景&#xff0c…

6、QtCharts 悬浮曲线效果

文章目录 效果dialog.hdialog.cpp悬浮槽函数 效果 dialog.h #ifndef DIALOG_H #define DIALOG_H#include <QDialog> #include <QtCharts> #include <QLineSeries> #include <QGraphicsScene> #include <QTimer> #include <QSplineSeries>…

《实战:如何搭建一个完整的 Vue2.0 项目》- 7、Vue2.x 项目 webpack 4 升级 5(半自动升级)

1.自动升级 先全局安装升级插件 npm i npm-check npm-check-updates -g检查依赖 npm-check更新检查后的依赖并展示版本号&#xff0c;此时 package.json还没有更新 npm-check-updates升级 package.json&#xff0c;下图显示更新版本&#xff0c;此时 package.json文件已变更…

【JavaScript】变量提升

变量提升&#xff08;Hoisting&#xff09;被认为是&#xff0c;Javascript 中执行上下文&#xff08;特别是创建和执行阶段&#xff09;工作方式的一种认识。 提升&#xff08;Hoisting&#xff09;这个词。不过&#xff0c;需要注意的是&#xff0c;开始时&#xff0c;这个概…

FFmpeg——使用Canvas录制视频尚存问题的解决方案

个人简介 &#x1f440;个人主页&#xff1a; 前端杂货铺 &#x1f64b;‍♂️学习方向&#xff1a; 主攻前端方向&#xff0c;正逐渐往全干发展 &#x1f4c3;个人状态&#xff1a; 研发工程师&#xff0c;现效力于中国工业软件事业 &#x1f680;人生格言&#xff1a; 积跬步…

编译正点原子LINUXB报错make: arm-linux-gnueabihf-gcc:命令未找到

编译正点原子LINUX报错make: arm-linux-gnueabihf-gcc&#xff1a;命令未找到 1.报错内容2.解决办法3./bin/sh: 1: lzop: not found4.编译成功 1.报错内容 make: arm-linux-gnueabihf-gcc&#xff1a;命令未找到CHK include/config/kernel.releaseCHK include/generat…