2.5 条件编译
在编译一个程序的时候我们如果要将一条语句(一组语句)编译或者放弃是很方便的。因为我们有条件编译指令。
比如说:
调试性的代码,删除可惜,保留又碍事,所以我们可以选择性的编译。
我们来举个栗子:
上述代码的意思是:如果已经定义了MAX,那么就打印出hehe,否则就打印出haha,
#ifdef MAX还可以换一种表达: #if defined(MAX)
这种方式只与你是否定义了MAX有关,不关心MAX的值,区别一下下面的代码:
这个的意思是如果1,1为真就执行下面的语句,与我们用的if else 语句有点儿类似。
与此类似:#ifndef MAX == #if !defined(MAX)
小结:
1. #if 常量表达式 //... #endif //常量表达式由预处理器求值。 如: #define __DEBUG__ 1 #if __DEBUG__ //.. #endif 2.多个分支的条件编译 #if 常量表达式 //... #elif 常量表达式 //... #else //... #endif 3.判断是否被定义 #if defined(symbol) #ifdef symbol #if !defined(symbol) #ifndef symbol 4.嵌套指令 #if defined(OS_UNIX) #ifdef OPTION1 unix_version_option1(); #endif #ifdef OPTION2 unix_version_option2(); #endif #elif defined(OS_MSDOS) #ifdef OPTION2 msdos_version_option2(); #endif #endif
那上面的#if( ) 与if有什么区别吗?
#if( ) 在预处理的时候就会被处理,如果( )中值为真,后面表达式就保留,否则直接去掉后面表达式,而if( )后面表达式的值在预处理是不会直接去除的,只会在执行的时候选择性的执行。
2.6 文件包含
我们已经知道, #include 指令可以使另外一个文件被编译,就像它实际出现于 #include 指令的地方一样。
这种替换的方式很简单:预处理器先删除这条指令,并用包含文件的内容替换。这样一个源文件被包含 10 次,那就实际被编译 10 次。
2.6.1 头文件被包含的方式
本地文件包含:
#include "filename"
查找策略:先在源文件所在目录下查找,如果该头文件未找到,编译器就像查找库函数头文件一样在标 准位置查找头文件。 如果找不到就提示编译错误。
Linux环境的标准头文件的路径:
/usr/include
VS环境的标准头文件的路径:
C:\Program Files (x86)\Microsoft Visual Studio 12.0\VC\include //这是VS2013的默认路径
库文件包含 :
#include <filename.h>
查找头文件直接去标准路径下去查找,如果找不到就提示编译错误。
这样是不是可以说,对于库文件也可以使用 “” 的形式包含?
答案是肯定的, 可以 ,但是这样做查找的效率就低些,当然这样也不容易区分是库文件还是本地文件了。
2.6.2 嵌套文件包含
如果出现这样的场景:
如何解决这个问题?
答案:条件编译。
每个头文件的开头写:
#ifndef __TEST_H__ #define __TEST_H__ //头文件的内容 #endif //__TEST_H__
或者:
#pragma once
就可以避免头文件的重复引入。
面试题:
1. 头文件中的 ifndef / define / endif 是干什么用的 ?
2. #include < filename . h > 和 #include "filename.h" 有什么区别 ?
回答:
1 防止头文件被重复包含。
2 #include "filename.h" 先去用户的工作目录下去搜索,没有找到再去库函数的标准目录下搜索,还没有找到就报错;而#include <filename.h>直接去库函数的标准目录下搜索,没有找到就报错。