一、#define的基本用法

1.1 #define的概念

#define命令是C语言中的一个宏定义命令,它用来将一个标识符定义为一个字符串,该标识符被称为宏名,被定义的字符串称为替换文本。

  • 简单的宏定义
    • #define <宏名> <字符串>
    • 例: #define PI 3.1415926
  • 带参数的宏定义
    • #define <宏名> (<参数表>) <宏体>
    • 例: #define A(x) x

1.2 宏替换发生的时机

(1)文件包含
可以把源程序中的#include 扩展为文件正文,即把包含的.h文件找到并展开到#include 所在处。
(2)条件编译
预处理器根据#if#ifdef等编译命令及其后的条件,将源程序中的某部分包含进来或排除在外,通常把排除在外的语句转换成空行。
(3)宏展开
预处理器将源程序文件中出现的对宏的引用展开成相应的宏定义,
即本文所说的#define的功能,由预处理器来完成。
经过预处理器处理的源程序与之前的源程序有所有不同,在这个阶段所进行的工作只是纯粹的替换与展开,没有任何计算功能,所以在学习#define命令时只要能真正理解这一点,这样才不会对此命令引起误解并误用。

2.1 简单宏定义使用中出现的问题

在简单宏定义的使用中,当替换文本所表示的字符串为一个表达式时,容易引起误解和误用。

#define N 2+2
void main()
{
   int a=N*N;
   printf(“%d”,a);
}

结果为8 而不是16

如1节所述,宏展开是在预处理阶段完成的,这个阶段把替换文本只是看作一个字符串,并不会有任何的计算发生,在展开时是在宏N出现的地方 只是简单地使用串2+2来代替N,并不会增添任何的符号,所以对该程序展开后的结果是a=2+2*2+2,计算后=8,这就是宏替换的实质,如何写程序才能完成结果为16的运算呢

将宏定义写成如下形式
#define N (2+2)

二、define中的三个特殊符号:#,##,#@

  • #define Conn(x,y) x##y

    • 表示x连接y
    • int n = Conn(123,456); /* 结果就是n=123456;*/
    • char* str = Conn("asdf", "adf"); /结果就是 str = "asdfadf";/
  • #define ToChar(x) #@x

    • 给x加上单引号,结果返回是一个const char
    • char a = ToChar(1);结果就是a='1';
    • 做个越界试验char a = ToChar(123);结果就错了;
    • 但是如果你的参数超过四个字符,编译器就给给你报错了!error C2015: too many characters in constant :P

  • #define ToString(x) #x
    • 给x加双引号
    • char* str = ToString(123132);就成了str="123132";

三、常用的一些宏定义

1、防止一个头文件被重复包含

#ifndef BODYDEF_H 
#define BODYDEF_H 
 //头文件内容 
#endif

2、得到指定地址上的一个字节或字

#include <iostream>
#include <windows.h>
#define MEM_B(x) (*((byte*)(x)))
#define MEM_W(x) (*((WORD*)(x)))
int main()
{
    int bTest = 0x123456;
    byte m = MEM_B((&bTest));/*m=0x56*/
    int n = MEM_W((&bTest));/*n=0x3456*/
    return 0;
}

Q.E.D.


ALL WILL BE CLEAR