零基础非科班也能掌握的C语言知识22 预处理详解(完结)

news2025/1/23 11:35:42

预处理详解

  • 1.预处理符号
  • 2.#define 定义常量
  • 3.#define 定义宏
  • 4.带有副作用的宏参数
  • 5.宏替换的规则
  • 6.宏函数的对比
    • 6.1 例子
      • 6.1 .1
      • 6.1.2
      • 6.1.3
  • 7.命名约定
  • 8.undefin
  • 9.命令行定义(博主没办法演示)
  • 10.条件编译
  • 11.头文件的包含
    • 11.1本地文件
    • 11.2库文件的包含
    • 11.3 嵌套文件的包含
  • 12.其他预处理指令
  • 13.总结

1.预处理符号

#include <stdio.h>
 int main()
 {
 	printf("%s\n", __FILE__);//进⾏编译的源⽂件 
 	printf("%d\n", __LINE__);//⽂件当前的⾏号
 	printf("%s\n", __DATE__);//⽂件被编译的⽇期 
 	printf("%s\n", __TIME__);//⽂件被编译的时间 
 	printf("%d\n", __STDC__);//说明gcc完全遵循ANSI C

 	return 0;
 }

2.#define 定义常量

基本语法

#define name stuff
在这里插入图片描述
将华氏温度转化为对应的设置度,我们将系数进行了定义

3.#define 定义宏

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

#define name(parament-list) stuff

其中(parament-list)可以理解为参数
注意:参数列表的左括号必须与name紧邻,如果两者之间有任何空⽩存在,参数列表就会被解释为stuff的⼀部分。
举例:

#include<stdio.h>
#define SQUARE(n) n*n
#define DOUBLE(n) n+n
int main()
{
	printf("%d\n", SQUARE(5));//本意求得25
	printf("%d\n", SQUARE(5+1));//本意求得36
	printf("%d\n", DOUBLE(2)); //本意求得4
	printf("%d\n", 6*DOUBLE(2));//本意求得24
	return 0;
}

注意:宏只是对文本的替换也就是说
SQUARE(5)会替换成55
SQUARE(5+1)会替换成5+1
5+1
DOUBLE(2)会替换成2+2
6DOUBLE(2)会替换成62+2
这样的输出结果可能就不是我们的本意了
输出结果:
在这里插入图片描述
因此当我们使用宏定义时不应该吝啬圆括号也就是

#include<stdio.h>
#define SQUARE(n) ((n)*(n))//传的参数加括号,返回值再加括号
#define DOUBLE(n) ((n)+(n))
int main()
{
	printf("%d\n", SQUARE(5));//本意求得25
	printf("%d\n", SQUARE(5+1));//本意求得36
	printf("%d\n", DOUBLE(2)); //本意求得4
	printf("%d\n", 6*DOUBLE(2));//本意求得24
	return 0;
}

4.带有副作用的宏参数

有关副作用我在有关恼人的结合性一文已经解释过了(感兴趣的同学可以自己看看)。
当宏参数在宏的定义中出现超过⼀次的时候,如果参数带有副作⽤,那么你在使⽤这个宏的时候就可能出现危险,导致不可预测的后果。副作⽤就是表达式求值的时候出现的永久性效果。

#define MAX(x, y)  ((x)>(y)?(x):(y))

int main()
{
	int a = 10;
	int b = 20;
	//int m = MAX(a++, b++);
	int m = ((a++) > (b++) ? (a++) : (b++));
	       //10    > 20    ?  x    :  21  
	       // a=11 b=22
	printf("%d\n", m);//?
	printf("a=%d b=%d\n", a, b);

	return 0;
}

5.宏替换的规则

在这里插入图片描述

6.宏函数的对比

宏通常被应用于执行简单的运算
宏与函数的对比
在这里插入图片描述
但是宏能做到的函数一定做不到如

6.1 例子

6.1 .1

#define MALLOC(n, type)    (type*)malloc(n*sizeof(type))

int main()
{
	int *p = MALLOC(10, int);//也就是说宏的参数可以是一个类型
	if (p == NULL)
	{
		perror("error");
		return;
	}
	free(p);
	p = NULL;
	return 0;
}

函数能做到吗
还有两个例子,但是先介绍两个运算符#运算符和##运算符

6.1.2

