一、使用proteus绘制简单的电路图,用于后续仿真
二、编写程序
/********************************************************************************************************************
---- @Project: Pointer
---- @File: main.c
---- @Edit: ZHQ
---- @Version: V1.0
---- @CreationTime: 20200809
---- @ModifiedTime: 20200809
---- @Description:
---- 波特率是:9600 。
---- 通讯协议:EB 00 55 XX YY
---- 把5个随机数据按从大到小排序,用冒泡法来排序。
---- 通过电脑串口调试助手,往单片机发送EB 00 55 08 06 09 05 07 指令,其中EB 00 55是数据头,08 06 09 05 07 是参与排序的5个随机原始数据。单片机收到指令后就会返回13个数据,最前面5个数据是第1种方法的排序结果,中间3个数据EE EE EE是第1种和第2种的分割线,为了方便观察,没实际意义。最后5个数据是第2种方法的排序结果.
----
---- 比如电脑发送:EB 00 55 08 06 09 05 07
---- 单片机就返回:09 08 07 06 05 EE EE EE 09 08 07 06 05
---- 单片机: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_array_size 5 /* 参与排序的数组大小 */
#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; /*蜂鸣器鸣叫的持续时间计数器*/
unsigned char ucUsartBuffer[const_array_size]; /* 从串口接收到的需要排序的原始数据 */
unsigned char ucGlobalBuffer_3[const_array_size]; /* 第3种方法,参与具体排序算法的全局变量数组 */
unsigned char ucGlobalBuffer_4[const_array_size]; /* 第4种方法,用来接收输出接口数据的全局变量数组 */
/**
* @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 unsigned char ucSendData * @retval 往上位机发送一个字节的函数 **/ void eusart_send(unsigned char ucSendData) { ES = 0; /* 关串口中断 */ TI = 0; /* 清零串口发送完成中断请求标志 */ SBUF = ucSendData; /* 发送一个字节 */ Delay_Short(400); /* 每个字节之间的延时,这里非常关键,也是最容易出错的地方。延时的大小请根据实际项目来调整 */ TI = 0; /* 清零串口发送完成中断请求标志 */ ES = 1; /* 允许串口中断 */ } /** * @brief 第3种方法 * @param p_ucInputBuffer p_ucOutputBuffer * @retval * 第3种方法,为了改进第2种方法的用户体验,用指针为函数多增加一个数组输出接口。 * 这样,函数的数组既有输入接口,又有输出接口,已经堪称完美了。 * 本程序中*p_ucInputBuffer输入接口,*p_ucOutputBuffer是输出接口。 **/ void big_to_small_sort_3(unsigned char *p_ucInputBuffer, unsigned char *p_ucOutputBuffer) { unsigned char i; unsigned char k; unsigned char ucTemp; /* 在两两交换数据的过程中,用于临时存放交换的某个变量 */ unsigned char ucBuffer_3[const_array_size]; /* 第3种方法,参与具体排序算法的局部变量数组 */ for(i = 0; i < const_array_size; i ++) /* 参与排序算法之前,先把输入接口的数据全部搬移到局部变量数组中。 */ { ucBuffer_3[i] = p_ucInputBuffer[i]; } /* 冒泡法 */ for(i = 0; i < (const_array_size - 1); i ++) /* 冒泡的次数是(const_array_size-1)次 */ { for(k = 0; k < (const_array_size - 1 - i); k ++) { if(ucBuffer_3[const_array_size - 1 - k] > ucBuffer_3[const_array_size - 1 - 1 - k]) { ucTemp = ucBuffer_3[const_array_size - 1 - 1 - k]; ucBuffer_3[const_array_size - 1 - 1 - k] = ucBuffer_3[const_array_size - 1 - k]; ucBuffer_3[const_array_size - 1 - k] = ucTemp; } } } for(i = 0; i < const_array_size; i ++) /* 参与排序算法之后,把运算结果的数据全部搬移到输出接口中,方便外面程序调用 */ { p_ucOutputBuffer[i] = ucBuffer_3[i]; } } /** * @brief 第4种方法 * @param p_ucInputAndOutputBuffer * @retval * 第4种方法.指针在函数的接口中,天生就是既可以做输入,也可以是做输出,它是双向性的,类似全局变量的特点。 * 我们可以根据实际项目的情况,在必要的时候可以直接把输入接口和输出接口合并在一起, * 这种方法的缺点是没有把输入和输出分开,没有那么直观。但是优点也是很明显的,就是比较 * 省程序ROM容量和数据RAM容量,而且运行效率也比较快。现在介绍给大家。 * 本程序的*p_ucInputAndOutputBuffer是输入输出接口。 **/ void big_to_small_sort_4(unsigned char *p_ucInputAndOutputBuffer) { unsigned char i; unsigned char k; unsigned char ucTemp; /* 在两两交换数据的过程中,用于临时存放交换的某个变量 */ /* 冒泡法 */ for(i = 0; i < (const_array_size - 1); i ++) /* 冒泡的次数是(const_array_size-1)次 */ { for(k = 0; k < (const_array_size - 1 - i); k ++) { if(p_ucInputAndOutputBuffer[const_array_size - 1 - k] > p_ucInputAndOutputBuffer[const_array_size - 1 - 1 - k]) { ucTemp = p_ucInputAndOutputBuffer[const_array_size - 1 - 1 - k]; p_ucInputAndOutputBuffer[const_array_size - 1 - 1 - k] = p_ucInputAndOutputBuffer[const_array_size - 1 - k]; p_ucInputAndOutputBuffer[const_array_size - 1 - k] = ucTemp; } } } } /** * @brief 串口服务程序 * @param 无 * @retval * 以下函数说明了,在空函数里,可以插入很多个return语句。 * 用return语句非常便于后续程序的升级修改。 **/ void usart_service(void) { unsigned char i = 0; // /*如果超过了一定的时间内,再也没有新数据从串口来*/ // 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) { for(i = 0; i < const_array_size; i ++) /* 从串口接收到的需要被排序的原始数据 */ { ucUsartBuffer[i] = ucRcregBuf[uiRcMoveIndex+3+i]; } /* 第3种运算方法,依靠指针为函数增加一个数组的输出接口 */ /* 通过指针输出接口,排序运算后的结果直接从这个输出口中导出到ucGlobalBuffer_3数组中 */ big_to_small_sort_3(ucUsartBuffer, ucGlobalBuffer_3); /* ucUsartBuffer是输入的数组,ucGlobalBuffer_3是接收排序结果的数组 */ for(i = 0; i < const_array_size; i ++) { eusart_send(ucGlobalBuffer_3[i]); /* 把用第3种方法排序后的结果返回给上位机观察 */ } /* 为了方便上位机观察,多发送3个字节ee ee ee作为第1种方法与第2种方法的分割线 */ eusart_send(0xee); eusart_send(0xee); eusart_send(0xee); /* 第4种运算方法,依靠一个指针作为函数的输入输出接口。 */ /* 通过这个指针输入输出接口,ucGlobalBuffer_4数组既是输入数组,也是输出数组,排序运算后的结果直接存放在它本身,类似于全局变量的特点。 */ for(i = 0; i < const_array_size; i ++) /* 从串口接收到的需要被排序的原始数据 */ { ucGlobalBuffer_4[i] = ucUsartBuffer[i]; } big_to_small_sort_4(ucGlobalBuffer_4); /* ucUsartBuffer是输入的数组,ucGlobalBuffer_3是接收排序结果的数组 */ for(i = 0; i < const_array_size; i ++) { eusart_send(ucGlobalBuffer_4[i]); /* 把用第3种方法排序后的结果返回给上位机观察 */ } 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; /*将串口接收到的数据缓存到接收缓冲区里*/
上一篇:为指针加上紧箍咒const,避免意外修改了只做输入接口的数据
下一篇:指针的第三大好处,指针作为数组在函数中的输出接口
推荐阅读
史海拾趣
机顶盒,全称为数字视频变换盒,是连接电视与外部信号源的关键设备,其专业性与科普性并重。作为现代家庭娱乐不可或缺的一部分,机顶盒不仅实现了数模转换功能,让用户在模拟电视机上也能观看数字电视,还通过不断的技术创新,丰富了用户的视听体验。
从广义上讲,机顶盒涵盖了从模拟频道增补器到数字卫星综合接收解码器(IRD)、有线电视数字机顶盒等多种类型。而从狭义上看,主要聚焦于数字设备,如数字卫星机顶盒(DVB-S)、数字地面机顶盒(DVB-T)及IPTV机顶盒等。这些机顶盒能接收来自有线电缆、卫星天线、宽带网络及地面广播的信号,并将其转换为电视机可识别的音视频格式,实现高清甚至4K超高清的播放效果。
随着互联网的普及,网络机顶盒更是成为连接电视与互联网的桥梁。它不仅提供传统的电视节目服务,还通过集成智能技术,实现了语音助手、智能推荐等功能,极大提升了用户体验。此外,网络机顶盒还能接入各种流媒体平台,让用户在电视上观看在线视频、玩游戏、浏览网页等,实现了电视功能的多元化和智能化。
综上所述,机顶盒以其强大的功能和不断的技术创新,成为现代家庭娱乐的重要组成部分,为用户带来了更为丰富、便捷、智能的视听享受。
为了加速技术的商业化进程,Fractus积极寻求与全球领先企业的合作。自2000年以来,公司陆续与多家知名手机厂商、电信运营商以及设备制造商建立了战略合作关系。这些合作不仅为Fractus带来了稳定的收入来源,也促进了其技术的广泛应用和持续改进。同时,Fractus还通过签订全球许可协议和建立战略伙伴关系,进一步拓展了其业务范围和市场影响力。
随着市场竞争的加剧,南晶电子意识到技术创新是企业发展的核心动力。公司投入大量研发资金,与多所高校和科研机构建立合作关系,共同研发新型半导体器件。经过不懈努力,南晶电子成功推出了一系列具有自主知识产权的高性能产品,赢得了市场的广泛认可。
FlyWin集团旗下的另一家公司,专注于电子元器件的回收与再利用。随着电子产品的快速迭代,废旧电子设备的处理成为一大难题。该公司通过建立完善的回收网络和先进的处理技术,有效回收了包括粤翔贴片排阻在内的多种电子元器件。这些回收的元件经过专业检测和处理后,部分被重新用于生产,部分则被安全环保地处置,有效减少了电子垃圾对环境的污染。这一举措不仅为公司带来了经济效益,也为社会的可持续发展贡献了一份力量。
近年来,随着全球电子产业的快速发展,中国市场的重要性日益凸显。COTO TECHNOLOGY也看到了这一机遇,开始深耕中国市场,寻求与当地企业的合作。通过与成都迈极芯科技等公司的战略合作,COTO不仅将先进的技术和产品引入中国,还为中国半导体测试产业提供了更优质的产品和服务。这一系列的合作不仅促进了COTO在中国市场的发展,也推动了中国电子产业的进步。
这五个故事只是COTO TECHNOLOGY发展历程中的一部分,但它们充分展示了这家公司在电子行业中的成长与变迁。从初创时期的线圈绕组制造商,到后来的干簧继电器领导者,再到全球范围内的扩张与战略合作,COTO始终保持着对技术的追求和对市场的敏锐洞察。这些故事不仅反映了COTO的发展历程,也见证了整个电子行业的变革与进步。
品质是企业的生命线。安碁科技自成立以来,一直注重品质管理和技术创新。2000年,公司引进了全自动化制程设备,正式生产SMD产品,并通过了ISO9002认证。此后,安碁科技陆续通过了多项国际认证,如TL-9000、TS16949等,证明了其在品质管理方面的卓越表现。同时,公司也不断投入研发资源,推动技术创新和产品升级,以满足市场不断变化的需求。
本板用于继电器单独控制打开、关闭。上位机(PC机)通过串口连接到本控制继电器板后,向控制继电器板发送一串ASCII代码指令既可控制某个(或全部)继电器打开或关闭,完全不必要考虑继电器的硬件电路上选择、驱动等问题。比如,串口输入“K6”,则第 ...… 查看全部问答∨ |
|
请问制作WINCE软件需要准备哪些东西?最近要开发一个ARM板上的WINCE程序,目前没什么头绪,请大家指导下。 若开发语言选择C++(EVC),我要从开发到将程序完全部署到我的板子上还需要哪些东西(包括硬件/软件等)。 … 查看全部问答∨ |
AT89S51不用介绍了。AM29LV800DB是1MByte的Flash,工作电压为3V。 AT89S51和1MByte的Am29LV800DB的Flash构成电路,对AM29LV800操作和RAM一样,但是我读不出其中的数据。 检查过电路,接线都正确,但是我就是读不出Am29LV800DB中的数据。 请问, ...… 查看全部问答∨ |
wince5和wince6下的3G\\CDMA\\EVDO驱动已经完成了,我用串口工具已经能够发送指令, 并可以拨打电话,发送短信等。 虚拟了多个串口,第一个串口作为拨号上网串口,我建立连接之后,选择此端口, 不能正常连接上互联网 我估计是Unimodem的初始 ...… 查看全部问答∨ |
INPUT Input; wchar_t Infor[500]; ...... //Infor已经赋值 for( int i=0 ; Infor!=\'\ ...… 查看全部问答∨ |
MSP430也是有竞争对手的,下面的一篇报告,是来自TI针对430产品线与同类型竞争产品的测试报告。知己知彼百战不殆,这个报告也可以让我们对其他产品有个定性的了解… 查看全部问答∨ |