C++预处理器

预处理器是一些指令,指示表一起在实际编译之前所需要完成的预处理。

所有的预处理器指令都是以井号(#)开头,只有空格字符可以出现在预指令处理之前。预处理指令不是C++语句,所以他们不会以分号(;)结尾。

 

#define预处理

#define预处理指令用于创建符号常量。该符号常量通常称之为宏,指令的一般行为为

#define macro-name replacement-text

当这行代码出现在一个文件中,在该文件中后续出现的所有宏都会在程序编译之前被替换成成replacement-text。

/***
define.cpp
***/
#include<iostream>
using namespace std;#define PI 3.14159int main()
{cout << "Value of PI :" << PI << endl;return 0;
}

使用-E选项进行编译,并把结果重定向到define.p 现在查看一下相关信息:

exbot@ubuntu:~/wangqinghe/C++/20190816$ gcc -E define.cpp > define.p

条件编译

有几个指令可以用来有选择地对部分程序源代码进行编译,这个过程称之为条件编译。

条件预处理器地结构与if选择结构很像。

#ifdef NULL#define NULL 0
#denif

在调试时进行编译,调试开关可以使用一个宏来实现

#ifdef DEBUGcerr << “Variable x = ” << x << endl;
#endif

如果在指令 #ifdef DEBUF 之前定义了符号常量,则会对程序中cerr语句进行编译。可以使用 #if 0 语句注释掉程序的一部分。

#if 0 不进行编译的代码
#endif

实例:

/***
ifdef.cpp
***/
#include<iostream>
using namespace std;
#define DEBUG#define MIN(a,b) (((a) < (b)) ? a : b)int main()
{int i,j;i = 100;j = 20;
#ifdef DEBUGcerr << "Trace: Inside main function" << endl;
#endif#if 0cout << "MKSTR(HELLO C++)" << endl;
#endifcout << "The minimum is " << MIN(i,j) << endl;#ifdef DEBUGcerr << "Trace: Comint out of main function" << endl;
#endifreturn 0;
}

运行结果:

exbot@ubuntu:~/wangqinghe/C++/20190816$ g++ ifdef.cpp -o ifdef

exbot@ubuntu:~/wangqinghe/C++/20190816$ ./ifdef

Trace: Inside main function

The minimum is 20

Trace: Comint out of main function

#和##运算符

#和##预处理运算符在C++和ANSI/ISO C中都是可用的

#运算符会把repalcement-text令牌转换成引号引起来的字符串

/***
pound.cpp
***/
#include<iostream>
using namespace std;#define MKSTR(x) #xint main()
{cout << MKSTR(HELLO C++) << endl;return 0;
}

运行结果:

exbot@ubuntu:~/wangqinghe/C++/20190816$ g++ pound.cpp -o pound

exbot@ubuntu:~/wangqinghe/C++/20190816$ ./pound

HELLO C++

 

C++预处理器把下面这行

cout << MKSTR(HELLO C++) << endl;

转化成了:

cout << “HELLO C++” << endl;

##运算符用于连接两个令牌

#define CONCAT(x,y) x ## y

当CONCAT出现在程序中,他的参数机会被连接起来,并用来取代宏,程序中CONCAT(HELLO C++)会被替换成“HELLO C++”如下图:

/***
concat.cpp
***/
#include<iostream>
using namespace std;#define concat(a,b) a##b
int main()
{int xy = 100;cout << concat(x,y) << endl;return 0;
}

运行结果:

exbot@ubuntu:~/wangqinghe/C++/20190816$ g++ concat.cpp -o concat

exbot@ubuntu:~/wangqinghe/C++/20190816$ ./concat

100

 

C++预处理器把下面这行:

cout << concat(x,y) << endl;

转化成了:

cout << xy;

C++中的预定义宏

 

C++提供了下表所示的一些预定义宏:

描述

__LINE__

这会在编译时包含当前行号

__FILE__

这会在编译时包含当前文件名

__DATA__

这会包含一个形式为month/day/year的字符串,它表示把源文件转换成目标代码的日期

__TIME__

这会包含一个形式为hour:minute:second的字符串,它表示程序被编译的时间

 

实例:

/***
macro.cpp
***/
#include<iostream>
using namespace std;int main()
{cout << "Value of __LINE__ :" << __LINE__ << endl;cout << "Value of __FILE__ :" << __FILE__ << endl;cout << "Value of __DATE__ :" << __DATE__ << endl;cout << "Value of __TIME__ :" << __TIME__ << endl;return 0;
}

运行结果:

exbot@ubuntu:~/wangqinghe/C++/20190816$ g++ macro.cpp -o macro

exbot@ubuntu:~/wangqinghe/C++/20190816$ ./macro

Value of __LINE__ :6

Value of __FILE__ :macro.cpp

Value of __DATE__ :Aug 16 2019

Value of __TIME__ :14:06:20

# 和 ## 运算符

# 字符串化的意思,出现在宏定义中的#是把跟在后面的参数转换成一个字符串。

当用作字符串化操作时,# 的主要作用是将宏参数不经扩展地转换成字符串常量。

 宏定义参数的左右两边的空格会被忽略,参数的各个 Token 之间的多个空格会被转换成一个空格。

 宏定义参数中含有需要特殊含义字符如"或\时,它们前面会自动被加上转义字符 \。

## 连接符号,把参数连在一起。

将多个 Token 连接成一个 Token。要点:

 

 它不能是宏定义中的第一个或最后一个 Token。

 前后的空格可有可无。

 

转载于:https://www.cnblogs.com/wanghao-boke/p/11363571.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/384869.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C++信号处理

信号是由操作系统传给进程的中断&#xff0c;会提早终止一个程序。在UNIX、LINUX、Mac OS或windows系统上&#xff0c;可以通过按Ctrlc产生中断。有些信号不能被程序捕获&#xff0c;但是下表所列信号可以在程序中被捕获&#xff0c;并可以基于信号采取适当的动作。这些信号是定…

Python—“helloworld”

接触一门计算机新语言&#xff0c;第一件事就是要准备好一个编译器用来打代码。 网上很多环境搭建的方法&#xff0c;具体参照https://www.runoob.com/python/python-install.html 由于我之前一直在使用Ubuntu的gcc编译&#xff0c;恰好也可以运行python程序&#xff0c;我也就…

创建线程

pthread_self函数 获取线程ID。其作用对应进程中 getpid() 函数。 pthread_t pthread_self(void); 返回值&#xff1a;成功&#xff1a;0&#xff1b; 失败&#xff1a;无&#xff01; 线程ID&#xff1a;pthread_t类型&#xff0c;本质&#xff1a;在Linux下为无符号…

C/C++头文件

C/C头文件一览C、传统 C#include <assert.h> //设定插入点#include <ctype.h> //字符处理#include <errno.h> //定义错误码#include <float.h> //浮点数处理#include <fstream.h> //文件输入&#xff0f;输出#include <iomanip.h> //参数化…

Python变量类型

变量存储在内存中的值&#xff0c;这就意味着在创建变量时会在内存开辟一个空间。 基于变量的数据类型&#xff0c;解析器会分配指定内存&#xff0c;并决定什么数据可以被存储在内存中。 因此变量可以指定不同的数据类型&#xff0c;这些变量可以存储整数、小数、或字符。 变量…

线程退出

pthread_exit函数 将单个线程退出 void pthread_exit(void *retval); 参数&#xff1a;retval表示线程退出状态&#xff0c;通常传NULL 思考&#xff1a;使用exit将指定线程退出&#xff0c;可以吗&#xff1f; …

线程分离

pthread_detach函数 实现线程分离 int pthread_detach(pthread_t thread); 成功&#xff1a;0&#xff1b;失败&#xff1a;错误号 线程分离状态&#xff1a;指定该状态&#xff0c;线程主动与主控线程断开关系。线程结束后&#xff0c;其退出状态不由其他线程获取&#x…

线程知识点

控制原语对比 进程 线程 fork pthread_create exit pthread_exit wait pthread_join kill pthread_cancel getpid pthread_self 命名空间 线程属性 本节作为指引性介绍&…

读写锁

读写锁 与互斥量类似&#xff0c;但读写锁允许更高的并行性。其特性为&#xff1a;写独占&#xff0c;读共享。 读写锁状态&#xff1a; 一把读写锁具备三种状态&#xff1a; 1. 读模式下加锁状态 (读锁) 2. 写模式下加锁状态 (写锁) 3. 不加锁状态 读写锁特性&#xff1a; 读…

条件变量

条件变量&#xff1a; 条件变量本身不是锁&#xff01;但它也可以造成线程阻塞。通常与互斥锁配合使用。给多线程提供一个会合的场所。 主要应用函数&#xff1a; pthread_cond_init函数 pthread_cond_destroy函数 pthread_cond_wait函数 pthread_cond_timedwait函数 pthread_c…

文件锁

借助 fcntl函数来实现锁机制。 操作文件的进程没有获得锁时&#xff0c;可以打开&#xff0c;但无法执行read、write操作。 fcntl函数&#xff1a; 获取、设置文件访问控制属性。 int fcntl(int fd, int cmd, ... /* arg */ ); 参2&#xff1a; F_SETLK (struct flock *) 设置…

进程间同步

互斥量mutex 进程间也可以使用互斥锁&#xff0c;来达到同步的目的。但应在pthread_mutex_init初始化之前&#xff0c;修改其属性为进程间共享。mutex的属性修改函数主要有以下几个。 主要应用函数&#xff1a; pthread_mutexattr_t mattr 类型&#xff1a; 用于定义…

Python3字符串

字符串是Python中最常用的数据类型&#xff0c;可以使用单引号或双引号来创建字符串 创建字符串很简单&#xff0c;为变量分配一个值即可。 val1 ‘hello world’ var2 “Runoob” Python访问字符串的值 Python不支持单字符类型&#xff0c;单字符在Python中也是作为 一个字符…

服务器客户端编程

server 下面通过最简单的客户端/服务器程序的实例来学习socket API。 server.c的作用是从客户端读字符&#xff0c;然后将每个字符转换为大写并回送给客户端。 #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #incl…

Python3元组

Python的元组与列表相似&#xff0c;不同之处在于元组的元素不能修改 元组使用小括号&#xff0c;列表使用方括号 元组创建很简单&#xff0c;只需要在括号中添加元素&#xff0c;并使用逗号隔开即可。 创建空元组 tup1 (); tup2 (1,) 元组只包含一个元素时&#xff0c;需要在…

Python3字典

字典是另一种可变容器模型&#xff0c;可存储任意类型的对象。 字典的每个键值(key>value)对用冒号分隔&#xff0c;每个对之间用逗号分隔&#xff0c;整个字典包括在花括号里&#xff0c;格式如下 d {key1 : value,key2 : value2} 键必须是唯一&#xff0c;但值则不必。 值…

线程回收

pthread_join函数 阻塞等待线程退出&#xff0c;获取线程退出状态 其作用&#xff0c;对应进程中 waitpid() 函数。 int pthread_join(pthread_t thread, void **retval); 成功&#xff1a;0&#xff1b;失败&#xff1a;错误号 参数&#xff1a;thread&#xff1a;线…

Python3数字

Python3数字数据类型用于存储数值。 数据类型是不允许改变的&#xff0c;这就意味着&#xff0c;如果改变数字数据类型的值&#xff0c;将重新分配内存空间。 Python支持三种不同不同的数值类型&#xff1a; 整型&#xff08;int&#xff09;&#xff1a;通常是被称为整型或整数…

多进程服务器

注意&#xff1a;包含了“wrap.c” 和“wrap.h”文件在上篇博客中 /*** server.c ***/ #include<stdio.h> #include<string.h> #include<netinet/in.h> #include<arpa/inet.h> #include<signal.h> #include<sys/wait.h> #include<ctype…

服务器之select

select select能监听的文件描述符个数受限于FD_SETSIZE,一般为1024&#xff0c;单纯改变进程打开的文件描述符个数并不能改变select监听文件个数解决1024以下客户端时使用select是很合适的&#xff0c;但如果链接客户端过多&#xff0c;select采用的是轮询模型&#xff0c;会大…