#运算符将宏的⼀个参数转换为字符串字⾯量。它仅允许出现在带参数的宏的替换列表中。
#运算符所执⾏的操作可以理解为”字符串化“。

#define PRINT(format, n)    printf("the value of " #n " is " format"\n", n)
//这样的话打印下来的就是the value of n is n

int main()
{
	int a = 10;
	PRINT("%d", a);
	//printf("the value of " "a" " is " "%d""\n", a);
	//printf("the value of n is " "%d""\n", a);
	//printf("the value of a is %d\n", a);

	int b = 20;
	PRINT("%d", b);
	//printf("the value of b is %d\n", b);

	float f = 5.5f;
	PRINT("%f", f);
	//printf("the value of " "f" " is " "%f""\n", f)
	//printf("the value of n is " "%f""\n", f);
	//printf("the value of f is %f\n", f);


	return 0;
}

6.1.3

‘## 可以把位于它两边的符号合成⼀个符号,它允许宏定义从分离的⽂本⽚段创建标识符。 ## 被称为记号粘合
这样的连接必须产⽣⼀个合法的标识符。否则其结果就是未定义的。
为什么会要这样大家可以了解一下词法分析中的“分析法”

//函数也做不到
//生成函数的模板
#define GENERIC_MAX(type) \
type type##_max(type x, type y)\
{\
	return x>y?x:y;\
}

//使用上面的模板定义函数
//int_max
GENERIC_MAX(int)
//float_max
GENERIC_MAX(float)

