35.【C/C++ 枚举(bool)类型和宏定义 (超详细)】(一)

简介: 35.【C/C++ 枚举(bool)类型和宏定义 (超详细)】

(一)、宏定义

1.什么是宏定义?

1)#define 叫做宏定义命令它也是C语言预处理命令的一种,所谓宏定义,就是用一个标识符来表示一个字符串。如果在后面的代码中出现了该标识符,那么就全部替换成指定的字符串。

2)#define N 100 就是宏定义,N为宏名100是宏的内容(宏所表示的字符串)。在预处理阶段,对程序中所有出现的“宏名”,预处理器都会用宏定义中的字符串区代换,这称为“宏替换”或“宏展开”

宏定义是由源程序中的宏定义命令** define完成**的,宏替换是由预处理程序完成的。

2.宏定义的格式:

#define        宏名 字符串

1)#表示这是一条预处理命令,所有的预处理命令都以 # 开头宏名是标识符的一种,命名规则和变量相同。字符串可以是数字、表达式、if语句、函数等。

2)这里所说的字符串是一般意义上的字符序列,不要和C语言中的字符串等同,它不需要双引号

3)程序中反复使用的表达式就可以使用宏定义(替换方便)

3.宏定义注意事项和细节

0)宏定义实质:只替换,不计算

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

2)宏定义不是说明或语句,在行末不必加分号如果加上分号则连分号一起替换

代码演示:
#include <iostream>
using namespace std;
#define PI 3.14;      //设置竑名的值位3.14
int main()
{
  cout << PI    //输出3.14
  cout << PI << endl;   //会报错,
  return 0;
}
效果展示:

3)宏定义必须写在函数之外,其作用域为宏定义命令起到源程序结束。如要终止其作用域可使用#undef命令

代码演示:
#include <iostream>
using namespace std;
#define PI 3.14       //设置竑名的值位3.14
int main()
{
  cout << PI << endl;   //输出3.14
  #undef PI    //取消宏定义 PI 
  //cout << PI << endl;   //会报错,显示未识别标识符PI
  return 0;
}
效果展示:

4)代码中的宏名如果被引号包围,那么预处理程序不对其作宏代替

代码展示:
#include <iostream>
using namespace std;
#define PI 3.14       //设置竑名的值位3.14
int main()
{
  cout << "PI=" << PI << endl;   //输出3.14
  return 0;
}
效果展示:

5)宏定义允许嵌套,在宏定义的字符串中可以使用已经定义的宏名,在宏展开时由预处理程序层层代换

代码展示:
#include <iostream>
using namespace std;
#define PI 3.14      
#define S  PI*2        //  嵌套PI
int main()
{
  cout << "PI=" << PI << endl;   //输出3.14
  cout << "S=" << S << endl;
  return 0;
}
效果展示:

6)习惯上宏名用大写字母表示,以便于与变量区别。但也允许用小写字母

7)可用宏定义表示数据类型,使书写方便

代码展示:
#include <iostream>
using namespace std;
#define PI int      
int main()
{
  PI a = 3;
  cout << "a=" << a << endl;
  return 0;
}
效果展示:

8.typedef与define 的区别

8)宏定义表示数据类型和用typedef定义数据说明符的区别宏定义只是简单的字符串替换,由预处理器来处理;而typedef 是在编译阶段由编译器处理的它并不是简单的字符串替换,而给原有的数据类型起一个新的名字,将它作为一种新的数据类型。

typedef定义格式:

typedef  替换目标   代替者
代码展示:
#include <iostream>
using namespace std;
#define PI int      
int main()
{
  typedef double s;     //定义
  PI a = 3;
  cout << "a=" << a << endl;
  s  b = 2.3;
  cout << "b=" << b << endl;
  return 0;
}
效果展示:


相关文章
|
3月前
|
存储 编译器 程序员
C++类型参数化
【10月更文挑战第1天】在 C++ 中,模板是实现类型参数化的主要工具,用于编写能处理多种数据类型的代码。模板分为函数模板和类模板。函数模板以 `template` 关键字定义,允许使用任意类型参数 `T`,并在调用时自动推导具体类型。类模板则定义泛型类,如动态数组,可在实例化时指定具体类型。模板还支持特化,为特定类型提供定制实现。模板在编译时实例化,需放置在头文件中以确保编译器可见。
41 11
|
4月前
|
安全 程序员 C语言
C++(四)类型强转
本文详细介绍了C++中的四种类型强制转换:`static_cast`、`reinterpret_cast`、`const_cast`和`dynamic_cast`。每种转换都有其特定用途和适用场景,如`static_cast`用于相关类型间的显式转换,`reinterpret_cast`用于低层内存布局操作,`const_cast`用于添加或移除`const`限定符,而`dynamic_cast`则用于运行时的类型检查和转换。通过具体示例展示了如何正确使用这四种转换操作符,帮助开发者更好地理解和掌握C++中的类型转换机制。
|
5月前
|
C++
使用 QML 类型系统注册 C++ 类型
使用 QML 类型系统注册 C++ 类型
113 0
|
6月前
|
编译器 C++ 运维
开发与运维函数问题之函数的返回类型如何解决
开发与运维函数问题之函数的返回类型如何解决
43 6
|
5月前
|
存储 C++
【C/C++学习笔记】string 类型的输入操作符和 getline 函数分别如何处理空白字符
【C/C++学习笔记】string 类型的输入操作符和 getline 函数分别如何处理空白字符
57 0
|
5月前
|
设计模式 安全 IDE
C++从静态类型到单例模式
C++从静态类型到单例模式
43 0
|
6月前
|
编译器 C++
C++从遗忘到入门问题之C++中的浮点数类型问题如何解决
C++从遗忘到入门问题之C++中的浮点数类型问题如何解决
|
2月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
63 2
|
2月前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
113 5
|
2月前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
112 4