宏(使用注意事项、主要用途)------c++程序设计原理与实践(进阶篇)
使用宏的时候一定要小心:在c中没有真正有效的方法来避免使用宏,但宏带有严重的副作用,因为宏不遵守通常的c(或c++)作用域和类型规则——它只是一种文本替换。
宏的使用注意事项:
- 所以宏名全部大写。
- 不是宏的结构不要使用全部大写的名字。
- 不要为宏取短的或“有趣”的名字,如max或min。
- 期望其他人也遵守上面简单而常见的规范。
宏的主要用途:
- 定义“常量”。
- 定义类似函数的结构。
- “改进”语法。
- 控制条件编译。
- 其他不常见用途。
我们认为宏被过度使用了,但在c++程序中有时很难避免使用宏(特别是当编写的程序需要移植到很老的编译器上或者有特殊限制的平台上,在c程序中还没有一种合理而完整的替代方法。)。
下面介绍一些有关的宏技术。
类函数宏:
下面是一个非常典型的类函数宏:
#define MAX(x,y)((x)>=(y)?(x):(y))
我们为宏取名为全大写字母的MAX,以便与常用的函数名max区别开来,显然,它与函数还是有很大区别的:没有参数类型、没有语句块、没有返回值语句等。另外,宏定义中的那些括号是起什么作用呢?考虑如下代码:
int aa=MAX(1,2);
double dd=MAX(aa++,2);
char cc=MAX(dd,aa)+2;
宏替换后,程序扩展为:
int aa=((1)>=(2)?(1):(2));
double dd=((aa++)>=(2)?(aa)++:(2));
char cc=((dd)>=(aa)?(dd):(aa))+2;
在宏的定义中,使用任何参数时都应将其置于括号之中(当作表达式)。如果在宏定义中没有使用“那些括号”。最后一条语句会扩展为:
char cc=dd>=aa?dd:aa+2;
也就是说,cc的值将和你根据其定义推断出的值不同。
另一方面,对于第二条语句,使用再多括号耶解决不了问题。宏参数x被替换为aa++,由于x在MAX使用了两次,因此x进行了两次增1运算。注意,不要向宏传递可能引起副作用的参数。
如何取消定义宏:
#undef MAX(x,y);
并不是所以宏参数都被用作表达式。例如:
#define ALLOC(T,n) ((T*)malloc(sizeof(T)*n)); //malloc 为c中的函数 类似c++的new
这是来自实际程序中的例子,内存分配时sizeof中使用的类型与所需类型可能不匹配,这个宏对避免此类错误很有用:
double* p= malloc(sizeof(int)*10); //可能错误
如果还希望宏能捕获内存耗尽的错误:假如已经定义了error_var和error(),可以这样定义宏:
#define ALLOC(T,n) (errpr_var=(T*)malloc(sizeof(T)*n),\
(error_var==0)\
?(error("memory allocation failure"),0)\
:error_var)
行结尾出的 \ 并非输入错误,将一个较长的宏分成几行,就必须在非结尾行的最后加上 \ 。
语法宏:
你可以定义这样一类宏,它们能使源程序形式上更符合你的偏好,例如:
#define forever for(;;)
#define CASE break;case
#define begin {
#define end }
但不建议使用。
条件编译:
假设某个头文件有两个版本,比如说一个是Linux版,另一个是Windows版。在程序中你如何选择使用哪个版本呢?常用方法如下:
#ifdef WINDOWS
#lnclude"my_windows_header.h"
#else
#include"my_linux_header.h"
#endif
现在,如果有人在编译之前定义了宏 WINDOWS,则效果为:
#lnclude"my_windows_header.h"
否则,效果为:
#include"my_linux_header.h"
#indef WINDOWS 并不关心 WINDOWS被定义成什么,它只关心WINDOWS是否被定义。
很多大型系统(包括所有操作系统)都会定义类似WINDOWS这样的宏,以供我们检验。
我们还可以这样来检查程序是在被c++编译器编译还是被c编译器编译:
#ifdef __cplusplus
//in c++
#else
//in c
#endif
还有一种类似的结构,通常被人们称为包含保护,常常用来防止头文件被保护多次:
#ifndef MY_WINDOWS_HEADER
#define MY_WINDOWS_HEADER
//下面是头文件内容
#endif
#ifndef检测宏是否未被定义,即它与#ifdef是相对的。逻辑上,用于源文件控制的宏与其他修改源码(宏替换)的宏有很大不同。他们只是使用了相同的下层语言机制。
c++程序设计原理与实践(进阶篇)
宏(使用注意事项、主要用途)------c++程序设计原理与实践(进阶篇)的更多相关文章
- 数值限制------c++程序设计原理与实践(进阶篇)
每种c++的实现都在<limits>.<climits>.<limits.h>和<float.h>中指明了内置类型的属性,因此程序员可以利用这些属性来检 ...
- 编码原则实例------c++程序设计原理与实践(进阶篇)
编码原则: 一般原则 预处理原则 命名和布局原则 类原则 函数和表达式原则 硬实时原则 关键系统原则 (硬实时原则.关键系统原则仅用于硬实时和关键系统程序设计) (严格原则都用一个大写字母R及其编号标 ...
- 函数形参为基类数组,实参为继承类数组,下存在的问题------c++程序设计原理与实践(进阶篇)
示例: #include<iostream> using namespace std; class A { public: int a; int b; A(int aa=1, int bb ...
- 函数返回值string与返回值bool区别------c++程序设计原理与实践(进阶篇)
为什么find_from_addr()和find_subject()如此不同?比如,find_from_addr()返回bool值,而find_subject()返回string.原因在于我们想说明: ...
- (c++11)随机数------c++程序设计原理与实践(进阶篇)
随机数既是一个实用工具,也是一个数学问题,它高度复杂,这与它在现实世界中的重要性是相匹配的.在此我们只讨论随机数哦最基本的内容,这些内容可用于简单的测试和仿真.在<random>中,标准库 ...
- 实现求解线性方程(矩阵、高斯消去法)------c++程序设计原理与实践(进阶篇)
步骤: 其中A是一个n*n的系数方阵 向量x和b分别是未知数和常量向量: 这个系统可能有0个.1个或者无穷多个解,这取决于系数矩阵A和向量b.求解线性系统的方法有很多,这里使用一种经典的方法——高斯消 ...
- 有符号数和无符号数------c++程序设计原理与实践(进阶篇)
有符号数与无符号数的程序设计原则: 当需要表示数值时,使用有符号数(如 int). 当需要表示位集合时,使用无符号数(如unsigned int). 有符号数和无符号数混合运算有可能会带来灾难性的后果 ...
- bitest(位集合)------c++程序设计原理与实践(进阶篇)
标准库模板类bitset是在<bitset>中定义的,它用于描述和处理二进制位集合.每个bitset的大小是固定的,在创建时指定: bitset<4> flags; bitse ...
- gets()scanf()有害------c++程序设计原理与实践(进阶篇)
最简单的读取字符串的方式是使用gets(),例如: char a[12]; gets(a); 但gets()和scanf()是有害的,曾经有大约1/4的成功黑客攻击是由于gets()和它的近亲scan ...
随机推荐
- crush class实验
标签(空格分隔): ceph,ceph实验,crushmap luminous版本的ceph新增了一个功能crush class,这个功能又可以称为磁盘智能分组.因为这个功能就是根据磁盘类型自动的进行 ...
- Vue源码学习(一):调试环境搭建
最近开始学习Vue源码,第一步就是要把调试环境搭好,这个过程遇到小坑着实费了点功夫,在这里记下来 一.调试环境搭建过程 1.安装node.js,具体不展开 2.下载vue项目源码,git或svn等均可 ...
- node.js中的buffer.fill
buffer.fill(value, [offset], [end]) 接收参数: value 将要填充的数据 offet 填充数据的开始位置,不指定默认为 0 ...
- 用一两句话说一下你对“盒模型”这个概念的理解,和它都涉及到哪些css属性
网页设计中常听的属性名:内容(content).填充(padding).边框(border).边界(margin), CSS盒子模式都具备这些属性. 这些属性我们可以用日常生活中的常见事物——盒子作一 ...
- CDM中,创建一个或多个组合属性的唯一约束
除主键外,有时还需要创建一个或多个组合字段的唯一约束,方法如下: 双击打开实体,在idntifier标签页中可看到默认主键的唯一约束,在其下方添加一条记录,然后双击该记录,打开约束设置窗口 在该窗口的 ...
- 第八章 数据库连接JDBC(待续)
············
- ocx控件针对网页刷新和关闭分别进行区分处理
当ocx加载在网页上时,如果对网页执行F5刷新事件,ocx控件会销毁ocx的窗口类,但是ocx的APP类是不会销毁的. 只有当网页被关闭时,才销毁app类. --------------------- ...
- SDL_AudioSpec(转)
转贴地址:http://www.cnblogs.com/nanguabing/archive/2012/04/12/2444084.html ============================= ...
- namespace mismatch错误处理方法
在基于axis2进行WebService开发部署时(使用Java客户端调用服务)可能会报namespace mismatch require http://ws.apache.org/axis2 fo ...
- Oracle字符集的查看查询和Oracle字符集的设置修改(转)
最近郁闷的字符集2014年7月31日16:32:58 本文主要讨论以下几个部分:如何查看查询oracle字符集. 修改设置字符集以及常见的oracle utf8字符集和oracle exp 字符集问题 ...