c/c++开发,无可避免的宏定义使用案例

news2025/1/1 10:29:01

一、c/c++宏定义的来源

        宏定义,就是用一个标识符来表示一个字符串,如果在后面的代码中出现了该标识符,那么就全部替换成指定的字符串。通常c/c++宏定义这几处出处:

        1)最常见的就是来自于开发者编码过程中采用宏定义命令“#define”来定义,它是一种C语言预处理命令。

        2) ANSI标准也自行提供了多个预定义的宏名,例如__DATE__,__TIME__,__FILE__, __LINE__, __FUNCTION__等,实际上也是采用“#define”来定义,只是标准库已经定义并占据了这些命名,开发者直接调用即可。

        3)另外编译器也会自带一些宏定义,类似于WIN32 __LINUX__等,另外开发者在编译过程中也可以通过配置文件、命令语句来传递宏定义参与编译。例如在makefile文件加入"MYID:=100",或在CMakeLists.txt 中加入“add_definitions( -DMYID)”,又或在gcc指令中直接加入“ -DMYID”,其实都可以看做是宏传递参与编译的一种。

二、宏定义应用

        宏定义命令“#define”是c/c++编程宏使用最多情况,具体展开有很多细节内容,但归根结底就两种用法,带参数宏定义和不带参数宏定义。

        不带参数宏定义格式:# define 标识符 字符序列

        例如:

#define PI (3.1415926)            //常量宏定义

        又例如:

//#define PYFREE 1  
#define PYFREE    //没给出字符序列也是可以的

        又例如,我们遇到最多的就头文件防止重复编译进行的宏定义

#ifndef _TEST__H_	//条件编译
#define _TEST__H_	//防止一个头文件被重复包含
//头文件内容
#endif //_TEST__H_	

        带参数宏定义格式:# define 标识符(参数表)  字符序列

        注意,宏调用时是以实参代换形参,而不是“值传送”,因此对于参数不要吝惜括号吧,例如:

#define CIRCLE_S(R)    PI*(R)*(R)    //带参宏定义

        又例如,参数还可以进行多重嵌套:

#define SET_VAL(VARIABLE,VAL)    ((VARIABLE) = (VAL))    //设值
#define SET_CLASS_VAL(INSTANCE, SUB_VARIABLE,VAL) SET_VAL(INSTANCE->SUB_VARIABLE, VAL)    //多重嵌套,类设值

        另外,通过"#"可以实现参数转字符串操作,通过“##”可以实现宏参数粘合在一起

#define STR_TRAN(arg)   #arg			//#把宏参数arg变为一个字符串
//STR_TRAN(100)等同于"100"
#define CONS(a,b) 		STR_TRAN(a##b)	//##把两个宏参数贴合在一起
//CONS(12,34) 等同于 "1234"

        ANSI标准库预定义的宏名,例如__DATE__,__TIME__,__FILE__, __LINE__, __FUNCTION__等,可以在程序直接调用。例如结合“##”,可以构造出类似printf函数类似的宏定义:

//ANSI标准库自带的宏定义:__DATE__,__TIME__,__FILE__, __LINE__, __FUNCTION__等
#define Print_INFO(log_fmt,...) \
	do{ \
		printf("[%s %s][%s:%d][%s] \n"log_fmt"\n",\
        __DATE__, __TIME__, __FILE__, __LINE__, __FUNCTION__, ##__VA_ARGS__); \
	}while (0)

//在main函数直接调用如下:
Print_INFO("hi");
//输出示例如下:
//[Feb 10 2023 14:29:05][test.c:47][main]
//hi

        通过编译指令或工程配置文件传递进来参与编译的宏,其使用和“#define”命令得到的宏是一致的,例如:

//gcc test.c -o test.exe -DGCC_CMD_DEF //传递宏定义GCC_CMD_DEF
#ifdef GCC_CMD_DEF	//条件编译
printf("GCC_CMD_DEF is define!\n");
#endif

又或
//gcc test.c -o test.exe -DGCC_CMD_DEF=1 //传递宏定义GCC_CMD_DEF并指定字符序列
#if GCC_CMD_DEF	//条件编译
printf("GCC_CMD_DEF is define!\n");
#endif

三、宏定义应用测试案例

        由于宏调用就是将宏名替换为字符串, 掌握"宏"概念的关键是“换”,并且这个“换”是在预处理(预编译)完成的,因此准确理解宏调用语句之前就先要“换”,再去阅读理解。

        按上述涉及到宏应用知识点,创建test.h/c源文件:

        test.h

#ifndef _TEST__H_	//条件编译
#define _TEST__H_	//防止一个头文件被重复包含

#define PI (3.1415926)			//常量宏定义
#define CIRCLE_S(R)	PI*(R)*(R)	//带参宏定义

#define SET_VAL(VARIABLE,VAL)	((VARIABLE) = (VAL))	//设值
#define SET_CLASS_VAL(INSTANCE, SUB_VARIABLE,VAL) SET_VAL(INSTANCE->SUB_VARIABLE, VAL)	//多重嵌套,类设值

#define STR_TRAN(arg)   #arg			//#把宏参数arg变为一个字符串
#define CONS(a,b) 		STR_TRAN(a##b)	//##把两个宏参数贴合在一起

#define PYFREE	//宏定义,用于条件编译
//ANSI标准库自带的宏定义:__DATE__,__TIME__,__FILE__, __LINE__, __FUNCTION__等
#define Print_INFO(log_fmt,...) \
	do{ \
		printf("[%s %s][%s:%d][%s] \n"log_fmt"\n",__DATE__, __TIME__, __FILE__, __LINE__, __FUNCTION__, ##__VA_ARGS__); \
	}while (0)
	
//
#ifdef WIN32	//条件编译
#pragma message("this is window platform")
#else
#pragma message("this is not window platform")	
#endif

#endif //_TEST__H_

        test.c

#include "test.h"
//gcc test.c -o test.exe

typedef struct Data_Test
{
	char cVal;
	int iVal;
}*pData,Data;

int main(int argc, char* argv[])
{
	printf("hello,def test!\n");
	float r = 2.0;
	printf("c=2*PI*r=%0.4f!\n",2*PI*r);
	printf("CIRCLE_S(r)=%0.4f!\n",CIRCLE_S(r));
	pData pd;
	SET_CLASS_VAL(pd,iVal,10);
	printf("pd->iVal=%d\n",pd->iVal);
	printf("STR_TRAN(val) is \"%s\" \n",STR_TRAN(val));
	printf("CONS(a,b) is \"%s\" \n",CONS(a,b));
	
	#ifdef PYFREE	//已定义编译条件
	printf("PYFREE is be defined!\n");	//执行
	#endif
	#if defined(PYFREE)		//条件编译
	//想想"#if PYFREE"呢,为何其不能编译通过,如果“#defined PYFREE 1”呢,有如何
	printf("PYFREE is really be defined!\n");	//执行
	#endif 
	#undef PYFREE	//取消宏定义
	#ifdef PYFREE
	printf("PYFREE is be defined!\n");	//则不执行
	#endif
	#ifndef PYFREE	//没定义编译条件,,#ifndef与#ifdef相反
	printf("PYFREE is not be defined!\n");	//执行
	#endif
	
	#ifdef GCC_CMD_DEF	//编译命令指定宏定义,gcc test.c -o test.exe -DGCC_CMD_DEF
	printf("GCC_CMD_DEF is define!\n");
	#else
	printf("GCC_CMD_DEF is not define!\n");	
	#endif
	Print_INFO("hi,ANSI define!\n");
	return 0;
}

        通过gcc指令编译执行(本文是win 系统下执行的gcc指令):

        或者建立Makefile文件,添加如下内容:

CX	=	g++

GCC_CMD_DEF := 1    #宏定义
BIN 		:= .
TARGET      := test.exe
FLAGS		:= -static

Include		:= .
source		:= test.c
$(TARGET) :
	$(CX) $(FLAGS) $(source) -I$(Include) -o $(BIN)/$(TARGET)

clean:
	rm  $(BIN)/$(TARGET)

        编译及运行如下:

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

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

相关文章

发烧友实测 | 飞凌嵌入式OKA40i-C开发板试用体验之远程视频监控

本篇试用报告由发烧友zealsoft提供,感谢zealsoft的支持。飞凌嵌入式会在电子发烧友和电路城论坛持续开展开发板有奖试用活动,更有京东E卡等着你!欢迎大家的持续关注。“感谢飞凌嵌入式公司提供了本次OKA40i-C开发板的评测机会。上次我们介绍了…

网络流量传输MTU解析

基本概念 以太网的链路层对数据帧的长度会有一个限制,其最大值默认是1500字节,链路层的这个特性称为MTU,即最大传输单元 Maximum Transmission Unit,最大传输单元,指的是数据链路层的最大payload,由硬件网…

高压放大器在孔道灌浆非线性超声测试中的应用

实验名称:高压放大器在孔道灌浆非线性超声测试中的应用研究方向:无损检测测试目的:超声波作为频率高于20kHz的声波被广泛应用于各类结构的无损检测中,以超声波作为探伤波的无损检测法称为超声波无损检测法,简称超声波法…

嵌入式开发:通过嵌入式虚

嵌入式虚拟化为实现多核处理能力的优势提供了一种可扩展的机制。嵌入式应用中的虚拟化与其企业和桌面应用有许多共同之处。独特的嵌入式使用案例和专业的底层技术为嵌入式开发人员提供了优化性能和响应设计的新机会。在台式机、数据中心以及现在的嵌入式设计中采用多核技术可以…

React hooks之useState用法(一)

系列文章目录 学习React已经有很长的一段时间了,今天决定重新回顾一下跟React相关的一些知识点 文章目录系列文章目录结构如下一、hooks是什么?useState可以能做什么二、如何使用useState()第一步:创建【函数组件&…

java 代码

java 分层架构的由来目录概述需求:设计思路实现思路分析参考资料和推荐阅读Survive by day and develop by night. talk for import biz , show your perfect code,full busy,skip hardness,make a better result,wait for change,challenge Survive. ha…

运维必会:ansible剧本(piaybook)

playbooks 概述以及实例操作 Playbooks 组成部分: Inventory Modules Ad Hoc Commands Playbooks Tasks: 任务,即调用模块完成的某些操作 Variables: 变量 Templates: 模板 Handlers: 处理器,由某时间触发执行的操作 Roles: 角色 YAML 介绍…

Individual Tree Segmentation from LiDAR Point Clouds for Urban Forest Inventory

Abstract 本研究的目的是使用 LiDAR 点云数据开发单棵树级别的自动化城市森林清单的新算法。激光雷达数据包含三维结构信息,可用于估算树高、基高、树冠深度和树冠直径。这使得精确的城市森林库存可以细化到单棵树。与大多数已发布的从 LiDAR 派生的栅格表面检测单…

学了这么久python,不会连自己啥python版本都不知道吧?

人生苦短,我用Python 源码资料电子书:点击此处跳转文末名片获取 查看 Python 版本 我们可以在命令窗口(Windows 使用 winR 调出 cmd 运行框)使用以下命令查看我们使用的 Python 版本: python -V 或 python --version 以上命令执行结果如下: …

Axure 初学者容易涉及的雷区

​工具学习是成为产品经理的一部分学习,工具学习总是伴随着痛苦和煎熬的,因为学习本身就伴随着枯燥和重复。 在未来你的Axure学习可能会出现这些情况呢?还没接触过axure的或者打算进行axure的朋友可能会有疑问。这里根据我们学员学习axure的经…

CentOS8基础篇4:使用U盘备份文件

一、挂载点 所谓的挂载点就是文件系统中存在的一个目录,通常情况下,创建在/mnt目录下,挂载成功后,访问挂载点就是访问新的存储设备。 挂载点应该是空目录,否则原来该挂载点中存在的文件将会被隐藏。而且,…

谷粒学苑第二章前端框架-2.1登录功能

一、vue-admin-template的config模块 vue-admin-template支持多环境,config配置模块提供了dev和prod两种环境。而BASE_API存储的是URL前部分,再拼接上controller的URL,即是完整的URL。修改为自己的协议://ip:port 二、vue-admin-template的s…

.net6API使用SignalR+vue3聊天+WPF聊天

目录 一、.net6api接口 二、vue3前端 三、WPF客户端 此案例分为3部分。首先创建.net6api接口,然后使用前端vue3进行聊天,再使用wpf客户端进行聊天,并且互通聊天。 一、.net6api接口 1.首先建立一个能正常运行的api,然后增加Ch…

redis未授权访问漏洞的三种场景复现以及加固思路

1.redis简介 redis是一个 非常快速 的,开源的,支持网络,可以基于内存,也可以持久化的日志型, 非关系型 的键值对数据库。并提供了多种语言的api。有java,c/c,c#,php,JavaScript,per…

1.Linux编程-gcc编译器

gcc的工作流程 gcc编译器将c源文件到生成一个可执行程序,中间一共经历了四个步骤: 四个步骤并不是gcc独立完成的,而是在内部调用了其他工具,从而完成了整个工作流程, 其中编译最耗时, 因为要逐行检查语法. gcc的工作流程: 1 预处理: cpp预处理器, 去掉注释, 展开头文件, …

Java进程CPU高负载排查步骤

近期发现服务器Java进程负载,超过100%一、采用top命令定位进程登录服务器,执行top命令,查看CPU占用情况,找到进程的pid很容易发现,PID为29706的java进程的CPU飙升到700%多,且一直降不下来,很显然…

python—requests模块详解

一、前言 1、requests简介 requests是一个很实用的Python HTTP客户端库,爬虫和测试服务器响应数据时经常会用到,它是python语言的第三方的库,专门用于发送HTTP请求,使用起来比urllib更简洁也更强大。 2、requests库的安装 方法…

CUDA中的数学方法

CUDA中的数学方法 文章目录CUDA中的数学方法1. Standard FunctionsSingle-Precision Floating-Point FunctionsDouble-Precision Floating-Point Functions2. Intrinsic FunctionsSingle-Precision Floating-Point FunctionsDouble-Precision Floating-Point Functions参考手册…

ROPR:一款功能强大的极速多线程ROPGadget查找工具

关于ROPR ROPR是一款速度极快且功能强大的ROPGadget查找工具,该工具支持多线程运行,可以帮助广大研究人员快速寻找和定位目标ROPGadget。 ROP(Return Oriented Programming),即返回导向编程,而ROPGadget是…

算法训练营DAY52|1143.最长公共子序列、1035.不相交的线、53. 最大子序和

前两道题思路是一模一样的,但是需要认真理解,最后一道虽然思路不算难,但是需要注意的细节一点不少。 1143. 最长公共子序列 - 力扣(LeetCode)https://leetcode.cn/problems/longest-common-subsequence/最长公共子序列…