#include指令

  • #include <头文件.h>//在标准系统目录中寻找头文件
  • #include "头文件.h"//先在当前目录下面寻找头文件,然后在标准系统目录下寻找头文件

预处理器在碰到#inclide指令时,就会将后面附加的头文件的内容包含到当前文件中。

头文件

后缀为.h的文件是头文件。这类文件经常包含预处理器所需要的语句。

头文件内容最常见形式包括:

  • 明显常量
  • 宏函数
  • 函数声明
  • 结构模板定义
  • 类型定义

其他指令

#undef

#undef指令取消一个给定的已定义#define.

#define FF 10

#undef FF //FF定义被取消

条件编译

#ifdef #else #endif

#ifdef后面的标识符如果已经定义了,那么执行它的分支控制的指令,知道下一个#else#endif为止。如果有#else存在,那么在未定义时会执行#else知道#endif之间的所有指令。

#define FF 10

#ifdef FF
#define AA 100
#else
#define AA 1000
#endif int main() {
printf("FF :%d\n",AA);
}

#ifndef

#ifndef#ifdef的反义词,它是用来确定后面的表示符未被定义。通常用来定义此前未被定义的常量。

#ifndef FF
#define FF 199
#endif
int main() {
printf("FF :%d\n",FF);
}

#ifndef也可以用于防止多次包含同一文件。

/*
*假设这个头文件是test.h
*/
//原本头文件的内容直接写,现在我们把头文件的内容放到下面这样的结构里:
#ifndef _TEST_H
#define _TEST_H
/*
*#define和#endif之间存放头文件的所有内容
*/
#endif

来看看上面这段代码什么意思:

首先,明确我们的目的:防止一个文件被多次的include。为什么会存在多次include呢?因为可能一个文件会include多个文件,而不同文件有可能include同一个文件,而在一个文件中只能对一个文件中include一次。举个栗子:存在a.h,b.h,test.h,有可能a.h中有一个#include "test.h",同时b.c中有两行

#include "a.h"
#include "test.h"

这样就出现了b.h被包含了两次,而C是不允许这样的。上面用到#ifndef就解决了这个问题。

test.h未被定义,也就是说这个头文件被第一次包含了,会定义一个_TEST_H的常量,但没有body,因为我们不需要这个常量具有body。这时候头文件里的所有内容被预处理器执行。当test.h被第二次包含时,预处理器检测到已经存在_TEST_H这个量,就会跳过,也就不会重复执行头文件里的内容了。这里的关键是保持_TEST_H的唯一。我们这里用头文件的文件名做标识符,将文件名前面加下划线,并将点换成下划线,能够保证这个标识符不重复。但这种做法是编译器提供商为了标准头文件采用的方法,我们平时自己写的时候要想防止与标准头文件冲突,就不要按这种方法来取标识符。例如可以把下划线加到文件名后面等等。

#if#elif

#if#elif后面跟常量整数表达式,当这个表达式值为非0值时为真,为0位假,类似于C程序中的if-else分支结构。可以应用C的关系运算符以及逻辑运算符。

#if defined(FF)
#define SS 9999
#elif defined(_TEST_H)
#define EE 99
#endif

预定义宏

一些常见的预定义宏:

意义
__DATE__ 进行预处理操作的日期字符串
__FILE__ 当前源代码文件名的字符串
__LINE__ 当前所在代买行号
__TIME__ 源文件编译时间
__STDC_VERSION__ 当前编译器遵循的C标准版本,当为C99时值为199901L

#line#error

#line指令用于重置由__LINE____FILE__宏报告的行号和文件名。

使用方法:

#line 100 //当前行号重置为100
#line 10 "test.c" //把行号重置为10,文件名重置为test.c
printf("%d\n",__LINE__);
#line 1000
printf("%d\n",__LINE__);
printf("%d\n",__LINE__);

#error指令使预处理器发出一条错误消息,该消息包含指令中的文本。可能的话,编译过程应该中断。

#if __STDC_VERSION__ != 199901L
#error Not C99
#endif

内联函数

内联函数主要是为了把解决函数调用过程中建立调用、传递参数、跳转到函数代码段返回耗费的时间较长的问题。它适用于较短代码块,创建内联函数需要用到inline关键字。

inline double square(double x);
double square(double x){
return x*x;
}
int main() {
double a = square(3);
printf("%lf\n",a);
}
//内联函数这里其实把函数体直接写到了main里面,不用调用了。
/*
*int main(){
* double a = a * a;
* printf("%lf\n",a);
*}
*/

无法获取内联函数的地址,因为内联函数并没有被分配单独的空间。内联函数适用于短小函数,长的函数调用造成的时间远远小于函数体执行时间,使用内联函数并没什么意义。

