您的位置 首页 软件

C51编程经历三则

在单片机的开发应用中,已逐渐开始引入高级语言,C语言就是其中的一种。用惯了汇编的人,总觉得高级语言可控性不好,不如汇编那样随心所欲

单片机的开发使用中,已逐步开端引进高档言语,C言语便是其间的一种。用惯了汇编的人,总觉得高档言语“可控性”欠好,不如汇编那样为所欲为。以下是笔者在C51编程中的几点经历,期望对初学C51者有所协助。
  一、C51热发动代码的编制
  工业操控核算机,往往设有看门狗电路,看门狗动作,核算机复位,这便是热发动。热发动时,一般不允许程序从头开端,由于这将使丈量或核算值复位,导致体系作业反常。故程序有必要判别是热发动仍是冷发动。常用的办法是:设定某内存单位为标志位(如0x7f位和0x7e位),发动时首先读该内存单元的内容,假如它等于一个特定的值(例如两个内存单元的都是0xaa),就认为是热发动,不然便是冷发动,程序履行初始化部分,并将0xaa赋予这两个内存单元。
  依据以上的规划思路,编程时,设置一个指针,指向特定的内存单元如0x7f,然后在程序中依据特定内存单元的值判别冷/热发动,程序如下:
  void main()
  {  char data*HotPoint=(char*)0x7f;
    if((*HotPoint==0xaa)&&(*(--Hot
    Point)==0xaa))
     {     /*热发动的处理   */
    }
    else
     {  HotPoint=0x7e; /*冷发动的处理
       *HotPoint-0xaa;
       *(++HotPoint)=0xaa;
     }
    /*正常作业代码*/
  }
  实践调试中发现,无论是热发动仍是冷发动,开机后一切内存单元的值都被复位为0,完结不了热发动的要求。这是为什么呢?本来,用C言语编程时,开机时履行的代码并非是从main()函数的榜首句子开端的,在main()函数的榜首句子履行前要先履行一段‘开端代码’。正是这段代码履行了内存清零的作业。C编译程序供给了这段开端代码的源程序,名为CSTARTUPA51,翻开这个文件,能够看到如下代码:
  IDATALEN EQU 8011 the length of IDATA memory m bytes
  STARTUP1:
  IF IDATALEN<>0
          MOV R0,#IDATALEN-I
          CLR A
  IDATALOOP:  MOV @R0,A
          DJNZ R0,IDATALOOP
  ENDIF
  可见,在履行到判别是否热发动的代码之前,开端代码已将一切内存单元清零。怎么处理这个问题呢?好在开端代码是能够更改的,办法是:修正startup.a51源文件,然后用编译程序所顺便的a51.exe程序对startup.a51编译,得到startup.obj文件,用这段代码替代本来的开端代码。详细步骤是(设C源程序名为HOTSTARTC):
  1修正startup.a51源文件(这个文件在C51LIB目录下)。
  2履行如下指令:
  A51 startup.a51得到startup.obj文件。将此文件拷入HOTSTARTC地点目录。
  3将编好的C源程序用C51EXE编译好,得到方针文件HOTSTARTOBJ。
  4用L51 HOTSTART,STARTUPOBJ指令衔接,得到肯定方针文件HOTSTART。
  5用OHS51 HOTSTART得到HOTSTARTHEX文件,即可完结发动代码的修正。
  关于startup.a51的修正,依据自己的需求进行,如将IDATALEN EQU 80H中的80H改为70H,就能够使6F到7F的16字节内存不被清零。
  二、直接调用EPROM中已固化的程序
  笔者用的仿真机,由6位数码管显现,在DE00H处寄存显现子程序,只需将显现的数存入显现缓冲区,然后调用显现子程序就能够了,汇编指令为:
  LCALL 0DE00H
  在用C言语编程时,怎么完结这一功用呢?C言语中有指向函数的指针这一概念,能够用来完结用函数指针调用函数。指向函数的指针变量的界说格局为:
  类型标识符(*指针变量名)();
  在界说好指针后就能够给指针变量赋值,使其指向某个函数的开端地址,然后用(*指针变量名)()即可调用这个函数。程序如下例:
  void main(void)
  {
     void (*DispBuffer)();/*界说指向函数指针*/
    DispBuffer=0xde00; /*赋值*/
    for(;;)
    { Key();
      DispBuffer();
    }
  }
  三、将浮点数转化为字符数组
  笔者在编制使用程序时有这样的要求:将运算的成果(浮点数)存入E2PROM中。咱们知道,浮点数在C言语中是以IEEE格局存储的,一个浮点数占四个字节。例如浮点数34526存为160、26、10、66四个数。要将该浮点数存入E2PROM,实践上便是要存这四个数。怎么在程序中得到一个浮点数的组成数呢?
  浮点数在存储时,是存储在接连的字节中的,只需设法找到存储方位,就能够得到这些数了。能够界说一个void指针,将此指针指向需求存储的浮点数,然后再将此指针强制转化为char型。这样,使用指针就能够得到组成该浮点数的各个字节的值了。详细程序如下:
#define uchar unsigned char
#define uint unsigned int
void FtoC(void)
{  float a;
  uchar I,*px
  uchar x[4];/*界说字符数组,预备存储浮点数的四个字节*/
  void *pf;
  px=x; /*px指针指向数组x*/
  pf=&a;/*void型指针指向浮点数首地址*/
  a=34.526;
  for(I=0;I<4;I++)
  { *(px+I)=*((char *)pf+I);/*强制void型指针转成char型,由于void型指针不能运算*/
  }      
}
  假如已将数存入E2PROM,要将其取出兼并,办法也是相同,可参阅下面的程序。
#define uchar unsigned char
#define uint unsigned int
void CtoF(void)
{  float a;
   uchar I,*px
   uchar x[4]-{56,180,150,73};
   void *pf;
   px=x;
   pf=&a;
   for(I=o;I<4;I++)
   { *((char *)pf+I)=*(px+I)
  }
}
  以上程序所用C言语为FRANKLIN C51 VER 32。

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/qianrushi/ruanjian/261110.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部