详解#define

我们要知道,#define后面定义的标识符只进行替换而不进行计算,我们不能根据惯性自动给它计算了,这样可能会出错。


目录

1.关于#define

1.1#define定义标识符

1.2#define定义宏

1.3#define的替换规则

2.#和##

1.#

2.##

3.带副作用的宏参数

4.宏和函数的比较

5.命名的约定

6.#undef:可以移除宏定义


1.关于#define

1.1#define定义标识符

使用格式:#define name stuff

例如:#define MAX  1000   (这个是我们经常用到的)

          #define reg   register    为register创建一个简短的名字reg

          #define do_forever  for(  ; ; )   用更形象的符号来替换一种实现

          #define CASE break;case    写case语句的时候自动把break写上

我们可以举个代码例子来理解

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>//CASE break 来举例#define CASE break;case//它的意思就是把原来case的位置,替换为break;case
int main()
{int input = 0;int flag = 0;printf("请选择:>");scanf("%d ", &input);switch (input){case 1:flag = 1;CASE 2:  flag = 2;CASE 3:flag = 3;CASE 4:flag = 4;default:break;}printf("flag=%d\n", flag);return 0;
}

我们可以看到,对应CASE的位置被替换为 break;case  这样我们就不必每次都写break,因为我们有时会忘记写break,这样就很方便了,是不是很奇妙。

还有一点,在define定义标识符的时候,后面一般不要加“ ;”有时会出错误

还记得我们以前说过的悬空else吗,我们来举个例子:

#define MAX 100;int main()
{int m = 0;scanf("%d", &m);if (m >= 0)m = MAX;elsem = -1;printf("%d\n", m);return 0;
}

1.2#define定义宏

1.#define机制包括了一个规定,允许把参数替换到文本中,这种实现通常称为宏(macro)或定义宏(define macro)

2.宏的申明方式:#define  name(parament_list) stuff

其中: name(parament_list)是指宏的参数

        parament_list是一个由逗号隔开的符号表,它可能出现在stuff中

        stuff是宏的内容

3.参数列表的左括号必须与name紧邻,如果二者之间有任何空白存在,参数列表就会被解释为stuff的一部分

我们看个例子:

#define SQUARE(x) x*xint main()
{printf("%d\n", SQUARE(5));printf("%f\n", SQUARE(5.0));return 0;
}

但是这种定义方式存在隐患,我们之前说过#define定义的内容只进行替换而不计算,如果此时,我们把5改为5+1,它的值又是多少呢?

我们不妨先猜测一下,相信大多数人都回答36,但我们试着编译一下,发现结果和我们想象中不同

#define SQUARE(x) x*xint main()
{printf("%d\n", SQUARE(5+1));return 0;
}

我们发现结果是11,并不是我们想想中的36,我们来解释一下

SQUARE(5+1)就是 5+1*5+1 =11

所以此时,我们又得到了几个注意点:

在写宏时,我们要勇于加括号,防止代码中进行替换时,代码结果可能出现错误

当我们希望计算结果是一个整体时,建议整体给stuff加括号

例如:#define SQUARE(x)   ((x)*(x))这样就不会出现错误了

同时,宏的参数也可以是多个

例子:

#define MAX(x,y) ((x)>(y)?(x):(y))int main()
{int m = MAX(100, 443);printf("m=%d\n", m);return 0;
}

1.3#define的替换规则

虽然我们前面说define时可能已经说过了它的替换规则,但是在这里还是要给大家总结一下,方便大家总结:

在程序中扩展#define定义符号和宏时,需要涉及几个步骤:

1.在调用宏时,首先对参数进行检查,看看是否包含任何由define定义的符号,如果是,它们首先被替换

2.替换文本随后被插入到程序中原来的文本位置,对于宏,参数名被它们的值所替换

3.最后,再次对结果文件进行扫描,看看它是否包含任何由#define 定义的符号,如果是,就重复上述处理过程

注意:

1.宏参数和#define定义中可以出现其他#define定义的符号,但对于宏,不能出现递归

2.当预处理器搜索到#define定义的符号的时候,字符串常量的内容并不被搜索

2.#和##

1.#

 1.#它把参数插入到字符串中

2.用#把一个宏参数变为对应的字符串

