指针的第四大好处,指针作为数组在函数中的输入输出接口

发布者:数据探险家最新更新时间:2021-11-04 来源: eefocus关键字:指针  数组  函数  输入输出接口 手机看文章 扫描二维码
随时随地手机看文章

一、使用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; /*将串口接收到的数据缓存到接收缓冲区里*/

[1] [2]
关键字:指针  数组  函数  输入输出接口 引用地址:指针的第四大好处,指针作为数组在函数中的输入输出接口

上一篇:为指针加上紧箍咒const,避免意外修改了只做输入接口的数据
下一篇:指针的第三大好处,指针作为数组在函数中的输出接口

推荐阅读最新更新时间:2024-10-15 08:46

低成本函数发生器
用一个EPROM、一个标准D/A变换器和一个计数器可构成一个简单灵活的函数发生器。其原理是基于直接数字合成(DDS)基础上的,它把所需函数的数字取样存储在存储器(如EPROM)中并以周期性方式读出。 数字取样由存储器时钟频率周期T时分。D/A变换器把数字化信号变换为模拟信号,而低通滤波器选择基带信号(第1Nyquist视窗,从O到Fs/2,其中Fs=T/2)。 大多数市场出售的DDS芯片具有正弦波输出,其输出频率由用户时钟设置。用EPROM可以建造所希望的信号周期长度。为避免失真,第一取样和最后取样应该相同,如同时序重复,在输出没有不连续性。 图1所示电路将产生所需要的时序,其频率限制在16kHz和500kHz之间。4
[嵌入式]
一个简单的SOCKET程序的数据包结构和封解包函数
练习写套接字通信程序时候写的一段代码,本来想写个聊天室但写来写去进度卡在界面上接节下来都是通信部分的代码 因为只是试验用所以都是用C写的,等界面部分完工后会用类来封装一下 因为本人E文很烂所以变量和函数的命名是具有中国特色的,求理解.不过我注释的很详细了 谨以此文纪念我那坑爹的编程自学生涯...................... #include stdio.h #include windows.h //////////////////////////////数据包接构////////////////////////////////////// //数据包类型CTOS为客户端使用的数据包,STOC为服务端使用的数据包 #
[单片机]
步进电机S曲线生成器的计算以及使用
一.计算原理 根据上一节内容,已经计算了一条任意S曲线的函数。在步进电机S曲线加减速的控制中,需要的S曲线如图1所示,横轴为时间,纵轴为角速度,其中w0为起始角速度,w1为终止角速度 在S曲线加减速控制中,加减速的角度是已知的,根据第五节内容公式③,已经计算了角度与步数的函数关系式为下式 根据第五节内容公式⑥,已经计算了 定时器 初值与速度的函数关系式为下式,其中速度变量n的单位为RPM 将图1的横轴变换为步数,纵轴变换的转速(RPM)后,得到图2关系图 根据第五节内容公式⑤,S曲线的函数为下式,其中该S曲线的起点P1与终点P2坐标分别为(1,n0), (,n1),带入公式后即可求得K,B值,曲线方程即计算完成。
[嵌入式]
步进电机S曲线生成器的计算以及使用
嵌入式便携设备中电源管理的分析与研究
引言   现今对电子系统设备性能的要求越来越高,在权衡电子系统的性能和功耗时,电子系统的性能往往得到更多的重视。容量有限的电池是便携设备的惟一能量来源,而电池容量的提高速度明显赶不上中央处理器性能的提高速度,因此,如何利用有限的电能为便携设备提供最高性能,是便携设备中电源管理的主要目标。除此之外,电源管理还要兼顾稳定性和散热性。电源管理模块是在可编程电源管理的设备上,为电源管理提供实现各种功耗模式的应用编程接口的软件模块。   功率消耗有两种方式: 静态功耗和动态功耗。静态功耗主要为晶体管泄漏(leakage)功率;动态功耗则来源于电路有效性激活,例如地址线或者数据线输入时引起的寄存器线路的有效性激活。开关电容所消耗的功率是动
[应用]
【单片机笔记】keil c51编译环境不能跳转函数的解决办法
keil c51 不能使用:Go to Definition of....的解决方法 最近使用keil c51 开发usb固件,当向vc一样使用Go to Definition of....时,出现警告对话框: no browse information available in'工程目录' check- then rebuild the target! 查了一些资料,原因大概有二: 1、代码中嵌有汇编,汇编代码去掉后能查看。 2、选中target1,然后点击右键,选择 option for target 'target1' ,在弹出的对话框中选择output选项卡,看Browse
[单片机]
西门子博途:SET:置位字节数组指令参数及工作原理
说明 可以使用“置位位数组”指令,将指定区域中位的信号状态置位为“1”。在参数 S_BIT 中使用指针定义范围起点。如果指针指向外部 I/O 的存储区,则不执行该指令。在参数 N 中指定要在指定区域中置位的位数。如果参数 N 的值为“0”,则调用该指令不起作用。 只有 MCR 位为“1”时,才会执行该指令。如果 MCR 位的信号状态为“0”,则指定区域中的位保持不变。 说明 参数 N 在 TIA Portal 中,还可以将参数 N 中数据块的元素进行互连。 参数 下表列出了“置位位数组”指令的参数: 参数 声明 数据类型 存储区 说明 S_BIT Input POINTER I、Q、M、D 指向区域中第一个位的
[嵌入式]
西门子博途:SET:置位字节<font color='red'>数组</font>指令参数及工作原理
STM32时钟代码的解析,以及启动函数指针跳转
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE; //HSE表示使用外部时钟,HSI表示使用的是内部时钟 RCC_OscInitStruct.HSEState = RCC_HSE_ON; // 打开外部时钟 RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON; RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE; //时钟源设置 RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL6; //表示6倍频率,如果外部时钟源是8MHZ
[单片机]
stm32库函数GPIO_Init()解析
GPIO_Init函数是IO引脚的初始化函数,进行个个引脚的初始化配置,主要接受两个参数,一个是配置引脚组(GPIO_TypeDef* GPIOx),一个是配置的参数( GPIO_InitTypeDef* GPIO_InitStruct),具体如下 void GPIO_Init(GPIO_TypeDef* GPIOx, GPIO_InitTypeDef* GPIO_InitStruct) /*其中第一个参数为那组引脚,每组拥有16个引脚,每组都具有不同的寄存器配置地址,第二个参数是一个数据结构,也就是将基本配置信息放在这个数据结构里面,再将这个结构传入函数进行配置*/ //其中数据机构可以表示为如下 typede
[单片机]
小广播
设计资源 培训 开发板 精华推荐

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

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

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

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

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