一、使用proteus绘制简单的电路图,用于后续仿真

二、编写程序


/********************************************************************************************************************

---- @Project: return

---- @File: main.c

---- @Edit: ZHQ

---- @Version: V1.0

---- @CreationTime: 20200808

---- @ModifiedTime: 20200808

---- @Description:

---- 波特率是:9600 。

---- 通讯协议:EB 00 55  XX YY  

---- 加无效填充字节后,上位机实际上应该发送:00  EB 00 55  XX YY 

---- 其中第1位00是无效填充字节,防止由于硬件原因丢失第一个字节。

---- 其中第2,3,4位EB 00 55就是数据头

----           后2位XX YY就是有效数据

---- 任意时刻,单片机从电脑“串口调试助手”上位机收到的一串数据中,只要此数据中包含关键字EB 00 55 ,并且此关键字后面两个字节的数据XX YY 分别为01 02,那么蜂鸣器鸣叫一声表示接收的数据头和有效数据都是正确的。

---- 单片机:AT89C52

********************************************************************************************************************/

#include "reg52.h"

/*——————宏定义——————*/

#define FOSC 11059200L

#define BAUD 9600

#define T1MS (65536-FOSC/12/500)   /*0.5ms timer calculation method in 12Tmode*/

 

#define const_voice_short 19 /*蜂鸣器短叫的持续时间*/

#define const_rc_size 10 /*接收串口中断数据的缓冲区数组大小*/

 

#define const_receive_time 5 /*如果超过这个时间没有串口数据过来,就认为一串数据已经全部接收完,这个时间根据实际情况来调整大小*/

 

/*——————变量函数定义及声明——————*/

/*蜂鸣器的驱动IO口*/

sbit BEEP = P2^7;

/*LED*/

sbit LED = P3^5;

 

unsigned int uiSendCnt = 0; /*用来识别串口是否接收完一串数据的计时器*/

unsigned char ucSendLock = 1; /*串口服务程序的自锁变量,每次接收完一串数据只处理一次*/

unsigned int uiRcregTotal = 0; /*代表当前缓冲区已经接收了多少个数据*/

unsigned char ucRcregBuf[const_rc_size]; /*接收串口中断数据的缓冲区数组*/

unsigned int uiRcMoveIndex = 0; /*用来解析数据协议的中间变量*/

 

unsigned int uiVoiceCnt = 0; /*蜂鸣器鸣叫的持续时间计数器*/

 

/**

* @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 初始化T0

**/

void Init_USART(void)

{

SCON = 0x50;

TMOD = 0x21;                    

TH1=TL1=-(FOSC/12/32/BAUD);

}

 

/**

* @brief  外围初始化函数

* @param  无

* @retval 初始化外围

* 让数码管显示的内容转移到以下几个变量接口上,方便以后编写更上一层的窗口程序。

* 只要更改以下对应变量的内容,就可以显示你想显示的数字。

**/

void Init_Peripheral(void)

{

ET0 = 1;/*允许定时中断*/

TR0 = 1;/*启动定时中断*/

TR1 = 1;

ES = 1; /*允许串口中断*/

EA = 1;/*开总中断*/  

}

 

/**

* @brief  初始化函数

* @param  无

* @retval 初始化单片机

**/

void Init(void)

{

LED  = 0;

Init_T0();

Init_USART();

}

/**

* @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 

* 以下函数说明了,在空函数里,可以插入很多个return语句。

* 用return语句非常便于后续程序的升级修改。

**/

void usart_service(void)

