C++ Typedef和define的用法

Define

定义:

define,宏定义,C语言中预处理命令一种。分为无参宏定义和带参宏定义。

  1. 无参宏定义的一般形式为:#define 宏名 字符串
  2. 带参宏定义的一般形式为:#define 宏名(参数表) 字符串

#define 命令是 C 语言中的一个宏定义命令 ,它用来将一个标识符定义为一个字符串 , 该标识符被称为宏名, 被定义的字符串称为替换文本。 [1]  该命令有两种格式:一种是简单的宏定义, 另一种是带参数的宏定义。

分类:

无参宏定义:

C语言中无参宏定义的一般形式为:#define 宏名 字符串 

其中的“#”表示这是一条预处理命令。凡是以“#”开头的均为预处理命令。“define”为宏定义命令。“标识符”为所定义的宏名。“字符串”可以是常数、表达式、格式串等。

例如: 

#define M (a+b) 

它的作用是指定标识符M来代替表达式(a+b)。在编写源程序时,所有的(a+b)都可由M代替,而对源程序作编译时,将先由预处理程序进行宏代换,即用(a+b)表达式去置换所有的宏名M,然后再进行编译。

有参宏定义:

C语言允许宏带有参数。在宏定义中的参数称为形式参数,在宏调用中的参数称为实际参数。对带参数的宏,在调用中,不仅要宏展开,而且要用实参去代换形参

带参宏定义的一般形式为:  #define 宏名(形参表)字符串 [2] 

在字符串中含有各个形参。 带参宏调用的一般形式为:  宏名(形参表)

例如:

#define M(y) ((y)*(y)+3*(y))

k=M*5

在宏调用时,用实参5去代替形参y,经预处理宏展开后的语句为:  k=5*5+3*5

注意:

1.宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名,这只是一种简单的代换,字符串中可以含任何字符,可以是常数,也可以是表达式,预处理程序对它不作任何检查。如有错误,只能在编译已被宏展开后的源程序时发现。

2.宏定义不是说明或语句,在行末不必加分号,如加上分号则连分号也一起置换。

3.宏定义其作用域为宏定义命令起到源程序结束。如要终止其作用域可使用#undef命令。

条件编译:

一般情况下,源程序中所有的行都参加编译,但是有时希望只对其中一部分内容在满足一定条件时才进行编译,这时就需要使用到一些条件编译命令。而在#if条件编译命令中,需要判断符号常量所定义的具体值,但有时并不需要判断具体值,只需要知道这个符号常量是否被定义了。这时就不需要使用#if,而采用#ifdef和#ifndef,分别表示“如果有定义”及“如果无定义”。同时,如果我们需要删除事先定义的宏定义,可以使用#undef命令。 [2] 

#ifdef

如果有定义的#ifdef的一般形式为:#ifdef 宏替换名 语句段 #endif

含义为:如果宏替换名已被定义过,则对“语句段”进行编译;如果未定义#ifdef后面的宏替换名,则不对语句段进行编译。

#ifndef

如果无定义的#ifndef的一般形式为:#ifndef 宏替换名 语句段 #endif

含义为:如果未定义#ifndef后面的宏替换名,则对“语句段1”进行编译;如果定义#ifndef后面的宏替换名,则不执行语句段。

#undef

#undef命令可以删除事先定义了的宏定义。

#undef命令的一般形式如下:#undef 宏替换名

例子:

#define MAX_SIZE 100

char array[MAX_SIZE];

#undef   MAX_SIZE

上述代码中,首先使用#define定义标识符MAX_SIZE,直到遇到#undef语句之前,MAX_SIZE的定义都是有效的。

说明:#undef的主要目的是将宏名局限在仅需要它们的代码段中。


 

Typedef

定义:

typedef是在计算机编程语言中用来为复杂的声明定义简单的别名,它与宏定义有些差异。它本身是一种存储类的关键字,与auto、extern、mutable、static、register等关键字不能出现在同一个表达式中。

用法:

使用typedef为现有类型创建别名,定义易于记忆的类型名

typedef int size;

size len = file.getLength();

此处size代表int类型

typedef char Line[81];

Line line;

此处Line代表具有81个元素的字符数组

typedef char* pstr

int mystrcmp(const pstr pq,const pstr p3)

此处pstr代表指向字符类型变量的指针

typedef的特殊用法:

1.typedef与结构体结合使用

typedef struct tagMyStruct

{

int iNum;

long lLength;

}MyStruct;

分析:

tagMyStruct称为“tag”,即“标签”,实际上是一个临时名字,struct关键字和tagMyStruct一起,构成了这个结构类型,不论是否有typedef,这个结构都存在。

我们可以用struct tagMyStruct varName来定义变量,但要注意,使用tagMyStruct varName来定义变量是不对的,因为struct 和tagMyStruct合在一起才能表示一个结构类型。

typedef为这个新的结构起了一个名字,叫MyStruct。

typedef struct tagMyStruct MyStruct;

因此,MyStruct实际上相当于struct tagMyStruct,我们可以使用MyStruct varName来定义变量。

复杂的变量声明

下面是三个变量的声明,我想使用typdef分别给它们定义一个别名,请问该如何做?

int *(*a[5])(int, char*);

void (*b[10]) (void (*)());

double(* (*pa)[9])();

分析:

对复杂变量建立一个类型别名的方法很简单,你只要在传统的变量声明表达式里用类型名替代变量名,然后把关键字typedef加在该语句的开头就行了。

1:int *(*a[5])(int, char*);

//pFun是我们建的一个类型别名

typedef int *(*pFun)(int, char*);

//使用定义的新类型来声明对象,等价于int* (*a[5])(int, char*);

pFun a[5];

>2:void (*b[10]) (void (*)());

//首先为上面表达式加粗部分声明一个新类型

typedef void (*pFunParam)();

//整体声明一个新类型

typedef void (*pFun)(pFunParam);

//使用定义的新类型来声明对象,等价于void (*b[10]) (void (*)());

pFun b[10];

>3. double(* [1]  (*pa)[9])() [2]  ;

//首先为上面表达式蓝色部分声明一个新类型

typedef double(*pFun)();

//整体声明一个新类型

typedef pFun (*pFunParam)[9];

//使用定义的新类型来声明对象,等价于double(*(*pa)[9])();

pFunParam pa;

pa是一个指针,指针指向一个数组,这个数组有9个元素,每一个元素都是“doube(*)()”--也即一个指针,指向一个函数,函数参数为空,返回值是“double”。


Define和Typedef的区别:

typedef char* pStr1;

#define pStr2 char* 

pStr1 s1,s2;

pStr2 s3,s4;

在上述的变量定义中,s1、s2、s3都被定义为char *,而s4则定义成了char,不是我们所预期的指针变量,根本原因就在于#define只是简单的字符串替换而typedef则是为一个类型起新名字。

上例中define语句必须写成 pStr2 s3, *s4; 这样才能正常执行。

#define宏定义有一个特别的长处:可以使用 #ifdef ,#ifndef等来进行逻辑判断,还可以使用#undef来取消定义。

 typedef也有一个特别的长处:它符合范围规则,使用typedef定义的变量类型其作用范围限制在所定义的函数或者文件内(取决于此变量定义的位置),而宏定义则没有这种特性。

主要资料来源百度百科:

  1. https://baike.baidu.com/item/typedef/9558154?fr=aladdin
  2. https://baike.baidu.com/item/define/577777?fr=aladdin
发布了17 篇原创文章 · 获赞 9 · 访问量 8660

猜你喜欢

转载自blog.csdn.net/qq_39025293/article/details/104415336