首页 > 其他分享 >预处理详情

预处理详情

时间:2024-10-25 13:46:34浏览次数:3  
标签:__ return 定义 int 详情 printf 预处理 define

目录

1. 预定义符号

 2. #define 定义常量

3. #define定义宏

4. 带有副作⽤的宏参数

 5. 宏替换的规则

 6. 宏函数的对⽐

7. #和##

7.1 #运算符

7.2 ## 运算符

 8. 命名约定

9. #undef

 10. 命令行定义

 11. 条件编译

11.1#if

2.多个分⽀的条件编译

3.判断是否被定义

4.嵌套指令

 12. 头⽂件的包含

 12.1.本地⽂包含

12. 2 库⽂件包含

12.3嵌套⽂件包含


这也将是c语言的最后一节课,希望大家好好学习,

1. 预定义符号

C语⾔设置了⼀些预定义符号,可以直接使⽤,预定义符号也是在预处理期间处理的。

FILE__ //进⾏编译的源⽂件

__LINE__ //⽂件当前的⾏号

__DATE__ //⽂件被编译的⽇期

__TIME__ //⽂件被编译的时间

__STDC__ //如果编译器遵循ANSI C,其值为1,否则未定义

#include <stdio.h>
int main()
{
	printf("%s\n", __FILE__);
	printf("%d\n", __LINE__);
	printf("%s\n", __DATE__);
	printf("%s\n", __TIME__);
//	printf("%s\n", __STDC__);

	return 0;
}

  

 当我们把这个注释掉其他的打印内容会是什么样的呢?

 2. #define 定义常量

#define name stuf

 #define MAX 1000
#define reg register //为 register这个关键字,创建⼀个简短的名字
#define do_forever for(;;) //⽤更形象的符号来替换⼀种实现
#define CASE break;case //在写case语句的时候⾃动把 break写上。
// 如果定义的 stuff过⻓,可以分成⼏⾏写,除了最后⼀⾏外,每⾏的后⾯都加⼀个反斜杠(续⾏
符)。
#define DEBUG_PRINT printf("file:%s\tline:%d\t \
 date:%s\ttime:%s\n" ,\
 __FILE__,__LINE__ , \
 __DATE__,__TIME__ )

 思考:在define定义标识符的时候,要不要在最后加上 ; ? ⽐如:

efine MAX 1000;

#define MAX 1000#d

 如果是加了分号的情况,等替换后,if和else之间就是2条语句,⽽没有⼤括号的时候,if后边只能有⼀ 条语句。这⾥会出现语法错误。

3. #define定义宏

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

#define name( parament-list ) stuff

 其中的 parament-list 是⼀个由逗号隔开的符号表,它们可能出现在stuff中。

注意: 参数列表的左括号必须与name紧邻,如果两者之间有任何空⽩存在,参数列表就会被解释为stuff的 ⼀部分

定义一个宏,数的平方

#define QUT(n) n*n
int main()
{
	int a = 0;
	scanf("%d", &a);
	int ret = QUT(a);
	printf("%d ", ret);


	return 0;
}

 这个宏接收⼀个参数 x .如果在上述声明之后,你把 SQUARE( 5 ); 置于程序中,预处理器就会⽤ 下⾯这个表达式替换上⾯的表达式: 5 * 5

警告: 这个宏存在⼀个问题: 观察下⾯的代码段:

你可能觉得这段代码将打印36,事实上它将打印11,为什么呢? 替换⽂本时,参数x被替换成a + 1,所以这条语句实际上变成了