这样说可能难以理解,我们直接看代码例子:

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>#define PRINT1(x) printf("the value of "#x" is %d\n",x)#define PRINT2(format,x) printf("the value of "#x" is "format" \n",x)int main()
{int a = 10;int b = 20;float f = 3.14f;
//#把宏参数插入到字符串中PRINT1(a);PRINT1(b);//把宏参数变成对应的字符串PRINT2("%f", f);return 0;
}

2.##

1.##可以把位于它两边的符号合成一个符号

2.它允许宏定义从分离的文件片段创建标识符

但是我们要注意,这样的连接必须产生一个合法的标识符,否则其结果就是未定义的

看个代码例子:

#define CAT(x,y,z) x##y##zint main()
{int iampig = 2023;printf("%d\n", CAT(i,am,pig));return 0;
}

3.带副作用的宏参数

当宏参数在宏的定义中出现超过一次的时候,如果参数带有副作用,那么你在使用这个宏的时候就可能出现危险导致不可预测的后果。副作用就是表达式求值的时候出现永久性效果

#define MAX(x,y) ((x)>(y)?(x):(y))
int main()
{int a = 4;int b = 6;int m = MAX(a++, b++);//带有副作用的参数printf("a=%d  b=%d  m=%d \n", a, b,m);return 0;
}

我们本以为打印出来的会是5,7,6.为什么会这样呢?我们来分析一下

但如果我们把它封装成函数,它就不会有这个副作用,这是因为函数会先计算再传参,和宏不一样,只替换不计算

我们看个函数例子:

int Max(int x, int y)
{return (x > y ? x : y);
}
int main()
{int a = 4;int b = 6;int m = Max(a++, b++);printf("a=%d  b=%d  m=%d \n", a, b,m);return 0;
}

所以,接下来就引出了函数和宏的对比,让我们接着学习

4.宏和函数的比较

我们从以下七点进行比较:

1.代码长度

宏:每次使用时,宏代码都会被插入到程序中,除了非常小的宏之外,程序的长度会大幅度增长

函数:函数代码只出现于一个地方,每次使用这个函数时,都会调用那个地方的同一份代码

2.执行速度
宏:更快

函数:存在函数的调用和返回的额外开销,所以相对慢一些

3.操作符的优先级

宏:宏参数的求值是在所以周围表达式的上下文环境里,除非加上括号,否则邻近操作符的优先级可能会产生不可预料的后果,所以建议宏在书写的时候多加括号

函数:函数参数只在函数调用的时候求值一次,它的结果值传递给函数,表达式的求值结果更容易预判

4.带有副作用的参数

宏:参数可能被替换到宏体中的多个位置,所以带有副作用的参数求值可能会产生不可预料的结果

函数:函数参数只在传参的时候求值一次,结果更容易控制

5.参数类型

宏:宏的参数与类型无关,只要对参数的操作是合法的,它就可以使用任何参数类型

函数:函数的参数是与类型有关的,如果参数的类型不同,就需要不同的函数,即使它们执行的任务是相同的

6.调试

宏:宏是不方便调试的

函数:函数是可以逐语句调试的

7.递归

宏:不能递归

函数:可以递归

5.命名的约定

一般来讲,函数和宏的使用语法很相似,所以C语言没有规定它们的使用语法

我们约定俗成

宏命名就大写,函数名不要全大写

但有时宏也有小写特例,我们要注意

6.#undef:可以移除宏定义

这个函数是可以移除宏定义的,我们看个例子:


#define MAX 100int main()
{printf("%d\n",MAX);//可以执行
#undef MAX//printf("%d\n",MAX);//不可执行return 0;
}

当我们#undef MAX后,在打印MXA,VS会提示我们MXA未定义,这就是它移除了宏定义。


好了,关于程序环境和预处理方面的知识,我们已经全部学完了,希望大家可以理解,关于C语言的理论方面的知识点到这里也是已经给大家全部说完了,对C语言说声拜拜。

我们下期再见!!!

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

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

相关文章

前端vue3——html2canvas给网站截图生成宣传海报

文章目录 ⭐前言⭐选择html2canvas实现网页截图&#x1f496; 截图 ⭐图片url截图显示不出来问题&#x1f496; 解决 ⭐最终效果&#x1f496; 定义海报 ⭐总结⭐结束 ⭐前言 大家好&#xff0c;我是yma16&#xff0c;本文分享关于 前端vue3——html2canvas给网站截图生成宣传…