{

// /*如果超过了一定的时间内,再也没有新数据从串口来*/

// if(uiSendCnt >= const_receive_time && ucSendLock == 1)

// {

// 原来的语句,现在被两个return语句替代了

if(uiSendCnt < const_receive_time) /* 延时还没超过规定时间,直接退出本程序,不执行return后的任何语句。 */

{

return; /* 强行退出本子程序,不执行以下任何语句 */

}

if(ucSendLock == 0) /* 不是最新一次接收到串口数据,直接退出本程序,不执行return后的任何语句。 */

{

return; /* 强行退出本子程序,不执行以下任何语句 */

}

/*

 * 以上两条return语句就相当于原来的一条if(uiSendCnt>=const_receive_time&&ucSendLock==1)语句。

 * 用了return语句后,就明显减少了一个if嵌套。

 */

ucSendLock = 0; /*处理一次就锁起来,不用每次都进来,除非有新接收的数据*/

/*下面的代码进入数据协议解析和数据处理的阶段*/

uiRcMoveIndex = 0; /*由于是判断数据头,所以下标移动变量从数组的0开始向最尾端移动*/

// /*

// * 判断数据头,进入循环解析数据协议必须满足两个条件:

// * 第一:最大接收缓冲数据必须大于一串数据的长度(这里是5。包括2个有效数据,3个数据头)

// * 第二:游标uiRcMoveIndex必须小于等于最大接收缓冲数据减去一串数据的长度(这里是5。包括2个有效数据,3个数据头)

// */

// while(uiRcregTotal >= 5 && uiRcMoveIndex <= (uiRcregTotal - 5))

// {

// 原来的语句,现在被两个return语句替代了

while(1) /* 死循环可以被以下return或者break语句中断,return本身已经包含了break语句功能。 */

{

if(uiRcregTotal < 5) /* 串口接收到的数据太少 */

{

uiRcregTotal = 0; /*清空缓冲的下标,方便下次重新从0下标开始接受新数据*/

return; /* 强行退出while(1)循环嵌套,直接退出本程序,不执行以下任何语句 */

}

if(uiRcMoveIndex > (uiRcregTotal - 5)) /* 数组缓冲区的数据已经处理完 */

{

uiRcregTotal = 0; /*清空缓冲的下标,方便下次重新从0下标开始接受新数据*/

return; /* 强行退出while(1)循环嵌套,直接退出本程序,不执行以下任何语句 */

}

/* 

 * 以上两条return语句就相当于原来的一条while(uiRcregTotal>=5&&uiRcMoveIndex<=(uiRcregTotal-5))语句。

 * 以上两个return语句的用法,同时说明了return本身已经包含了break语句功能,不管当前处于几层的内部循环嵌套,

 * 都可以强行退出循环,并且直接退出本程序。

 */

if(ucRcregBuf[uiRcMoveIndex + 0] == 0xeb && ucRcregBuf[uiRcMoveIndex + 1] == 0x00 && ucRcregBuf[uiRcMoveIndex + 2] == 0x55)

{

/*有效数据01 02的判断*/

if(ucRcregBuf[uiRcMoveIndex + 3] == 0x01 && ucRcregBuf[uiRcMoveIndex + 4] == 0x02)

{

uiVoiceCnt = const_voice_short; /*蜂鸣器发出声音,说明数据头和有效数据都接收正确*/

LED = ~LED; /*LED亮灭*/

}

break; /*退出while(1)循环*/

}

uiRcMoveIndex ++; /*因为是判断数据头,游标向着数组最尾端的方向移动*/

}

// }

uiRcregTotal = 0; /*清空缓冲的下标,方便下次重新从0下标开始接受新数据*/

// }

}

/**

* @brief  定时器0中断函数

* @param  无

* @retval 无

**/

void ISR_T0(void) interrupt 1

{

TF0 = 0;  /*清除中断标志*/

TR0 = 0; /*关中断*/

 

if(uiSendCnt < const_receive_time) /*如果超过这个时间没有串口数据过来,就认为一串数据已经全部接收完*/

{

uiSendCnt ++; /*表面上这个数据不断累加,但是在串口中断里,每接收一个字节它都会被清零,除非这个中间没有串口数据过来*/

ucSendLock = 1; /*开自锁标志*/

}

 

if(uiVoiceCnt != 0)

{

uiVoiceCnt --;

BEEP = 0;

}

else

{

;

BEEP = 1;

}

 

TL0 = T1MS;                     /*initial timer0 low byte*/

TH0 = T1MS >> 8;                /*initial timer0 high byte*/

  TR0 = 1; /*开中断*/

}

 

/**

* @brief  串口接收数据中断

* @param  无

* @retval 无

**/

void usart_receive(void) interrupt 4

{

if(RI == 1)

{

RI = 0;

++ uiRcregTotal;

if(uiRcregTotal > const_rc_size)

{

uiRcregTotal = const_rc_size;

}

ucRcregBuf[uiRcregTotal - 1] = SBUF; /*将串口接收到的数据缓存到接收缓冲区里*/

uiSendCnt = 0; /*及时喂狗,虽然main函数那边不断在累加,但是只要串口的数据还没发送完毕,那么它永远也长不大,因为每个中断都被清零。*/

}

else

{

TI = 0;

}

}

 

/*————————————主函数————————————*/

/**

* @brief  主函数

* @param  无

* @retval 实现LED灯闪烁

**/

void main()

{

/*单片机初始化*/

Init();

/*延时,延时时间一般是0.3秒到2秒之间,等待外围芯片和模块上电稳定*/

Delay_Long(100);

/*单片机外围初始化*/

Init_Peripheral();

while(1)

{

usart_service();

}

}

 

三、仿真实现


关键字:51单片机  return  程序框架 引用地址:51单片机(利用return)实现判断数据头来接收