#define SQU(n) n*n //这里是把5+1 都传入n中,5+1 *5+1=11
int main()
{
	
	int ret = SQU (5 + 1);
	printf("%d ", ret);

	return 0;

如果想要这个的结果变成36 只需把5+1 添加括号

#define SQU(n) (5+1)*(5+1) 
int main()
{

    int ret = SQU(5 + 1);
    printf("%d ", ret);

    return 0;
}

 

4. 带有副作⽤的宏参数

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

x+1;//不带副作⽤

x++;//带有副作⽤

 define MAX(a, b) ((a) > (b) ? (a) : (b))
...
x = 5;
y = 8;
z = MAX(x++, y++);
printf("x=%d y=%d z=%d\n", x, y, z);//输出的结果是什么?

z = ( (x++) > (y++) ? (x++) : (y++));

 所以输出的结果是:x=6 y=10 z=9

 5. 宏替换的规则

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

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

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

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

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

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

 6. 宏函数的对⽐

宏通常被应⽤于执⾏简单的运算。

⽐如在两个数中找出较⼤的⼀个时,写成下⾯的宏,更有优势⼀些。

#define MAX(a, b) ((a)>(b)?(a):(b))

 那为什么不⽤函数来完成这个任务? 原因有⼆:

1. ⽤于调⽤函数和从函数返回的代码可能⽐实际执⾏这个⼩型计算⼯作所需要的时间更多。所以宏⽐函数在程序的规模和速度⽅⾯更胜⼀筹。

2. 更为重要的是函数的参数必须声明为特定的类型。所以函数只能在类型合适的表达式上使⽤。反之 这个宏怎可以适⽤于整形、⻓整型、浮点型等可以⽤于 > 来⽐较的类型。宏的参数是类型⽆关 的。

和函数相⽐宏的劣势:

1. 每次使⽤宏的时候,⼀份宏定义的代码将插⼊到程序中。除⾮宏⽐较短,否则可能⼤幅度增加程序 的⻓度。

2. 宏是没法调试的。

3. 宏由于类型⽆关,也就不够严谨。

4. 宏可能会带来运算符优先级的问题,导致程容易出现错。 宏有时候可以做函数做不到的事情。⽐如:宏的参数可以出现类型,但是函数做不到。

#define MALLOC(num, type)\
 (type )malloc(num sizeof(type))
...
//使⽤
MALLOC(10, int);//类型作为参数
//预处理器替换之后:
(int*)malloc(10 sizeof(int));

 宏和函数的⼀个对⽐

7. #和##

7.1 #运算符

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

当我们有⼀个变量 int a = 10; 的时候,我们想打印出: the value of a is 10 . 就可以写:

#define PRINT(n) printf("the value of "#n " is %d", n);

 当我们按照下⾯的⽅式调⽤的时候: PRINT(a);//当我们把a替换到宏的体内时,就出现了#a,⽽#a就是转换为"a",时⼀个字符串

printf("the value of ""a" " is %d", a)

7.2 ## 运算符

 ## 可以把位于它两边的符号合成⼀个符号,它允许宏定义从分离的⽂本⽚段创建标识符。 ## 被称 为记号粘合 这样的连接必须产⽣⼀个合法的标识符。否则其结果就是未定义的。 这⾥我们想想,写⼀个函数求2个数的较⼤值的时候,不同的数据类型就得写不同的函数。

int int_max(int x, int y)
{
    return x > y ? x : y;
}
float float_max(float x, float y)
{
    return x > yx:y;
}

 但是这样写起来太繁琐了,现在我们这样写代码试试

//宏定义
#define GENERIC_MAX(type) \   续航符
type type##_max(type x, type y)\
{ \
 return (x>y?x:y); \
}

 GENERIC_MAX(int) //替换到宏体内后int##_max ⽣成了新的符号 int_max做函数名
GENERIC_MAX(float) //替换到宏体内后float##_max ⽣成了新的符号 float_max做函数名
int main()
{
    //调⽤函数
    int m = int_max(2, 3);
    printf("%d\n", m);
    float fm = float_max(3.5f, 4.5f);
    printf("%f\n", fm);
    return 0;
}

 8. 命名约定

把宏名全部⼤写

函数名不要全部⼤写

9. #undef

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


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

#define  M 10
int main()
{

	int a = M;
#undef M  //要取消的名字
	int b = M;

	return 0;
}

 10. 命令行定义

许多C 的编译器提供了⼀种能⼒,允许在命令⾏中定义符号。⽤于启动编译过程。

 #include <stdio.h>
int main()
{
    int array[ARRAY_SIZE];
    int i = 0;
    for (i = 0; i < ARRAY_SIZE; i++)
    {
        array[i] = i;
    }
    for (i = 0; i < ARRAY_SIZE; i++)
    {
        printf("%d ", array[i]);
    }
    printf("\n");
    return 0;
}

 编译指令:

//linux 环境演⽰

gcc -D ARRAY_SIZE=10 programe.c

 11. 条件编译

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

11.1#if

#if 常量表达式

//... #endif

int main()
{
#if 0
	printf("haha\n");

	printf("%d ",100);


	return 0;
 }
#endif//如果把#endif放在最后相当于被注释掉了。

#if 常量表达式

//... #elif 常量表达式

//... #else

//... #endif

2.多个分⽀的条件编译

#define M 5
int main()
{
#if M == 1
	printf("nihao ");
#elif M== 2
	printf("haoya ");
#elif  M== 4
	printf("nn ");
#else  M== 5
	printf("%d ",100);
#endif

	return  0;
}

3.判断是否被定义

#if defined(symbol)

#ifdef symbol

#if !defined(symbol)

#ifndef symbol

#define ZHANGSAN 
int main()
{
#ifdef EHANGSAN
	printf("ZHANGSAN\n");
#endif
	printf("WANGWU\n ");

	return 0;
}

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

 12. 头⽂件的包含

 

 12.1.本地⽂包含

#include "filename"

 Linux环境的标准头⽂件的路径:

1 /usr/include

VS环境的标准头⽂件的路径:

12. 2 库⽂件包含

查找头⽂件直接去标准路径下去查找,如果找不到就提⽰编译错误。 这样是不是可以说,对于库⽂件也可以使⽤ “” 的形式包含? 答案是肯定的,可以,但是这样做查找的效率就低些,当然这样也不容易区分是库⽂件还是本地⽂件 了。

12.3嵌套⽂件包含

#include "test.h"
#include "test.h"
#include "test.h"
#include "test.h"
#include "test.h"
int main()
{

    return 0;
}

 

 如果直接这样写,test.c⽂件中将test.h包含5次,那么test.h⽂件的内容将会被拷⻉5份在test.c中。 如果test.h ⽂件⽐较⼤,这样预处理后代码量会剧增。如果⼯程⽐较⼤,有公共使⽤的头⽂件,被⼤家 都能使⽤,⼜不做任何的处理,那么后果真的不堪设想。 如何解决头⽂件被重复引⼊的问题?答案:条件编译

#ifndef __TEST_H__
#define __TEST_H__
//头⽂件的内容
#endif //__TEST_H//防止头文件被多次包含的。

 或者

#pragma once //头文件只会包含一次

 就可以避免头⽂件的重复引⼊。

到这里c语言就结束了,希望越学越幸运。

撒花完结!!!!

标签:__,return,定义,int,详情,printf,预处理,define
From: https://blog.csdn.net/2402_82715436/article/details/143201053

相关文章

  • 提供一份 1688 商品详情接口的错误码及解决方法
    以下是一些常见的1688商品详情接口可能出现的错误码及解决方法:一、错误码:401(Unauthorized)含义:未授权访问,通常是因为调用接口时使用的认证信息(如token、密钥等)不正确或已过期。解决方法:仔细检查在请求中携带的认证信息是否正确,包括是否与在1688开放平台申请的信息一致,是......
  • 分享一些利用商品详情数据挖掘潜在需求的成功案例
    以下是一些利用商品详情数据挖掘潜在需求的成功案例:一、亚马逊的个性化推荐系统:案例背景:亚马逊是全球知名的电商平台,拥有海量的商品和庞大的用户群体。为了提高用户的购物体验和增加销售额,亚马逊投入大量资源开发个性化推荐系统。数据挖掘过程:亚马逊通过分析用户的购买历史、......
  • 【深度学习代码调试5】标准化数据集:TensorFlow Datasets (TFDS)自动化数据加载与预处
    【标准化数据集】TensorFlowDatasets、TFDS:自动化数据加载与预处理写在最前面1.什么是TensorFlowDatasets(TFDS)?主要特点:2.TFDS的核心API:`tfds.builder`和`download_and_prepare``tfds.builder`:创建数据集构建器示例:`download_and_prepare`:下载与准备数据集......
  • 代购系统的“闪电侠”:秒速获取商品详情
    亲爱的购物达人们,今天咱们不聊那些让人头大的购物车清空策略,也不谈那些让人眼花缭乱的折扣券收集。咱们来聊聊那些在代购世界里以光速奔跑,快速获取商品详情的“闪电侠”——我们的代购系统。代购系统的“闪电侠”:速度与激情的结合想象一下,你在代购的赛道上,眼前是一条条充满......
  • 探秘拼多多商品详情API接口:开启电商数据洞察之门
    作为一名技术员,在参与涉及拼多多商品详情API接口的项目后,我对这个接口有了深刻的认识和理解。它就像一把钥匙,为我们打开了电商数据洞察的大门。一、项目背景与需求在电商领域,数据是至关重要的。了解商品的详细信息、销售趋势、用户评价等,可以帮助商家做出更明智的决策,提高......
  • 快手根据ID取商品详情 API 返回值说明
    快手根据ID取商品详情API返回值说明item_get-根据ID取商品详情 ks.item_get公共参数请求地址:名称类型必须描述keyString是调用key(必须以GET方式拼接在URL中)secretString是调用密钥api_nameString是API接口名称(包括在请求地址中)[item_search,item_get,item_search_sho......
  • Python的京东探险记:揭秘商品详情的快速通道
    在一个充满无限可能的数字世界里,Python,这位编程界的多面手,正准备踏上一场刺激的探险之旅:快速获取京东商品的详情数据。这不仅是一次技术的挑战,更是一次与时间赛跑的较量。!Python先生,这位机智的代码探险家,打开了他的笔记本电脑,准备开始这场冒险。他知道,要快速获取京东的商品详情......
  • 如何利用商品详情数据挖掘消费者的潜在需求?
    以下是一些利用商品详情数据挖掘消费者潜在需求的方法:一、分析商品属性信息:材质与成分分析:例如对于服装产品,通过分析商品详情中不同材质的受欢迎程度、消费者对材质的评价等,可以发现消费者对于某些特定材质的偏好,以及对新型、舒适、环保材质的潜在需求。如果数据显示消费者对某......
  • 拼多多商品详情,一键获取的“魔法”
    亲爱的朋友们,你是否曾在拼多多上看到一件心仪的商品,却因为繁琐的手动操作而望而却步?是否在想,如果能有一种“魔法”,让我们能快速获取商品的详细信息,那该多好!今天,我要告诉你,这种“魔法”真的存在,它就是——PDDAPI!首先,让我们来聊聊拼多多。这个平台以其独特的团购模式和亲民的价格,......
  • CSS速刷 - 预处理器
    预处理器是什么?lessSass预处理器有啥功能?嵌套,反映了层级和约束变量和计算,减少了重复代码Extend和Mixin代码片段,就像具备同一个功能的函数。循环,适用于复杂有规律的样式importCSS文件模块化1.less嵌套Node写的,通过npm发布。&:同一层级2.Sass嵌套输出和less嵌套......