【超强笔记软件】Obsidian实现免费无限流量无套路云同步

【超强笔记软件】Obsidian如何实现免费无限流量无套路云同步&#xff1f; 目录 一、简介 软件特色演示&#xff1a; 二、使用免费群晖虚拟机搭建群晖Synology Drive服务&#xff0c;实现局域网同步 1 安装并设置Synology Drive套件 2 局域网内同步文件测试 三、内网穿透群…

【设计模式-2.1】创建型——单例模式

说明&#xff1a;设计模式根据用途分为创建型、结构性和行为型。创建型模式主要用于描述如何创建对象&#xff0c;本文介绍创建型中的单例模式。 饿汉式单例 单例模式是比较常见的一种设计模式&#xff0c;旨在确保对象的唯一性&#xff0c;什么时候去使用这个对象都是同一个…

第一百七十九回 自定义SlideImageSwitch

文章目录 1. 概念介绍2. 思路与方法2.1 实现思路 3. 代码与效果3.1 示例代码3.2 运行效果 4. 内容总结 我们在上一章回中介绍了"SlideSwitch组件"相关的内容&#xff0c;本章回中将介绍自定义SlideImageSwitch.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 概…

代码随想录算法训练营 ---第四十二天

今天开始学习 动态规划&#xff1a;背包问题 也是比较难的一部分了 动态规划&#xff1a;背包问题 理论基础 01背包&#xff08;二维数组&#xff09; 有n件物品和一个最多能背重量为w 的背包。第i件物品的重量是weight[i]&#xff0c;得到的价值是value[i] 。每件物品只能用…

网络运维与网络安全 学习笔记2023.11.25

网络运维与网络安全 学习笔记 第二十六天 今日目标 ACL原理与类型、基本ACL配置、高级ACL配置 高级ACL之ICMP、高级ACL之telnet ACL原理与类型 项目背景 为了企业的业务安全&#xff0c;要求不同部门对服务器有不同的权限 PC1不能访问Server PC2允许访问Server 允许其他所…

求集合的笛卡尔乘积

求集合的笛卡尔乘积 一&#xff1a;【实验目的】二&#xff1a;【实验内容】三&#xff1a;【实验原理】四&#xff1a;代码实现&#xff1a; 一&#xff1a;【实验目的】 通过编实现给定集合A和B的笛卡尔积CAA,DAB,EBA,FAAB,GA(A*B&#xff09;. 二&#xff1a;【实验内容】…

【Android Gradle】之Gradle入门及 wrapper 生成(一)

&#x1f604;作者简介&#xff1a; 小曾同学.com,一个致力于测试开发的博主⛽️&#xff0c;主要职责&#xff1a;测试开发、CI/CD 如果文章知识点有错误的地方&#xff0c;还请大家指正&#xff0c;让我们一起学习&#xff0c;一起进步。 &#x1f60a; 座右铭&#xff1a;不…

illuminate/database 使用 四

文档&#xff1a;Hyperf Database: Getting Started - Laravel 10.x - The PHP Framework For Web Artisans 因为hyperf使用illuminate/database&#xff0c;所以按照文章&#xff0c;看illuminate/database代码实现。 一、读写分离 根据文档读写的host可以分开。设置读写分…

系列十六、Spring IOC容器的扩展点

一、概述 Spring IOC容器的扩展点是指在IOC加载的过程中&#xff0c;如何对即将要创建的bean进行扩展。 二、扩展点 2.1、BeanDefinitionRegistryPostProcessor 2.1.1、概述 BeanDefinitionRegistryPostProcessor是bean定义的后置处理器&#xff0c;在BeanDefinition加载后&a…

142.【Nginx负载均衡-01】

Nginx_基础篇 (一)、Nginx 简介1.背景介绍(1).http和三大邮局协议(2).反向代理与正向代理 2.常见服务器对比(1).公司介绍(2).lls 服务器(3).Tomcat 服务器(4).Apache 服务器(5).Lighttpd 服务器(6).其他的服务器 3.Nginx的优点(1).速度更快、并发更高(2).配置简单&#xff0c;扩…

力扣:提莫攻击

代码&#xff1a; class Solution { public:int findPoisonedDuration(vector<int>& timeSeries, int duration){//根据数组中给出的元素的值来进行判断&#xff01;//若后面元素-前面元素>d 中了d秒&#xff01;// <d 中了差的秒数&…