一、使用proteus绘制简单的电路图,用于后续仿真
二、编写程序
/********************************************************************************************************************
----@Project:LED-74HC595
----@File:main.c
----@Edit:ZHQ
----@Version:V1.0
----@CreationTime:20200524
----@ModifiedTime:20200524
----@Description:两片联级的74HC595驱动的16个LED灯交叉闪烁。比如,先是第1,3,5,7,9,11,13,15八个灯亮,其它的灯都灭。然后再反过来,原来亮的就灭,原来灭的就亮。交替闪烁。
----单片机:AT89C52
********************************************************************************************************************/
#include "reg52.h"
/*——————宏定义——————*/
#define FOSC 11059200L
#define T1MS (65536-FOSC/12/1000) /*1ms timer calculation method in 12Tmode*/
#define const_time_level 400
/*——————变量函数定义及声明——————*/
/*定义74HC595*/
sbit Hc595_Sh = P2^3;
sbit Hc595_St = P2^4;
sbit Hc595_Ds = P2^5;
unsigned char ucLED1 = 0; /*代表16个灯的亮灭状态,0代表灭,1代表亮*/
unsigned char ucLED2 = 0;
unsigned char ucLED3 = 0;
unsigned char ucLED4 = 0;
unsigned char ucLED5 = 0;
unsigned char ucLED6 = 0;
unsigned char ucLED7 = 0;
unsigned char ucLED8 = 0;
unsigned char ucLED9 = 0;
unsigned char ucLED10 = 0;
unsigned char ucLED11 = 0;
unsigned char ucLED12 = 0;
unsigned char ucLED13 = 0;
unsigned char ucLED14 = 0;
unsigned char ucLED15 = 0;
unsigned char ucLED16 = 0;
unsigned char ucLed_update = 0; /*刷新变量。每次更改LED灯的状态都要更新一次。*/
unsigned char ucLedStep = 0; /*步骤变量*/
unsigned int uiTimeCnt = 0; /*统计定时中断次数的延时计数器*/
unsigned char ucLedStatus16_09 = 0; /*代表底层74HC595输出状态的中间变量*/
unsigned char ucLedStatus08_01 = 0; /*代表底层74HC595输出状态的中间变量*/
/**
* @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 初始化单片机
**/
voidInit(void)
{
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 595驱动函数 * @param 无 * @retval * 两个联级74HC595的工作过程: * 每个74HC595内部都有一个8位的寄存器,两个联级起来就有两个寄存器。ST引脚就相当于一个刷新 * 信号引脚,当ST引脚产生一个上升沿信号时,就会把寄存器的数值输出到74HC595的输出引脚并且锁存起来, * DS是数据引脚,SH是把新数据送入寄存器的时钟信号。也就是说,SH引脚负责把数据送入到寄存器里,ST引脚 * 负责把寄存器的数据更新输出到74HC595的输出引脚上并且锁存起来。 **/ void HC595_Drive(unsigned char ucLedStatusTemp16_09, unsigned char ucLedStatusTemp08_01) { unsigned char i; unsigned char ucTempData; Hc595_Sh = 0; Hc595_St = 0; ucTempData = ucLedStatusTemp16_09;/*先送高8位*/ for(i = 0; i < 8; i ++) { if(ucTempData >= 0x80) { Hc595_Ds = 1; } else { Hc595_Ds = 0; } Hc595_Sh = 0;/*SH引脚的上升沿把数据送入寄存器*/ Delay_Short(15); Hc595_Sh = 1; Delay_Short(15); ucTempData = ucTempData <<1; } ucTempData = ucLedStatusTemp08_01;/*再先送低8位*/ for(i = 0; i < 8; i ++) { if(ucTempData >= 0x80) { Hc595_Ds = 1; } else { Hc595_Ds = 0; } Hc595_Sh = 0;/*SH引脚的上升沿把数据送入寄存器*/ Delay_Short(15); Hc595_Sh = 1; Delay_Short(15); ucTempData = ucTempData <<1; } Hc595_St = 0;/*ST引脚把两个寄存器的数据更新输出到74HC595的输出引脚上并且锁存起来*/ Delay_Short(15); Hc595_St = 1; Delay_Short(15); Hc595_Sh = 0;/*拉低,抗干扰就增强*/ Hc595_St = 0; Hc595_Ds = 0; } /** * @brief LED更新函数 * @param 无 * @retval * 把74HC595驱动程序翻译成类似单片机IO口直接驱动方式的过程。 * 每次更新LED输出,记得都要把ucLed_update置1表示更新。 **/ void LED_Update() { if(ucLed_update == 1) { ucLed_update = 0;/*及时清零,让它产生只更新一次的效果,避免一直更新。*/ if(ucLED1 == 1) { ucLedStatus08_01 = ucLedStatus08_01 | 0x01; } else { ucLedStatus08_01 = ucLedStatus08_01 & 0xfe; } if(ucLED2 == 1) { ucLedStatus08_01 = ucLedStatus08_01 | 0x02; } else { ucLedStatus08_01 = ucLedStatus08_01 & 0xfd; } if(ucLED3 == 1) { ucLedStatus08_01 = ucLedStatus08_01 | 0x04; } else { ucLedStatus08_01 = ucLedStatus08_01 & 0xfb; } if(ucLED4 == 1) { ucLedStatus08_01 = ucLedStatus08_01 | 0x08; } else { ucLedStatus08_01 = ucLedStatus08_01 & 0xf7; } if(ucLED5 == 1) { ucLedStatus08_01 = ucLedStatus08_01 | 0x10; } else { ucLedStatus08_01 = ucLedStatus08_01 & 0xef; } if(ucLED6 == 1) { ucLedStatus08_01 = ucLedStatus08_01 | 0x20; } else { ucLedStatus08_01 = ucLedStatus08_01 & 0xdf; } if(ucLED7 == 1) { ucLedStatus08_01 = ucLedStatus08_01 | 0x40; } else { ucLedStatus08_01 = ucLedStatus08_01 & 0xbf; } if(ucLED8 == 1) { ucLedStatus08_01 = ucLedStatus08_01 | 0x80; } else { ucLedStatus08_01 = ucLedStatus08_01 & 0x7f; } if(ucLED9 == 1) { ucLedStatus16_09 = ucLedStatus16_09 | 0x01; } else { ucLedStatus16_09 = ucLedStatus16_09 & 0xfe; } if(ucLED10 == 1) { ucLedStatus16_09 = ucLedStatus16_09 | 0x02; } else { ucLedStatus16_09 = ucLedStatus16_09 & 0xfd; } if(ucLED11 == 1) { ucLedStatus16_09 = ucLedStatus16_09 | 0x04; } else { ucLedStatus16_09 = ucLedStatus16_09 & 0xfb; } if(ucLED12 == 1) { ucLedStatus16_09 = ucLedStatus16_09 | 0x08; } else { ucLedStatus16_09 = ucLedStatus16_09 & 0xf7; } if(ucLED13 == 1) { ucLedStatus16_09 = ucLedStatus16_09 | 0x10; } else { ucLedStatus16_09 = ucLedStatus16_09 & 0xef; } if(ucLED14 == 1) { ucLedStatus16_09 = ucLedStatus16_09 | 0x20; } else { ucLedStatus16_09 = ucLedStatus16_09 & 0xdf; } if(ucLED15 == 1) { ucLedStatus16_09 = ucLedStatus16_09 | 0x40; } else { ucLedStatus16_09 = ucLedStatus16_09 & 0xbf; } if(ucLED16 == 1) { ucLedStatus16_09 = ucLedStatus16_09 | 0x80; } else { ucLedStatus16_09 = ucLedStatus16_09 & 0x7f; } HC595_Drive(ucLedStatus16_09, ucLedStatus08_01); } } /** * @brief LED闪烁函数 * @param 无 * @retval 无 **/ void LED_Flicker(void) { switch(ucLedStep) { case 0: if(uiTimeCnt >= const_time_level) { uiTimeCnt = 0;/*时间计数器清零*/ ucLED1 = 1;/*每个变量都代表一个LED灯的状态*/ ucLED2 = 0; ucLED3 = 1; ucLED4 = 0; ucLED5 = 1; ucLED6 = 0; ucLED7 = 1; ucLED8 = 0; ucLED9 = 1; ucLED10 = 0; ucLED11 = 1; ucLED12 = 0; ucLED13 = 1; ucLED14 = 0; ucLED15 = 1; ucLED16 = 0; ucLed_update = 1; /*更新显示*/ ucLedStep = 1; } break; case 1: if(uiTimeCnt >= const_time_level) { uiTimeCnt = 0;/*时间计数器清零*/ ucLED1 = 0;/*每个变量都代表一个LED灯的状态*/ ucLED2 = 1; ucLED3 = 0; ucLED4 = 1; ucLED5 = 0; ucLED6 = 1; ucLED7 = 0; ucLED8 = 1; ucLED9 = 0; ucLED10 = 1; ucLED11 = 0; ucLED12 = 1; ucLED13 = 0; ucLED14 = 1; ucLED15 = 0; ucLED16 = 1; ucLed_update = 1; /*更新显示*/ ucLedStep = 0; } break; } } /** * @brief 定时器0中断函数 * @param 无 * @retval 无 **/ void ISR_T0(void)interrupt 1 { TF0 = 0; /*清除中断标志*/ TR0 = 0; /*关中断*/ if(uiTimeCnt < 0xffff)/*设定这个条件,防止uiTimeCnt超范围。*/ { uiTimeCnt ++; } TL0 = T1MS; /*initial timer0 low byte*/ TH0 = T1MS >> 8; /*initial timer0 high byte*/ TR0 = 1; /*开中断*/ } /*——————主函数——————*/ /** * @brief 主函数 * @param 无 * @retval 实现LED灯闪烁 **/ void main() { /*单片机初始化*/ Init(); /*延时,延时时间一般是0.3秒到2秒之间,等待外围芯片和模块上电稳定*/ Delay_Long(100); /*单片机外围初始化*/ Init_Peripheral(); while(1) { /*按键服务函数*/ LED_Flicker(); /*LED更新函数*/ LED_Update(); } } 三、仿真实现
上一篇:51单片机实现两片联级74HC595依次点亮LED后依次熄灭
下一篇:51单片机实现矩阵键盘的组合按键触发
推荐阅读

推荐帖子
- mini2440的按键控制LED小程序~~~
- 看到坛子里很多人拿到板子后不知道第一步该如何做, 在网上看到一个朋友的博客,感觉还不错,对于购买板子的朋友应该有所帮助,所以呈现出来: 来源:http://adreaman.com/0524arm-mini2440-led-key-non-os.html mini2440是当前最经济的arm9开发板了,目前就是500多元一块,麻雀虽小,重要的那几个部件也还算齐全,用它来做计算机系统底层的学习工具很合适。而且关于s3c2410/2440系列的教程资料和经验文章在网上很多,大家遇到问题时多
-
fish001
嵌入式系统
- MOS管 DS脚的峰值电压测试以哪个为标准
- MOS管的DS脚的峰值电压测试以哪个为标准呢 \0\0\0eeworldpostqqMOS管DS脚的峰值电压测试以哪个为标准
-
桂花蒸
电源技术
- PNG双缓冲问题,为什么我的图片不能显示呢??
- intwidth=dstRc.right-dstRc.left, heigth=dstRc.top-dstRc.bottom; HDCmemDC=CreateCompatibleDC(hdc);//关联两个DC HBITMAPhBitmap=CreateCompatibleBitmap(hdc,width,heigth);//在缓冲中建立BITMAP
-
kakashi2010
嵌入式系统
- EEWORLD大学堂----开关电源拓扑
-
hi5
电源技术
- 当CMOS遇上TTL
- 本帖最后由fish001于2018-6-2617:51编辑 当COMS遇上TTL,现阶段只有纠结,因为对两者之间的联系?区别?都没有理性的认识,为此今日在此做一小小比较。 TTL:Transistor—TransistorLogic.标准的TTL电路输入高电平最小为2V,输出高电平为2.4V,典型值为3.4V;输入低电平最高为0.8V,输出低电平最高0.4V,典型值为0.2V.TTL电路电源供电只允许在5V+/-10%范围内,且扇出数为10个以下的TTL门电路。
-
fish001
模拟与混合信号
- 【国产FPGA 正点原子DFPGL22G开发板】二、千兆以太网测评【硬件设计篇】
- 本帖最后由yyliu于2023-1-618:36编辑 声明:针对本帖中可能出现的侵权行为,请及时联系本人修改或删除。未经本人允许,请勿转载。若本帖存在错误或不足之处,烦请指正,本人会及时修改。 0.说明 第二期将对千兆以太网模块进行测评,由于本人从事硬件设计工作,第二期打算出两篇测评:硬件设计篇和程序设计篇。硬件设计篇主要介绍DFPGL22G开发板千兆以太网模块的硬件方案,以及一些硬件设计细节;程序设计篇主要介绍千兆以太网通信的程序框图和代码实现,有条件的话还会做一些
-
yyliu
国产芯片交流