内联函数的定义和对该函数的调用必须在同一文件中。因此,内联函数一般具有内部链接。在多文件程序中,要想使用相同的内联函数,必须在每个文件中对内联函数进行定义。为了方便,可以把内联函数的定义放在头文件中。

C Primer Plus--C预处理器和C库(2)的更多相关文章

  1. C Primer Plus--C预处理器和C库(1)

    目录 预处理符号 明显常量 #define 编译程序之前,先由预处理器检查程序(因此称为预处理器).根据程序中使用的预处理器指令,预处理用符号缩略语所代表的内容替换程序中的缩略语. 预处理器可以根据你 ...

  2. C Primer Plus之C预处理器和C库

    编译程序前,先由预处理器检查程序(因此称为预处理器).根据程序中使用的预处理器指令,预处理器用符号缩略语所代表的内容替换程序中的缩略语. 预处理器不能理解C,它一般是接受一些文件并将其转换成其他文本. ...

  3. C预处理器和C库

    #define #include #undef #ifdef #else #endif #if #elif #else #endif 预处理宏: p463 _ _fun_ _是预定义标识符(函数作用域 ...

  4. 第 16 章 C 预处理器和 C 库(可变参数:stdarg.h)

    /*------------------------------------------------- varargs.c -- use variable number of arguments -- ...

  5. 第 16 章 C 预处理器和 C 库(string.h 库中的 memcpy() 和 memmove())

    /*----------------------------------------- mems.c -- 使用 memcpy() 和 memmove() ---------------------- ...

  6. 第 16 章 C 预处理器和 C 库(qsort() 函数)

    /*---------------------------------------- qsorter.c -- 用 qsort() 排序一组数字 --------------------------- ...

  7. 第 16 章 C 预处理器和 C 库(直角坐标转换极坐标)

    /*------------------------------------- rect_pol.c -- 把直角坐标转换为极坐标 ---------------------------------- ...

  8. 第 16 章 C 预处理器和 C 库(预定义宏)

    /*------------------------------------- predef.c -- 预定义宏和预定义标识符 ------------------------------------ ...

  9. 第 16 章 C 预处理器和 C 库(条件编译)

    /*-------------------------------------- names_st.h -- names_st 结构的头文件 ----------------------------- ...

随机推荐

  1. 接口的鉴权cookie、session和token

    1.HTTP是无状态协议 什么是无状态?就是说这一次的请求和上一次的请求是没有任何关系的,无法共享信息.好处就是速度快. 2.cookie.session的加入 HTTP请求是无状态的,所以解决共享信 ...

  2. EF CodeFirst Dome学习

    创建ConsoleDome控制台应用程序 从NuGet包管理器安装EntityFramework 创建DbContextDome类并继承DbContext public class DbContext ...

  3. Mycat使用--分库分表和读写分离

    Mycat分库分表读写分离 1. 模拟多数据库节点 2. 配置文件 具体操作参看: https://blog.csdn.net/vbirdbest/article/details/83448757 写 ...

  4. HTTP协议的认识

    1.内容回顾 1.HTTP协议消息的格式: 1.请求(request) 请求方法 路径 HTTP/1.1 \r\n k1:v1\r\n ...\r\n \r\n 请求体 <--这里可以有可以没有 ...

  5. Spring源码——IOC控制反转

    1.基础知识 Spring有两个核心功能,分别是ioc和aop,其中ioc是控制反转,aop是切面编程. 在ioc中,还有一个名次叫DI,也就是依赖注入.嗯,好像IOC和DI是指同一个,好像又感觉他俩 ...

  6. 英语Bisynes商务

    英语bisynes商务概念的提出是改革的产物,有一个演变的过程:贸易部--商业部.外贸部--内贸部--内贸局--商务部.是内外贸一体化的概念. 中文名:商务 外文名:Business,Bisynes商 ...

  7. Navicat导出数据库设计文档

    前言:仅支持单表导出 导出sql: SELECT TABLE_NAME 表名, COLUMN_NAME 列名, COLUMN_TYPE 数据类型, COLUMN_KEY 主键, IF(IS_NULLA ...

  8. oracle中start with和connect by的用法理解

    转自:https://blog.csdn.net/qq_29274091/article/details/72627350 Oracle中start with和connect by 用法理解转自:ht ...

  9. MongoDB 设置参数

    服务器配置文件分析 bin目录下的mongod.cfg是服务器的配置文件,文件中主要的配置参数: 1.数据库文件的存放位置 2.服务器日志文件的存放位置 3.默认的IP地址.端口号 设置密码 默认情况 ...

  10. Load Balancing in gRPC

    背景 基于每次调用的负载均衡 需要注意的是,gRPC的负载均衡发生在每次调用时,而不是每次连接时.换句话说,就算所有的请求来自于同一个客户,我们也希望可以将它们负载均衡到所有的服务器. 负载均衡的方法 ...