当前位置: 首页 > news >正文

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

一、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.exetypedef 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 PYFREEprintf("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_DEFprintf("GCC_CMD_DEF is define!\n");#elseprintf("GCC_CMD_DEF is not define!\n");	#endifPrint_INFO("hi,ANSI define!\n");return 0;
}

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

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

CX	=	g++GCC_CMD_DEF := 1    #宏定义
BIN 		:= .
TARGET      := test.exe
FLAGS		:= -staticInclude		:= .
source		:= test.c
$(TARGET) :$(CX) $(FLAGS) $(source) -I$(Include) -o $(BIN)/$(TARGET)clean:rm  $(BIN)/$(TARGET)

        编译及运行如下:

相关文章:

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

一、c/c宏定义的来源 宏定义,就是用一个标识符来表示一个字符串,如果在后面的代码中出现了该标识符,那么就全部替换成指定的字符串。通常c/c宏定义这几处出处: 1)最常见的就是来自于开发者编码过程中采用宏定义命令“#…...

SaaS的阴暗面:网络攻击武器化、平民化

你不一定懂编程,甚至都看不懂几行代码,但依然能成为杀伤力十足的黑客,这就是现阶段不少网络攻击的特点:不需要掌握娴熟的技术或代码,仅仅利用成熟的武器化工具,就能通过简单的“一键操作”,对目…...

『CV学习笔记』图像处理透视变换(Python+Opencv)

图像处理透视变换(Opencv) 文章目录 一. 透视变换定义二. 代码实现2.1. order_points函数2.2. four_point_transform函数2.3. 程序主函数三. 参考文献一. 透视变换定义 【图像处理】透视变换 Perspective Transformation原理:https://blog.csdn.net/xiaowei_cqu/article/detai…...

使用Fetch时,post数据时,后端接收的Content-Type为text/plain

