C 防止 全局 变量 重复 定义 巧妙办法

 

C 防止  全局 变量 重复 定义 巧妙办法

C语言中使用extern 关键字来定义全局变量的时候,我们需要在.h文件和.c文件中重复定义,这种重复,导致了出错几率的增加。

研读了uCOSii操作系统的部分代码,后发现了一种非常巧妙的方法,可以称得上是“奇淫巧计”了。

 

 

ucos_ii.h中有如下定义

#ifdef    OS_GLOBALS

#define   OS_EXT

#else

#define   OS_EXT   extern

#endif

 

 

 

在之后使用OS_EXT来定义全局变量。以下是摘抄的一小部分

 

OS_EXT   INT32U             OSCtxSwCtr;                /* Counter of number of context switches            */

 

#if OS_EVENT_EN && (OS_MAX_EVENTS > 0)

OS_EXT   OS_EVENT          *OSEventFreeList;           /* Pointer to list of free EVENT control blocks     */

OS_EXT   OS_EVENT           OSEventTbl[OS_MAX_EVENTS];/* Table of EVENT control blocks                    */

#endif

 

#if (OS_VERSION >= 251) && (OS_FLAG_EN > 0) && (OS_MAX_FLAGS > 0)

OS_EXT   OS_FLAG_GRP        OSFlagTbl[OS_MAX_FLAGS];   /* Table containing event flag groups               */

OS_EXT   OS_FLAG_GRP       *OSFlagFreeList;            /* Pointer to free list of event flag groups        */

#endif

 

#if OS_TASK_STAT_EN > 0

OS_EXT   INT8S              OSCPUUsage;                /* Percentage of CPU used                           */

OS_EXT   INT32U             OSIdleCtrMax;              /* Max. value that idle ctr can take in 1 sec.      */

OS_EXT   INT32U             OSIdleCtrRun;              /* Val. reached by idle ctr at run time in 1 sec.   */

OS_EXT   BOOLEAN            OSStatRdy;                 /* Flag indicating that the statistic task is rdy   */

OS_EXT   OS_STK             OSTaskStatStk[OS_TASK_STAT_STK_SIZE];       /* Statistics task stack           */

#endif

 

 

如何使用这些全局变量呢?

在需要使用这些全局变量的地方我们引用头文件,并且#define OS_GOLBALS

这样,在编译这个.c文件的时候。编译器给每个全局变量分配内存空间,而当编译器处理其他.C文件时,OS_GLOBAL没有定义,OS_EXT被定义为extern,这样用户就可以调用外部全局变量。

例如在os_core.c开头宏定义处有如下定义:

#ifndef   OS_MASTER_FILE

#define   OS_GLOBALS

#include "ucos_ii.h"

#endif

 

之后,在os_core.c中就可以任意的使用那些全局变量了。

例如:

void   OSIntExit (void)

{

#if OS_CRITICAL_METHOD == 3                                 /* Allocate storage for CPU status register */

     OS_CPU_SR   cpu_sr = 0;

#endif

 

 

 

     if (OSRunning == TRUE) {

         OS_ENTER_CRITICAL();

         if (OSIntNesting > 0) {                             /* Prevent OSIntNesting from wrapping        */

             OSIntNesting--;

         }

         if (OSIntNesting == 0) {                            /* Reschedule only if all ISRs complete ... */

             if (OSLockNesting == 0) {                       /* ... and not locked.                       */

                 OS_SchedNew();

                 if (OSPrioHighRdy != OSPrioCur) {           /* No Ctx Sw if current task is highest rdy */

                     OSTCBHighRdy   = OSTCBPrioTbl[OSPrioHighRdy];

#if OS_TASK_PROFILE_EN > 0

                     OSTCBHighRdy->OSTCBCtxSwCtr++;          /* Inc. # of context switches to this task   */

#endif

                     OSCtxSwCtr++;                           /* Keep track of the number of ctx switches */

                     OSIntCtxSw();                           /* Perform interrupt level ctx switch        */

                 }

             }

         }

         OS_EXIT_CRITICAL();

     }

}

 

其中的 OSRunning OSIntNesting OSLockNesting OSPrioHighRdy OSPrioCur OSCtxSwCtr这些变量都是定义在ucos_ii.h中的全局变量。

 

好吧,假设到这里我和你都学会这个奇淫巧计了,我们开始尝试一下下。

Includes.h文件

#ifdef TEST_GLOBALS

#define TEST_EXT

#else

#define TEST_EXT extern

#endif

 

TEST_EXT int a;

extern int b;

 

Main.c文件

#define TEST_GLOBALS //define the Marco to use those global variables

#include "includes.h" //include the header file

#include <stdio.h>   //for printf

int b; //variables b is define by extern

void main()

{

printf("%d",b);

printf("%d",a);

}

这里的a不需要重新定义,而b变量需要定义。

1.03全局变量

    以下是如何定义全局变量。众所周知,全局变量应该是得到内存分配且可以被其他模块通过C语言中extern关键字调用的变量。因此,必须在 .C .H 文件中定义。这种重复的定义很容易导致错误。以下讨论的方法只需用在头文件中定义一次。虽然有点不易懂,但用户一旦掌握,使用起来却很灵活。表1.2中的定义出现在定义所有全局变量的.H头文件中。

 

程序清单 L 1.2  定义全局宏。

#ifdef   xxx_GLOBALS

#define  xxx_EXT

#else

#define  xxx_EXT extern

#endif

   

.H 文件中每个全局变量都加上了xxx_EXT的前缀。xxx代表模块的名字。该模块的.C文件中有以下定义:

 

#define  xxx_GLOBALS

#include "includes.h"

   

当编译器处理.C文件时,它强制xxx_EXT(在相应.H文件中可以找到)为空,(因为xxx_GLOBALS已经定义)。所以编译器给每个全局变量分配内存空间,而当编译器处理其他.C文件时,xxx_GLOBAL没有定义,xxx_EXT被定义为extern,这样用户就可以调用外部全局变量。为了说明这个概念,可以参见uC/OS_II.H,其中包括以下定义:

 

#ifdef   OS_GLOBALS

#define  OS_EXT

#else

#define  OS_EXT extern

#endif

 

OS_EXT  INT32U       OSIdleCtr;

OS_EXT  INT32U       OSIdleCtrRun;

OS_EXT  INT32U       OSIdleCtrMax;

 

同时,uCOS_II.H有中以下定义:

 

#define  OS_GLOBALS

#include “includes.h”

 

当编译器处理uCOS_II.C时,它使得头文件变成如下所示,因为OS_EXT被设置为空。

 

INT32U       OSIdleCtr;

INT32U       OSIdleCtrRun;

INT32U       OSIdleCtrMax;

   

这样编译器就会将这些全局变量分配在内存中。当编译器处理其他.C文件时,头文件变成了如下的样子,因为OS_GLOBAL没有定义,所以OS_EXT被定义为extern

 

extern INT32U       OSIdleCtr;

extern INT32U       OSIdleCtrRun;

extern INT32U       OSIdleCtrMax;

   

在这种情况下,不产生内存分配,而任何 .C文件都可以使用这些变量。这样的就只需在 .H 文件中定义一次就可以了。

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值