int main()
{
	//
	printf("%d\n", int_max(3, 5));
	printf("%f\n", float_max(3.0, 5.0));

	return 0;

7.命名约定

把宏名全部⼤写
函数名不要全部⼤写(如get_number或者GetNumber)

尽管ANSI C中没有严格要求,但这是一个约定俗成的风格,建议大家遵守

8.undefin

这条指令⽤于移除⼀个宏定义

#undef NAME
如果现存的⼀个名字需要被重新定义,那么它的旧名字⾸先要被移除。

9.命令行定义(博主没办法演示)

许多C的编译器提供了⼀种能⼒,允许在命令⾏中定义符号。⽤于启动编译过程。
例如:
当我们根据同⼀个源⽂件要编译出⼀个程序的不同版本的时候,这个特性有点⽤处。(假定某个程序中声明了⼀个某个⻓度的数组,如果机器内存有限,我们需要⼀个很⼩的数组,但是另外⼀个机器内存⼤些,我们需要⼀个数组能够⼤些。)

10.条件编译

在编译⼀个程序的时候我们如果要将⼀条语句(⼀组语句)编译或者放弃是很⽅便的。因为我们有条件编译指令。

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
#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
可以注意到在头文件中运用了大量这样的语句
在这里插入图片描述

11.头文件的包含

11.1本地文件

在这里插入图片描述

11.2库文件的包含

在这里插入图片描述

11.3 嵌套文件的包含

我们已经知道, #include 指令可以使另外⼀个⽂件被编译。就像它实际出现于 #include 指令的地⽅⼀样。
这种替换的⽅式很简单:预处理器先删除这条指令,并⽤包含⽂件的内容替换。
⼀个头⽂件被包含10次,那就实际被编译10次,如果重复包含,对编译的压⼒就⽐较⼤。

 #ifndef __TEST_H__
 #define __TEST_H__

 int Add(int x,int y);

 struct S
 {
     char c;
     int i;
 };
 #endif

我们最先包含头文件时未定义__TEST_H__就会执行下面的语句,再次包含时已经定义了__TEST_H__后面的语句在包含时就注释了(删除),当然我们也可以在相应的头文件中用一下预处理指令。

#pragma once

12.其他预处理指令

#error
#pragma
#line

#pragma pack()在结构体部分介绍。(改变默认对其量的)

13.总结

以上就是C语言的所有基础知识,满打满算学习了三个月的C语言知识。写了两个月的博客吧,写博客怎么说能一直在进步吧。但是总感觉自己写的也还不是很好,而且博客要写的东西也比较多,而且一直都在补前面的过程。怎么说呢,降低预期,保持努力吧。本来也不是科班的学生,甚至本专业没开设任何相关的课程狠狠地学环境,还是感谢大家的关注。我也不多想付出的努力一定会有相应的回报吧,因为这确实是我的兴趣所在。倘若我的博客能帮助到你,这就是对我最大的鼓励了。当然后面也会继续更新C语言的练习,还有数据结构,C++等等的知识。我也会优化自己博客的内容,给大家带来高质量的作品
心怀热爱,奔赴山海

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

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

相关文章

软件安全测评有哪些测试流程?第三方检测机构进行安全测评的好处

在今天的高科技时代&#xff0c;软件产品已经成为人们生活和工作的重要组成部分。然而&#xff0c;与其普及和深入应用的&#xff0c;软件安全问题也日益凸显。 为了保障软件产品在使用过程中的安全性&#xff0c;进行安全测评是必不可少的。安全测评可以全面评估软件系统的安…

GPT-4o多模态大模型的架构设计

GPT-4o&#xff1a;大模型风向&#xff0c;OpenAI大更新 OpenAI震撼发布两大更新&#xff01;桌面版APP与全新UI的ChatGPT上线&#xff0c;简化用户操作&#xff0c;体验更自然。同时&#xff0c;全能模型GPT-4o惊艳亮相&#xff0c;跨模态即时响应&#xff0c;性能卓越且性价比…

Java集合自测题

文章目录 一、说说 List , Set , Map 三者的区别&#xff1f;二、List , Set , Map 在 Java 中分别由哪些对应的实现类&#xff1f;底层的数据结构&#xff1f;三、有哪些集合是线程不安全的&#xff1f;怎么解决呢&#xff1f;四、HashMap 查询&#xff0c;删除的时间复杂度五…

k8s中的pod域名解析失败定位案例

问题描述 我在k8s中启动了一个Host网络模式的pod&#xff0c;这个pod的域名解析失败了。 定位步骤 敲kubectl exec -it [pod_name] -- bash进入pod后台&#xff0c;查看/etc/resolv.conf&#xff0c;发现nameserver配的有问题。这里我预期的nameserver应该使用宿主机的&…

【Linux】线程(一)

谈论之前需要先谈论一些线程的背景知识 其中就有进程地址空间&#xff0c;又是这个让我们又爱又恨的东西 目录 背景知识&#xff1a;地址空间&#xff1a; 背景知识&#xff1a; 地址空间&#xff1a; 说在前边&#xff0c;OS通常分为4个核心模块&#xff1a;执行流管理&…

【qt】绘图

绘图 一.画家二.绘图事件三.坐标体系四.画笔1.setColor2.setWidth3.setStyle4.setCapStyle5.setJoinStyle6.给画家配置笔 五.画刷1.setColor2.setStyle3.给画家设置刷子 六.用到的类汇总1.QRect 矩形2.QPoint 点3.QImage 图片4.QPixmap 图片5.QLine 线6.QPainterPath 路径 七.开…

如何关闭端口被占用的进程

如何关闭端口被占用的进程 操作步骤一、打开命令提示符&#xff08;Command Prompt&#xff09;二、查看占用端口的进程三、kill杀死占用端口的进程 操作步骤 一、打开命令提示符&#xff08;Command Prompt&#xff09; 使用 win R 打开命令行模式 然后在命令行窗口输入下…

【LLM Agent 长文本】Chain-of-Agents与Qwen-Agent引领智能体长文本处理革命

前言 大模型在处理长文本上下文任务时主要存在以下两个问题&#xff1a; 输入长度减少&#xff1a;RAG的方法可以减少输入长度&#xff0c;但这可能导致所需信息的部分丢失&#xff0c;影响任务解决性能。扩展LLMs的上下文长度&#xff1a;通过微调的方式来扩展LLMs的上下文窗…

基于DenseNet网络实现Cifar-10数据集分类

目录 1.作者介绍2.Cifar-10数据集介绍3.Densenet网络模型3.1网络背景3.2网络结构3.2.1Dense Block3.2.2Bottleneck层3.2.3Transition层3.2.4压缩 4.代码实现4.1数据加载4.2建立 DenseNet 网络模型4.3模型训练4.4训练代码4.5测试代码 参考链接 1.作者介绍 吴思雨&#xff0c;女…

搭建自己的多平台镜像站

# 1. 拉取代码 $ git clone https://github.com/wzshiming/crproxy.git $ cd crproxy/examples/default# 2. 修改网关域名 使用vim编辑start.sh文件&#xff0c;将第五行的gateway变量值修改为你自己设定的域名。 原&#xff1a;gatewaycr.zsm.io 修改为&#xff1a;gatewayXS…

go语言 | 快速生成数据库表的 model 和 queryset

就是生成 model 目录的 xxx.go 和 xxx_gen.go 文件 使用的工具&#xff1a; 快速生成 model&#xff1a;gentool&#xff1a;https://github.com/go-gorm/gen/tree/master/tools/gentool 根据 model 生成 queryset&#xff1a;go-queryset&#xff1a;https://github.com/jirfa…

layuimini框架实现点击菜单栏回到起始页

在layui页面中&#xff0c;如果使用了 window.location.href""进行了页面跳转&#xff0c;再点击菜单栏是不会显示起始页&#xff0c;而是跳转后的页面&#xff0c; 解决&#xff1a; 在miniTab.js文件中找到&#xff1a;listen方法 将其中修改为&#xff1a; if …

全球首创4090推理!昆仑万维开源Skywork-MoE模型

昆仑万维近期宣布开源了其2千亿参数规模的稀疏大模型Skywork-MoE。这个模型是基于他们之前开源的Skywork-13B模型中间checkpoint扩展而来的&#xff0c;并且宣称是首个完整应用MoE Upcycling技术的开源千亿MoE大模型。此外&#xff0c;它也是首个支持使用单台RTX 4090服务器&am…

MyBatisPlus代码生成器(交互式)快速指南

引言 本片文章是对代码生成器(交互)快速配置使用流程&#xff0c;更多配置方法可查看官方文档&#xff1a; 代码生成器配置官网 如有疑问欢迎评论区交流&#xff01; 文章目录 引言演示效果图引入相关依赖创建代码生成器对象引入Freemarker模板引擎依赖支持的模板引擎 MyBat…

Day 20:2806. 取整购买后的账户余额

Leetcode 2806. 取整购买后的账户余额 一开始&#xff0c;你的银行账户里有 100 块钱。 给你一个整数purchaseAmount &#xff0c;它表示你在一次购买中愿意支出的金额。 在一个商店里&#xff0c;你进行一次购买&#xff0c;实际支出的金额会向 最近 的 10 的 倍数 取整。换句…

【QT5】<总览二> QT信号槽、对象树及常用函数

文章目录 前言 一、QT信号与槽 1. 信号槽连接模型 2. 信号槽介绍 3. 自定义信号槽 二、QT的对象树 三、添加资源文件 四、样式表的使用 五、QSS文件的使用 六、常用函数与宏 前言 承接【QT5】&#xff1c;总览一&#xff1e; QT环境搭建、快捷键及编程规范。若存在版…

Python使用tkinter库设置背景图片、label显示位置和label设置显示图片

tkinter 设置背景图片 label显示位置 label设置显示图片 from tkinter import * import tkinter as tk from PIL import ImageTk from PIL import Imagedef get_img(filename, width, height):im Image.open(filename).resize((width, height))im ImageTk.PhotoImage(im)…

网络仿真方法综述

目录 1. 引言 2.仿真器介绍 2.1 NS-2 2.2 NS-3 2.3 OPNET 2.4 GNS3 3.仿真对比 4.结论 参考文献 1. 引言 网络仿真是指使用计算机模拟网络系统的行为和性能的过程。在网络仿真中&#xff0c;可以建立一个虚拟的网络环境&#xff0c;并通过模拟各种网络设备、协议和应用程…

this关键字,构造函数(构造器)

文章目录 thisthis是什么应用场景 构造器注意事项代码演示 this this是什么 this就是一个变量&#xff0c;可以在方法中&#xff0c;拿到当前对象 应用场景 解决变量名称 冲突问题 构造器 注意事项 必须和类名相同没有返回值只要参数不同&#xff08;个数不同&#xff0…

【C语言】10.操作符详解

一、操作符分类 • 算术操作符&#xff1a; 、- 、* 、/ 、% • 移位操作符: << 、 >> • 位操作符: & 、|、^、 ~ • 赋值操作符: 、 、 - 、 、 / 、% 、<< 、>> 、& 、| 、^ • 单目操作符&#xff1a; &#xff01;、、–、&、、、…