在使用 Fetch做一个前端的post请求时,直接从网上抄了一段代码 export async function postData(url, data){const response await fetch(url, {method: POST, // *GET, POST, PUT, DELETE, etc.mode: no-cors, // no-cors, *cors, same-originheaders: { Content-…...

大型智慧校园系统源码 智慧校园源码 Android电子班牌源码

一款针对中小学研发的智慧校园系统源码,智慧学校源码带电子班牌、人脸识别系统。系统有演示,可正常上线运营正版授权。 私信了解更多! 技术架构: 后端:Java 框架:springboot 前端页面:vue e…...

【MySQL】你知道的MySQL中的集合函数有哪些呢?

集合函数排名AVG()函数COUNT()函数SUM()函数MAX()函数 和 MIN()函数总结大家好,我是小冷。 上一篇写了 看看ChatGPT是如何回答面试官的问题的? 地址是&#x…...

科目一罚款题

第一种:20~200元关键字:遮车牌、改信息1、故意使用物品遮挡号牌的2、号牌刮花严重,掉漆未处理继续使用的3、驾驶人联系方式、地址等信息发生变化时,没有及时申报变更信息的4、驾驶人不在现场、或虽在现场但拒绝立即驶离&#xff0…...

Maven中央仓库地址大全

一、仓库地址配置方式关于 Maven 远程仓库地址的配置方式有两种:第1种:直接在项目的 pom.xml 文件中进行修改(不推荐,尤其是在多人协助的开发过程中非常的费事费力);第2种:将 Maven 的远程仓库统…...

人工智能的未来———因果推理what if 第11章(统计模型) 文章解读

我们在观察数据当中,一般使用样本均值去估计目标人群的均值 在所有情况都是理想的情况下: 平均因果效应...

百度贴吧发帖软件如何发布?

百度贴吧发帖软件如何发布?贴吧软件发帖顶帖视频教学,防删图技术视频教学#贴吧发帖#贴吧顶帖 大家好,今天给大家讲一下一个贴软件发电机顶帖的视频教学。先给大家讲一下软件一个发帖。今天我们用的一个软件叫做神机。神机我们现在看到的软件…...

如何成为java架构师?2023版Java架构师学习路线总结完成,真实系统有效,一切尽在其中

导读 从初级Java工程师成长为Java架构师,你需要走很长的路,很多有计划的人在学习之初就在做准备。你知道Java架构师学习路线该怎么走吗?成为一个优秀的Java架构师究竟需要学什么?接下来就跟小编一起揭晓答案。 架构师是一个充满挑战的职业&#xff0…...

tkinter 实现选择文件夹或者多个文件

import tkinter as tk from tkinter import filedialog def select_file(): # 单个文件选择 selected_file_path filedialog.askopenfilename() # 使用askopenfilename函数选择单个文件 select_path.set(selected_file_path) def select_files(): # 多个文件…...

深圳活动会议媒体邀约,电视台,网媒媒体资源

传媒如春雨,润物细无声,大家好,一,您可以考虑以下几种方式邀请媒体参加深圳的活动会议:1,直接联系:找到相关媒体的联系人信息,发送邀请函或电话邀请。2,通过公关公司&…...

JDBC与Druid连接池

1 什么是jdbc? JDBC(java database connectivity )是Java语言连接操作关系型数据库的一套解决方案,屏蔽了底层各数据库不同的差异。 具体是通过sun公司定义的统一的一套API【标准接口】来实现解决差异, 具体实现是各大数据库厂商…...

java易错题锦集二

源码 补码 int i 5; int j 10; System.out.println(i ~j);有个公式,-n~n1 另一种解题思路 ~代表对n按位取反 10的源码是: 00000000 00000000 00000000 1010 所以对10按位取反就是 11111111 11111111 11111111 0101 由于计算机中-1表示为 11111111 11111111 111…...

11.hadoop系列之MapReduce框架原理之InputFormat数据输入

我们先简要了解下InputFormat输入数据 1.数据块与数据切片 数据块: Block在HDFS物理上数据分块,默认128M。数据块是HDFS存储数据单位 数据切片: 数据切片只是在逻辑上对输入进行分片,并不会物理上切片存储。数据切片是MapReduce…...

LoadRunner

目录 为什么需要性能测试 性能测试实施流程 常见的性能测试指标 性能测试分类 1、一般性能测试 2、负载测试 3、压力测试 LoadRunner LoadRunner包括三个组件 VUG Controller Analysis 一个网站或者app的性能差,用户的使用体验就会很差 常见的性能问题&a…...

笔试题-2023-芯原-数字前端设计or验证【纯净题目版】

回到首页:2023 数字IC设计秋招复盘——数十家公司笔试题、面试实录 推荐内容:数字IC设计学习比较实用的资料推荐 题目背景 笔试时间:2022.08.28应聘岗位:数字前端设计/验证工程师笔试时长:60min笔试平台:nowcoder牛客网题目类型:企业知识题(2道)、智力题(5题)、技术…...

【软件测试】资深测试工程师说:你真的能做好bug分析吗?

目录:导读前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结(尾部小惊喜)前言 bug报告&#xff0c…...

NLP模型检查英语语法错误

当NLP模型产生了语法错误,怎么办?比如,He wants that you send him an email.没关系,现在可以像小时候的英语老师,改作文一样简单。只要装上一个专门纠正语法错误的库就可以,还是毫秒钟就可揪出来的那种。这…...

java调用dll出现unsatisfiedLinkError以及JNA和JNI的区别

UnsatisfiedLinkError 在对接硬件设备中,我们会遇到使用 java 调用 dll文件 的情况,此时大概率出现UnsatisfiedLinkError链接错误,原因可能有如下几种 类名错误包名错误方法名参数错误使用 JNI 协议调用,结果 dll 未实现 JNI 协…...

大语言模型如何处理长文本?常用文本分割技术详解

为什么需要文本分割? 引言:为什么需要文本分割?一、基础文本分割方法1. 按段落分割(Paragraph Splitting)2. 按句子分割(Sentence Splitting)二、高级文本分割策略3. 重叠分割(Sliding Window)4. 递归分割(Recursive Splitting)三、生产级工具推荐5. 使用LangChain的…...

Springcloud:Eureka 高可用集群搭建实战(服务注册与发现的底层原理与避坑指南)

引言:为什么 Eureka 依然是存量系统的核心? 尽管 Nacos 等新注册中心崛起,但金融、电力等保守行业仍有大量系统运行在 Eureka 上。理解其高可用设计与自我保护机制,是保障分布式系统稳定的必修课。本文将手把手带你搭建生产级 Eur…...

Python如何给视频添加音频和字幕

在Python中,给视频添加音频和字幕可以使用电影文件处理库MoviePy和字幕处理库Subtitles。下面将详细介绍如何使用这些库来实现视频的音频和字幕添加,包括必要的代码示例和详细解释。 环境准备 在开始之前,需要安装以下Python库:…...

爬虫基础学习day2

# 爬虫设计领域 工商:企查查、天眼查短视频:抖音、快手、西瓜 ---> 飞瓜电商:京东、淘宝、聚美优品、亚马逊 ---> 分析店铺经营决策标题、排名航空:抓取所有航空公司价格 ---> 去哪儿自媒体:采集自媒体数据进…...

2023赣州旅游投资集团

单选题 1.“不登高山,不知天之高也;不临深溪,不知地之厚也。”这句话说明_____。 A、人的意识具有创造性 B、人的认识是独立于实践之外的 C、实践在认识过程中具有决定作用 D、人的一切知识都是从直接经验中获得的 参考答案: C 本题解…...

JVM虚拟机:内存结构、垃圾回收、性能优化

1、JVM虚拟机的简介 Java 虚拟机(Java Virtual Machine 简称:JVM)是运行所有 Java 程序的抽象计算机,是 Java 语言的运行环境,实现了 Java 程序的跨平台特性。JVM 屏蔽了与具体操作系统平台相关的信息,使得 Java 程序只需生成在 JVM 上运行的目标代码(字节码),就可以…...

JavaScript 数据类型详解

JavaScript 数据类型详解 JavaScript 数据类型分为 原始类型(Primitive) 和 对象类型(Object) 两大类,共 8 种(ES11): 一、原始类型(7种) 1. undefined 定…...

4. TypeScript 类型推断与类型组合

一、类型推断 (一) 什么是类型推断 TypeScript 的类型推断会根据变量、函数返回值、对象和数组的赋值和使用方式,自动确定它们的类型。 这一特性减少了显式类型注解的需要,在保持类型安全的同时简化了代码。通过分析上下文和初始值,TypeSc…...

前端中slice和splic的区别

1. slice slice 用于从数组中提取一部分元素,返回一个新的数组。 特点: 不修改原数组:slice 不会改变原数组,而是返回一个新的数组。提取数组的部分:slice 会根据指定的开始索引和结束索引提取数组的一部分。不包含…...