0 引言
实时控制系统程序设计中,常涉及到小数运算问题.计算机系统中用二进制表示小数的方法有定点数表示法和浮点数表示法.采用浮点数表示法表示的小数范围大、精度高,但程序代码长,运算速度慢.定点数表示的小数范围小、精度低,但程序代码短,运算速度快.
使用C语言设计程序具有程序可读性强,编程方便等优点,但按常规方法设计程序时,实时性不如采用汇编语言设计的程序,这在涉及到小数运算时表现更为突出.这样就限制了C语言的应用.如果采用合适的计算方法,使用C语言编程可以获得与汇编语言编程同样的实时性.
实时控制系统中的前向通道采集的原始数据大多是定点整数,例如前向模拟通道的A/D转换器的转换结果,定时/计数器的计数结果等,都是定点整数.而系统的后向通道能接受的输入量也都为整数,即由量化产生的有限字长误差不可避免,精确到小数位的控制量因执行机构无法接受而不得不舍去.因而,虽然采用定点数表示小数的方法精度低,但在大多数情况下,仍能满足实时控制系统的控制精度要求.
MCS-51单片机的内部程序存储器仅有4K,运算速度较慢.对于实时性、代码长度限制要求较高的控制系统,采用MCS-51单片机控制时,不宜大量采用浮点运算.本文介绍Keil C51下的16位定点小数的乘法程序.
1 定点小数运算算法
1.1 控制算法的特点.
计算机实时控制系统中,控制算法通常可用下面的差分方程表示.
式中y[n]为第n个采样周期的输出,通常为二进制整数;x [n]为第n个采样周期的输入,通常也为二进制整数;ai、bi为实系数.在保证计算精度的条件下,计算上述差分方程时,将系数ai、bi转换成整数或定点小数,会大幅度提高运算速度和大幅度减少代码长度.这对于在程序存储器容量或运算速度有限的嵌入式控制器中实现快速控制算法计算有重要意义.
1.2 定点小数
小数可分为整数部分为0的纯小数和带整数的小数.纯小数可直接用定点小数表示,当使用16位定点小数时,分辩率可达2-16,可以获的足够的运算精度.
1.3 定点算法
设x为十进制纯小数,M为16位二进制整数.若程序需计算y=(x·M)取整,则可先将x转换成16位二进制定点小数.
X =(x·65536)取整 (2)
由于X的小数点在X的最高位前,2个16位二进制数相乘结果为32位二进制数,小数点在高16位和低16位间,乘法运算后的高16位为计算结果的整数部分,低16位为计算结果的小数部分.即
(x·M)取整=(X·M)取高16位 (3)
这样处理后可以大幅度提高运算速度,且大幅度减少代码长度.
汇编语言程序设计中的取整操作容易实现,在C语言中实现取整操作可以使用联合体,方法如下.
先定义2个联合体.
union{
unsigned char a_byte[4];
long a_long;
}r;
union{
unigned char b_byte[2];
int b_int;
}p;
第一个是长整数变量与4字节变量的联合体,长整型变量用于保存计算结果,第二个是整型变量与2个字节型变量的联合体,用于取整运算.在Keil C51中,长整数占4个字节,在RAM中按从高到低的顺序存放,r.a_byte[0]、r.a_byte[1]存放计算结果的整数部分,r.a_byte[2]、r.a_byte[3]存放计算结果的小数部分.
通过下列程序,实现取整运算.
p.b_byte[0]=r.a_byte[0];
p.b_byte[1]=r.a_byte[1];
这样p.b_int为计算结果的整数部分.以上程序在编译后仅为2条数据传送指令,需要4个机器周期的执行时间.与采用除法运算或移位运算实现取整运算相比,具有更快的执行速度.
2 程序
设程序需要计算0.12345乘16位二进制数后取整,采用浮点数时的程序如下所示.
main()
{
int b;
b=20000;
a=0.12345*b;
}
本程序的运行结果a=2527,程序编译后长度513字节,做浮点运算时需要602个机器周期.
main()
{
int a,b;
union{
char c[4];
long d;
}u1;
union{
char e[2];
int f;
}u2;
b=20000;
u1.d=(long)8090*b;
u2.e[0]=u1.c[0];
u2.e[1]=u1.c[2];
}
本程序的运行结果u2.f=2527,程序编译后长度129字节,做整数运算时仅需134个机器周期.
3 结束语
采用本文中所述方法,使用c语言设计MCS-51单片机控制算法程序,可获得与采用汇编语言设计的控制算法程序同样的效果.充分发挥了
c语言设计程序的优点,笔者在设计的某控制系统时,采用这种方法在获得了很好的效果.
关键字:Keil C51 小数运算
引用地址:
Keil C51下快速小数运算算法
推荐阅读最新更新时间:2024-03-16 15:21
如何使用keil 5 编写 51单片机 工程
目前我们通常编写51程序使用的是keil4,而好多编写STM32等单片机程序的使用keil5。那么如何在keil5中兼容51和STM32程序编写,省去切换版本的繁琐呢? 很简单只需两步就可以完成。 下面这个方法针对已破解keil5的stm32等一系列。这个肯定是最常见的,因为破解keil5然后编写32工程的教程一大把。 1、首先下载编写51的相关东西。 可以在官网上下载,例如百度keil官网。 点击右上角的DOWNLOADS进入相关的下载界面,下载你需要的安装包。MDK5、c51、c166等,这里只需要下载c51的。 当然也可以在下面这个百度网盘链接里下载: 51相关: 链接:http://pan.baid
[单片机]
学C51的基础 9 《 指针、结构、联合和枚举 》
指针、结构、联合和枚举 本节专门对第二节曾讲述过的指针作一详述。并介绍Turbo C新的数据类型: 结构、联合和枚举, 其中结构和联合是以前讲过的 五种基本数据类型(整型、浮点型、字符型、指针型和无值型)的组合。枚举是一个被命名为整型常数的集合。最后对类型说明 (typedef)和预处理指令作一阐述。 指 针(point) 学习Turbo C语言, 如果你不能用指针编写有效、正确和灵活的程序, 可以认为你没有学好C语言。指针、地址、数组及其相互 关系是C语言中最有特色的部分。规范地使用指针,可以使程序达到简单明了, 因此, 我们不但要学会如何正确地使
[单片机]
Keil5 STM32L151pack包的安装
最近工作上需要使用到stm32l151这个芯片,原因嘛就是因为这个芯片成本低,某宝售价2元钱,但是原本使用keil5没有这个芯片的型号选择,于是在官网http://www.keil.com/dd2/pack/#/eula-container 找到了这个系列的pack包, 点击下载即可 下载完成后直接点击安装 然后重启keil软件就会找到需要的那个芯片型号了 刚碰到这个问题的时候在网上找了好久都没有解决,后来去官网上看看文档,才发现这么简单啊,所有啊 !这是一个教训,网上的一些乱七八糟的不能全信,可能会走弯路的。开发的道路坑多慎行啊!!!
[单片机]
在KEIL中的模块化程序写法
在使用KEIL的时候,我们习惯上在一个.c的文件中把自己要写的东西按照自己思路的顺序进行顺序书写。这样是很普遍的写法,当程序比较短的时候比如几十行或者一百多行,是没有什么问题的。但是当程序很长的时候,比如你要用到LCD显示数据,就有几个LCD相关的函数,然后你想在LCD上显示温度,那么就要有DS18B20相关的操作,这又有几个相关的函数,如果你还想加上去DS1302的时间显示功能,那么又要多很多函数。这样的话一个程序下来几百行是很正常的事情,对于自己写的程序可能在自己的脑海中比较清晰,不会太乱,但是当把自己写的程序交给别人来看的时候,别人往往会看的云里雾里,经常会看着看着就不知道你写的是什么了。 如果大家写过类似电子钟这样
[单片机]
MSP432(Keil5)——9.ADC驱动
本次例程驱动了板载的ADC来读取一个模拟的角度传感器,大家可以在ADC读取中断里面换成其他的计算,具体引脚见程序代码。 adc.c #include adc.h void ADC_Init(void) { /* Initializing ADC (MCLK/1/4) */ MAP_ADC14_enableModule(); MAP_ADC14_initModule(ADC_CLOCKSOURCE_MCLK, ADC_PREDIVIDER_1, ADC_DIVIDER_4, 0); /* Configuring GPIOs (5.5 A0) */ MAP_GPIO_setAsPeripher
[单片机]
C51学习心得体会,产生随机数
(1)rand()函数产生一个0到RAND_MAX之间的整数,产生的是伪随机数,RAND_MAX =32767, i=rand(); i=1+rand()%data//产生1—data之间的随机数 (2)srand()函数产生不同的随机数序列,unsigned类型整数作为参数为函数rand设置随机数种子,种子不同,rand()函数得到随机数就不同。 srand(unsigned int i); e.g. #include stdlib.h #include stdio.h main() { int i; unsinged seed; printf( Enter seed: ); s CAN f(seed);//键盘输入种子
[单片机]
c51写的日历时钟程序2
********************************************************************************************************* * FORMAT CURRENT TIME INTO STRING * * Description : Formats the current time into an ASCII string. * Arguments : mode is the format type: * 1 will format the time as HH:MM:SS
[单片机]
Keil模式设置和编程事项
深入理解并应用C51对标准ANSIC的扩展是学习C51的关键之一。因为大多数扩展功能都是直接针对8051系列CPU硬件的。大致有以下8类: 8051存储类型及存储区域 , 存储模式 , 存储器类型声明 , 变量类型声明 , 位变量与位寻址 ,特殊功能寄存器(SFR) ,C51指针 l 函数属性 具体说明如下(8031为缺省CPU)。 第一节 Keil C51扩展关键字 C51 V4.0版本有以下扩展关键字(共19个): _at_ idata sfr16 alien interrupt small bdata large _task_ Code bit pdata using reentrant xdata compact
[单片机]