上一篇:指针的第一大好处,让一个函数可以封装多个相当于return语句返回的参数
下一篇:51单片机实现操作AT24C02时,改善数码管的闪烁现象

推荐阅读

在今年八月底,全球第二大晶圆代工厂格芯正式宣布,放弃7nm和更先进制程的研发,这个决定公布之后,引发了市场上的广泛讨论。大家关注的重点就在于格芯为什么要放弃先进制程?未来格芯将何去何从? 格芯全球副总裁兼大中华区总经理白农(左)、格芯全球销售和业务发展的高级副总裁Mike Cadigan(中)和格芯首席技术官兼全球研发高级副总裁Gary Patton(...
三星电子官员近日表示,该公司正寻求在中国推出5G网络之际,提升其在中国智能手机市场的份额。 10月18日,三星在上海著名的步行街南京东路开设了一家旗舰店,店内摆放着该公司最新的智能手机、平板电脑和智能手表。800平方米的面积使其成为中国最大的三星门店。 11月1日,中国三大运营商在北京、上海和广州等主要城市正式开启5G商用服务,并计划在年底前将...
集微网11月5日消息,华为花粉俱乐部昨天发文解答了最受用户关心的关于EMUI11的8大问题,一起来看看吧。1.被称为史上最强的华为Mate40,有EMUI 11功劳吗?有什么特别之处呢?当然有。EMUI 11跨终端体验如多屏协同、手机和手表协同,基于人因研究的动效交互和全新艺术化设计,36个月持久流畅的性能黑科技,以及“不打扰、不追踪、不泄露”的隐私安全体验,...
一、使用proteus绘制简单的电路图,用于后续仿真二、编写程序/********************************************************************************************************************---- @Project: Pointer---- @File: main.c---- @Edit: ZHQ---- @Version: V1.0---- @CreationTime: 20200808---- @ModifiedTime: 20200808---- @Description: ---- 波...

史海拾趣

问答坊 | AI 解惑

【SynPlify技术问题】请教: 使用Synplify_Pro9.6遇到个棘手的问题

最近做了个模块使用了ISE的IP生成的FIFO,在Synplify_Pro9.6下面建了个工程,综合以后从Synplify_Pro9.6中打开ISE10.1,然后直接在ISE中做implement时报错.\"ERROR:NgdBuild:76 - File \"<xxx>.ngo\" cannot be merged into block \"xxx\" (TYPE=\ ...…

查看全部问答∨

请各位前辈谈谈医用监护设备方面的嵌入式开发前影如何

小弟最近收到一家医用监护设备制造商的研发工程师offer,不知道这个领域今后发展怎么样,该不该进入这个领域。…

查看全部问答∨

qtopia4可执行文件无法在开发板上运行

我安装的是qtopia4.2.1,自己写的程序,并生成了可执行文件,没有报错。但是当利用nfs挂载开发板,运行可执行文件时,报错:./qtopiatarget: 1: Syntax error: "(" unexpected,其中qtopiatarget为可执行文件。 运行qt自带的可执行文件没问题 ...…

查看全部问答∨

mmu映射怎么调试,大家都用什么方法呢

大家都用什么方法呢,我觉得我走了很多弯路,有什么好方法让大家共享吗?…

查看全部问答∨

想问一下 如何在vxworks下控制蜂鸣器?

如题,我现在想实现一下 在vxworks下 控制蜂鸣器的工作 不知道你能不能提供一些意见谢谢…

查看全部问答∨

垃圾问题, 当Pull数据的时候, 错误表已存在

我用Pull从服务器把数据拉到mobile上,第一次没问题,因为数据表及错误表都是在Pull时指定的, ... rda.Pull(this.tbFillTableName.Text, this.tbSelectStr.Text, this.tbSQLServerConnStr.Text, RdaTrackOption.TrackingOn, errTableName); ... ...…

查看全部问答∨

用STM32实现USB的IAP成功后,如何软复位

已经可以自动升级了,但是成功后我手工复位,真实太失败了请问如何使用指令达到复位的目的PS:不想用函数指针的方式,因为会无端用一级堆栈…

查看全部问答∨

求助!

有哪位高手用软件仿真过脉冲调宽式晶体管调压器啊?给我些帮助吧!!…

查看全部问答∨

抓取LINUX桌面的视频 FFmpeg

抓取LINUX桌面的视频 $ ffmpeg -f x11grab -s wxga -r 25 -i :0.0 -sameq /tmp/filename.mpg 我们在一些视频网站上看到别人的3D桌面怎么怎么酷的视频,通常就是这么来的,ffmpeg可以直接解码X11的图形,并转换到相应输出格式。 ffmpeg ...…

查看全部问答∨
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
随便看看

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved