一、使用proteus绘制简单的电路图,用于后续仿真
二、编写程序
/********************************************************************************************************************
---- @Project: LED-74HC595
---- @File: main.c
---- @Edit: ZHQ
---- @Version: V1.0
---- @CreationTime: 20200628
---- @ModifiedTime: 20200629
---- @Description:
---- 刚上电开机时,红棋加时键对应S1键,红棋减时键对应S2键.。
---- 刚上电开机时,黑棋加时键对应S3键,黑棋减时键对应S4键.。
---- 比赛中途暂停双方计时的暂停按键对应S6键。刚上电时,复位双方默认20分时间的复位按键对应S7按键。
---- 红棋的抢时按键对应S13键,黑棋的抢时按键对应S16按键。
---- 棋类计时器有点像抢答器,本质上有两个计时器。比赛的时候对弈的两个棋友各用一个不同的按键抢时间,红棋走一步棋后,就按一下自己的抢时按键,这个时候红棋的计时器停止计时,而黑棋的计时器开始计时,黑棋走了一步棋后,按一下自己的计时器,黑棋停止计时,红棋继续计时,依次循环,谁的时间最先用完谁就输,蜂鸣器也会发出长鸣的声音提示时间到。
---- 上电开机默认双方各有20分钟的时间,左边显示的是红棋的时间,右边显示的是黑棋的时间。此时可以通过S1,S2.,S3,S4的加减按键来设置各自的最大倒计时时间。此时如果按下复位按键S7,会自动把双方的时间设置为默认的20分钟。
---- 设置好最大倒计时的时间后,此时任意一方按下各自的抢时按键(S13或者S16),则自己的计时器停止计时,而对方开始倒计时。此时数码管显示的是对方的时间,而自己的时间屏蔽不显示。
---- 在开始倒计时的时候,如果中途有棋友要接听电话或者忙别的事情,需要暂时暂停一下双方的时间,这个时候可以按S6暂停按键来暂停双方的计时,忙完后再次按下暂停按键会继续倒计时。任何一方的时间走完,都会蜂鸣器长鸣提示。
---- 单片机:AT89C52
********************************************************************************************************************/
#include "reg52.h"
/*——————宏定义——————*/
#define FOSC 11059200L
#define T1MS (65536-FOSC/12/500) /*0.5ms timer calculation method in 12Tmode*/
#define const_voice_short 40 /*蜂鸣器短叫的持续时间*/
#define const_voice_long 900 /*蜂鸣器长叫的持续时间*/
#define const_key_time 9 /*按键去抖动延时的时间*/
#define const_1s 96 /*大概产生一秒钟的时间基准*/
/*——————变量函数定义及声明——————*/
/*定义数码管的74HC595*/
sbit Dig_Hc595_Sh = P2^0;
sbit Dig_Hc595_St = P2^1;
sbit Dig_Hc595_Ds = P2^2;
/*定义蜂鸣器*/
sbit Beep = P2^7;
/*作为中途暂停指示灯 亮的时候表示中途暂停*/
sbit LED = P3^5;
/*定义按键*/
sbit Key_S1 = P0^0; /*第一行输入*/
sbit Key_S2 = P0^1; /*第二行输入*/
sbit Key_S3 = P0^2; /*第三行输入*/
sbit Key_S4 = P0^3; /*第四行输入*/
sbit Key_D1 = P0^4; /*第一列输入*/
sbit Key_D2 = P0^5; /*第二列输入*/
sbit Key_D3 = P0^6; /*第三列输入*/
sbit Key_D4 = P0^7; /*第四列输入*/
unsigned char ucKeyStep = 1; /*按键扫描步骤变量*/
unsigned int uiKeyTimeCnt = 0; /*按键去抖动延时计数器*/
unsigned char ucKeyLock = 0; /*按键触发后自锁的变量标志*/
unsigned char ucRowRecord = 1; /*记录当前扫描到第几列了*/
unsigned char ucKeySec = 0; /*被触发的按键编号*/
unsigned char ucDigShow8 = 0; /*第8位数码管要显示的内容*/
unsigned char ucDigShow7 = 0; /*第7位数码管要显示的内容*/
unsigned char ucDigShow6 = 0; /*第6位数码管要显示的内容*/
unsigned char ucDigShow5 = 0; /*第5位数码管要显示的内容*/
unsigned char ucDigShow4 = 0; /*第4位数码管要显示的内容*/
unsigned char ucDigShow3 = 0; /*第3位数码管要显示的内容*/
unsigned char ucDigShow2 = 0; /*第2位数码管要显示的内容*/
unsigned char ucDigShow1 = 0; /*第1位数码管要显示的内容*/
unsigned char ucDigDot8; /*数码管8的小数点是否显示的标志*/
unsigned char ucDigDot7 = 1; /*数码管7的小数点是否显示的标志*/
unsigned char ucDigDot6; /*数码管6的小数点是否显示的标志*/
unsigned char ucDigDot5; /*数码管5的小数点是否显示的标志*/
unsigned char ucDigDot4; /*数码管4的小数点是否显示的标志*/
unsigned char ucDigDot3 = 1; /*数码管3的小数点是否显示的标志*/
unsigned char ucDigDot2; /*数码管2的小数点是否显示的标志*/
unsigned char ucDigDot1; /*数码管1的小数点是否显示的标志*/
unsigned char ucDigShowTemp = 0; /*临时中间变量*/
unsigned char ucDisplayDriveStep = 1; /*动态扫描数码管的步骤变量*/
unsigned char ucWd = 1; /*本程序的核心变量,窗口显示变量。类似于一级菜单的变量。代表显示不同的窗口。*/
unsigned int uiRedTimeCnt = 0; /*红棋产生秒基准的时间计时器*/
unsigned int uiBlackTimeCnt = 0; /*黑棋产生秒基准的时间计时器*/
unsigned int uiRedTotal = 1200; /*红棋的总时间*/
unsigned int uiBlackTotal = 1200; /*黑棋的总时间*/
unsigned char ucRedFlag = 0; /*红棋是否开始计时的标志*/
unsigned char ucBlackFlag = 0; /*黑棋是否开始计时的标志*/
unsigned char ucDisplayUpdate = 1; /*更新显示标志*/
unsigned int uiVoiceCnt = 0; /*蜂鸣器鸣叫的持续时间计数器*/
void Dig_Hc595_Drive(unsigned char, unsigned char);
/*根据原理图得出的共阴数码管字模表*/
code unsigned char Dig_Table[] =
{
0x3f, /*0 序号0*/
0x06, /*1 序号1*/
0x5b, /*2 序号2*/
0x4f, /*3 序号3*/
0x66, /*4 序号4*/
0x6d, /*5 序号5*/
0x7d, /*6 序号6*/
0x07, /*7 序号7*/
0x7f, /*8 序号8*/
0x6f, /*9 序号9*/
0x00, /*不显示 序号10*/
0x40, /*- 序号11*/
0x73, /*P 序号12*/
};
/**
* @brief 定时器0初始化函数
* @param 无
* @retval 初始化T0
**/
void Init_T0(void)
{
TMOD = 0x01; /*set timer0 as mode1 (16-bit)*/
TL0 = T1MS; /*initial timer0 low byte*/
TH0 = T1MS >> 8; /*initial timer0 high byte*/
}
/**
* @brief 外围初始化函数
* @param 无
* @retval 初始化外围
* 让数码管显示的内容转移到以下几个变量接口上,方便以后编写更上一层的窗口程序。
* 只要更改以下对应变量的内容,就可以显示你想显示的数字。
**/
void Init_Peripheral(void)
{
ET0 = 1;/*允许定时中断*/
TR0 = 1;/*启动定时中断*/
EA = 1;/*开总中断*/
}
/**
* @brief 初始化函数
* @param 无
* @retval 初始化单片机
**/
void Init(void)
{
LED = 1;
Beep = 1;
Dig_Hc595_Drive(0x00, 0x00); /*关闭所有经过另外两个74HC595驱动的LED灯*/
Init_T0();
}
/**
* @brief 延时函数
* @param 无
* @retval 无
**/
void Delay_Long(unsigned int uiDelayLong)
{
unsigned int i;
unsigned int j;
for(i=0;i for(j=0;j<500;j++) /*内嵌循环的空指令数量*/ { ; /*一个分号相当于执行一条空语句*/ } } } /** * @brief 延时函数 * @param 无 * @retval 无 **/ void Delay_Short(unsigned int uiDelayShort) { unsigned int i; for(i=0;i ; /*一个分号相当于执行一条空语句*/ } } /** * @brief 时间应用函数 * @param 无 * @retval 放在定时中断里的时间应用程序 **/ void time_service(void) { if(ucRedFlag == 1) /*1代表红棋在运行中*/ { uiRedTimeCnt ++; if(uiRedTimeCnt > const_1s) { uiRedTimeCnt = 0; if(uiRedTotal > 0) { uiRedTotal --; } else /*时间到*/ { ucRedFlag = 0; /*红棋和黑棋同时停止计时*/ ucBlackFlag = 0; ucWd = 1; /*切换到第一个窗口的状态*/ uiVoiceCnt = const_voice_long; /*报警声音触发*/ } ucDisplayUpdate = 1; /*更新显示*/ } } if(ucBlackFlag == 1) /*1代表黑棋在运行中*/ { uiBlackTimeCnt ++; if(uiBlackTimeCnt > const_1s) { uiBlackTimeCnt = 0; if(uiBlackTotal > 0) { uiBlackTotal --; } else /*时间到*/ { ucRedFlag = 0; /*红棋和黑棋同时停止计时*/ ucBlackFlag = 0; ucWd = 1; /*切换到第一个窗口的状态*/ uiVoiceCnt = const_voice_long; /*报警声音触发*/ } ucDisplayUpdate = 1; /*更新显示*/ } } } /** * @brief 显示数码管字模的驱动函数 * @param 无 * @retval 动态驱动数码管的原理 * 在八位数码管中,在任何一个瞬间,每次只显示其中一位数码管,另外的七个数码管 * 通过设置其公共位com为高电平来关闭显示,只要切换画面的速度足够快,人的视觉就分辨不出来,感觉八个数码管 * 是同时亮的。以下dig_hc595_drive(xx,yy)函数,其中第一个形参xx是驱动数码管段seg的引脚,第二个形参yy是驱动 * 数码管公共位com的引脚。 **/ void Display_Drive(void) { switch(ucDisplayDriveStep) { case 1: /*显示第1位*/ ucDigShowTemp = Dig_Table[ucDigShow1]; if(ucDigDot1 == 1) { ucDigShowTemp = ucDigShowTemp | 0x80; /*显示小数点*/ } Dig_Hc595_Drive(ucDigShowTemp, 0xfe); break; case 2: /*显示第2位*/ ucDigShowTemp = Dig_Table[ucDigShow2]; if(ucDigDot2 == 1) { ucDigShowTemp = ucDigShowTemp | 0x80; /*显示小数点*/ } Dig_Hc595_Drive(ucDigShowTemp, 0xfd); break; case 3: /*显示第3位*/ ucDigShowTemp = Dig_Table[ucDigShow3]; if(ucDigDot3 == 1) { ucDigShowTemp = ucDigShowTemp | 0x80; /*显示小数点*/ } Dig_Hc595_Drive(ucDigShowTemp, 0xfb); break; case 4: /*显示第4位*/ ucDigShowTemp = Dig_Table[ucDigShow4]; if(ucDigDot4 == 1) { ucDigShowTemp = ucDigShowTemp | 0x80; /*显示小数点*/ } Dig_Hc595_Drive(ucDigShowTemp, 0xf7); break; case 5: /*显示第5位*/ ucDigShowTemp = Dig_Table[ucDigShow5]; if(ucDigDot5 == 1) { ucDigShowTemp = ucDigShowTemp | 0x80; /*显示小数点*/ } Dig_Hc595_Drive(ucDigShowTemp, 0xef); break; case 6: /*显示第6位*/ ucDigShowTemp = Dig_Table[ucDigShow6]; if(ucDigDot6 == 1) { ucDigShowTemp = ucDigShowTemp | 0x80; /*显示小数点*/ } Dig_Hc595_Drive(ucDigShowTemp, 0xdf); break; case 7: /*显示第7位*/ ucDigShowTemp = Dig_Table[ucDigShow7]; if(ucDigDot7 == 1) { ucDigShowTemp = ucDigShowTemp | 0x80; /*显示小数点*/ } Dig_Hc595_Drive(ucDigShowTemp, 0xbf); break; case 8: /*显示第8位*/ ucDigShowTemp = Dig_Table[ucDigShow8]; if(ucDigDot8 == 1) { ucDigShowTemp = ucDigShowTemp | 0x80; /*显示小数点*/ } Dig_Hc595_Drive(ucDigShowTemp, 0x7f); break; } ucDisplayDriveStep ++; /*逐位显示*/ if(ucDisplayDriveStep > 8) /*扫描完8个数码管后,重新从第一个开始扫描*/ { ucDisplayDriveStep = 1; } } /** * @brief 数码管的595驱动函数 * @param 无 * @retval * 如果直接是单片机的IO口引脚驱动的数码管,由于驱动的速度太快,此处应该适当增加一点delay延时或者 * 用计数延时的方式来延时,目的是在八位数码管中切换到每位数码管显示的时候,都能停留一会再切换到其它 * 位的数码管界面,这样可以增加显示的效果。但是,由于是间接经过74HC595驱动数码管的, * 在单片机驱动74HC595的时候,dig_hc595_drive函数本身内部需要执行很多指令,已经相当于delay延时了, * 因此这里不再需要加delay延时函数或者计数延时。 **/ void Dig_HC595_Drive(unsigned char ucDigStatusTemp16_09, unsigned char ucDigStatusTemp08_01) { unsigned char i; unsigned char ucTempData; Dig_Hc595_Sh = 0; Dig_Hc595_St = 0; ucTempData = ucDigStatusTemp16_09; /*先送高8位*/ for(i = 0; i < 8; i ++) { if(ucTempData >= 0x80) { Dig_Hc595_Ds = 1; } else { Dig_Hc595_Ds = 0; } /*注意,此处的延时delay_short必须尽可能小,否则动态扫描数码管的速度就不够。*/ Dig_Hc595_Sh = 0; /*SH引脚的上升沿把数据送入寄存器*/ Delay_Short(1); Dig_Hc595_Sh = 1; Delay_Short(1); ucTempData = ucTempData <<1; } ucTempData = ucDigStatusTemp08_01; /*再先送低8位*/ for(i = 0; i < 8; i ++) { if(ucTempData >= 0x80) { Dig_Hc595_Ds = 1; } else { Dig_Hc595_Ds = 0; } Dig_Hc595_Sh = 0; /*SH引脚的上升沿把数据送入寄存器*/ Delay_Short(1); Dig_Hc595_Sh = 1; Delay_Short(1); ucTempData = ucTempData <<1; } Dig_Hc595_St = 0; /*ST引脚把两个寄存器的数据更新输出到74HC595的输出引脚上并且锁存起来*/ Delay_Short(1); Dig_Hc595_St = 1; Delay_Short(1); Dig_Hc595_Sh = 0; /*拉低,抗干扰就增强*/ Dig_Hc595_St = 0; Dig_Hc595_Ds = 0; } /** * @brief 扫描按键 * @param 无 * @retval 放在定时中断里 **/ void Key_Scan(void) { switch(ucKeyStep) { case 1: /*按键扫描输出第ucRowRecord列低电平*/ if (ucRowRecord == 1) /*第一列输出低电平*/ { Key_D1 = 0;
上一篇:51单片机实现带数码管显示的加法简易计算器
下一篇:51单片机实现在数码管中实现iphone4S开机密码锁的程序
推荐阅读
史海拾趣
针对晶体二极管单相全波电阻负载整流电路,网友可能提出多个问题,以下是一些可能的问题及其详细回答:
1. 什么是晶体二极管单相全波电阻负载整流电路?
回答:晶体二极管单相全波电阻负载整流电路是一种将交流电(AC)转换为直流电(DC)的电路,它利用了晶体二极管的单向导电特性。这种电路通过四个二极管(通常构成桥式整流器)实现全波整流,即同时利用交流电的正负半周进行整流,从而提高整流效率。负载电阻则用于消耗整流后的直流电。
2. 该电路的工作原理是什么?
回答:晶体二极管单相全波电阻负载整流电路的工作原理基于二极管的单向导电性。在交流电的正半周,二极管D1和D3导通,D2和D4截止,电流通过D1、负载电阻和D3形成回路,负载上得到上正下负的电压。在交流电的负半周,情况相反,D2和D4导通,D1和D3截止,电流方向不变,同样通过负载电阻,从而在负载上持续得到方向不变的直流电压。
3. 为什么需要四个二极管而不是两个?
回答:使用四个二极管构成桥式整流器可以充分利用交流电的正负两个半周,实现全波整流。如果只使用两个二极管,则只能实现半波整流,即只利用交流电的一个半周进行整流,整流效率较低。桥式整流电路通过巧妙地连接四个二极管,使得在每个半周内都有电流通过负载,从而提高了整流效率。
4. 如何提高该电路的整流效率?
回答:提高晶体二极管单相全波电阻负载整流电路的整流效率可以从以下几个方面入手:
- 选择合适的变压器:根据整流电路的输入电压和输出电压要求,选择合适的变压器,以保证整流电路的稳定性和效率。
- 优化整流器设计:通过选择合适的二极管和优化整流器的布局,可以提高整流器的整流效率和稳定性。
- 增加滤波电路:在整流器的输出端增加滤波电路(如电容和电感),可以滤除整流后的脉动直流电中的交流成分,得到更加平滑的直流电,从而提高整流效率。
5. 该电路有哪些应用?
回答:晶体二极管单相全波电阻负载整流电路具有广泛的应用,包括但不限于以下几个方面:
- 电源适配器:将交流电转换为直流电,为各种电子设备提供稳定的电源。
- 充电器:用于为电池充电,将交流电转换为适合电池充电的直流电。
- 电源模块:作为电源模块的一部分,为各种电子设备提供稳定的直流电源。
- 工业控制:在工业控制领域,为各种控制设备提供稳定的直流电源。
6. 该电路是否存在局限性?
回答:尽管晶体二极管单相全波电阻负载整流电路具有许多优点,但也存在一些局限性。例如,整流后的直流电仍存在一定的脉动成分,需要通过滤波电路进一步处理;此外,整流过程中会产生一定的能量损失,包括二极管的正向导通压降和滤波电路中的能量损耗等。因此,在实际应用中需要根据具体需求选择合适的整流电路和滤波方案。
随着市场的不断变化和客户需求的日益多样化,兵字公司意识到,要想在市场中保持领先地位,就必须不断进行技术创新。于是,公司投入大量研发资源,成功研发出了小型精密互感器。这一创新产品不仅填补了市场空白,还为公司带来了可观的收益。同时,兵字公司也凭借这一技术突破,进一步巩固了在电子元器件领域的领先地位。
Murata是一家历史悠久的电子元器件公司,他们早在电子设备发展的初期就认识到了EMI滤波器的重要性。随着电子产品的不断小型化,Murata看到了小型化EMI滤波器的巨大市场需求。他们通过与多家科研机构和高校合作,成功研发出了一系列高性能、小型化的EMI滤波器。这些滤波器不仅性能优越,而且体积小巧,非常适合用于现代电子设备中。Murata因此赢得了大量的市场份额,并成为了EMI滤波器领域的佼佼者。
TDK公司,作为全球知名的电子元器件制造商,早在上世纪80年代就开始涉足EMI滤波器领域。当时,随着电子设备的日益普及,电磁干扰问题逐渐凸显。TDK看准市场需求,投入大量资源进行EMI滤波器的研发。初期,他们面临着技术难题和市场接受度的挑战。然而,通过不断的技术创新和产品迭代,TDK的EMI滤波器逐渐在市场上站稳了脚跟。他们的产品在计算机、通信设备等领域得到了广泛应用,有效解决了电磁干扰问题,提升了设备的稳定性和可靠性。
APTA Group Inc在企业管理方面同样敢于创新。公司引入了一套先进的管理系统,通过数据分析和智能化决策,大大提高了工作效率和决策准确性。同时,公司还注重员工培训和激励,打造了一支高效、专业的团队。这些创新的管理举措,为APTA的稳健发展提供了坚实保障。
本信息来自合作QQ群:NXP Cortex-M0/M3交流(87394268) 群管理员在坛子里的ID:pepsi360 请教一个问题,周立功自己做的TKStudio大家有用过吗?… 查看全部问答∨ |
|
公司要做一个智能手机上的软件,要求如下: 系统能够自动采集手机终端的相关网络数据,并自动生成测量报告,测量报告中包含以下内容: ? 当前小区信号强度 ? 邻小区信号强度 ? &n ...… 查看全部问答∨ |
|
昨天用TI的低功耗温度传感器TMP112,单片机是MSP430F2132,使用IO模拟I2C,总是在写入第二个字节后,传感器没有应答,第一个字节是器件地址,第二个字节是内部寄存器地址,不知道怎么回事?各位大侠有用过这个传感器的吗?… 查看全部问答∨ |
Xilinx DDS Compiler IP核使用问题,用过的高手来看一下 我要在FPGA中做数字正交,故此要用到DDS IP核,5.0版本的,但是研究了两天始终无法产生比较满意的正玄波,用ChipScope采到的波形要不是就是像下图这种,要不就是一些比较混乱的,反正不是比较正常的正玄波。 设置如下,外面给的aclk是50Mhz的, ...… 查看全部问答∨ |