C语言基础:预处理指令

2024-09-03 11:32
文章标签 语言 基础 指令 预处理

本文主要是介绍C语言基础:预处理指令,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文基于一个简单的Simulink生成的代码,研究MBD中比较常见的一些预处理指令。

文章目录

  • 1 C语言预处理指令
  • 2 #include
  • 3 #define
  • 4 #ifndef...#endif
  • 5 #pragma
  • 6 总结

1 C语言预处理指令

预处理是C语言编译过程中最初的一步,博主理解,预处理指令就是在预处理阶段告诉编译器要做些啥。

预处理指令很好辨别,一个单词以#开头就是预处理指令。如果用Notepad++打开,那么咖啡色的一大串就是预处理指令的东西了。
在这里插入图片描述
下面会简单讲讲博主目前在MBD的工作中比较常见的预处理指令,并做一些通俗的解释。

2 #include

#include预处理指令后面会跟着一个头文件。系统库的头文件和用户自己写的头文件长得是不一样的。例如下面的<stdio.h>是系统头文件,"myheader.h"是用户头文件。博主刚上班的时候就没分清楚,还好自己百度了一下,没拿去问别人,不然丢人就丢到家了。

#include <stdio.h>
#include "myheader.h"

#include其实就是告诉编译器,把这个头文件的内容添加到当前的源文件中。头文件中一般都会有一些变量声明,函数声明什么的,供各种C文件调用。

如果不做特殊配置,Embedded Coder就会生成一个和源文件同名的头文件。这个头文件中包含源文件的全局变量声明,函数声明,也会包含其他头文件。

在这里插入图片描述

3 #define

1.#define是宏定义的意思,简单理解就是换个马甲。比方说下面的宏定义:

#define FILE_SIZE 42

这里理解就是,编译器遇到了这一行宏定义了以后,就知道后面如果遇到了FILE_SIZE,就把这一串字符换成42这个数字来处理。

2.当然,除了赋予数字一个含义以外,也可以把字符换成字符。

#define name1 name2

这里就是遇到了name1,就把他当成name2。适用于模型开发中,假如两个接口都是同一个变量,但是又因为某种原因不能改名字,那就手写个宏定义来转换一下。

3.还有另一种情况,#define后面只跟了一个名字,没有第二个名字。

#define name1

这代表着告诉编译器,有这么一个宏,一般会配合其他预处理指令使用,比方说条件编译。

4 #ifndef…#endif

Embedded Coder生成的头文件中,总是能在一开始(除了注释)就有一个#ifndef,然后结尾有一个#endif。
在这里插入图片描述
这是一种条件编译。#ifndef后面跟了个宏,意思是如果没有定义这个宏,就编译#endif之前的一段代码;以前有定义过这个宏,就把这整段跳过去。在上述截图中,使用#ifndef来做了个防止重复编译本头文件内容的机制。

这个机制的意义在于,当第一次遇到了C代码包含了这个头文件的时候,没有定义过RTW_HEADER_rtmodel_h_这个宏,然后编译器就会加载本头文件中endif之前的内容。如果下一次别的C代码中也包含了这个头文件,因为第一次定义了宏,所以后面就不会再去加载了,省的做重复的事情。

(抱歉博主表达能力有限,这部分感觉自己理解的以及讲的都不清楚。)

相对应的,也有一个#ifdef预处理指令,和#ifndef意思相反,是定义过宏的情况下才加载内容。还有#else,#elif。

手写代码中,会用这种条件编译来做一个“开关”,以方便调试不同情况下用哪一大段代码。

5 #pragma

#pragma指令比较复杂,不过博主目前再MBD中遇到的只有#pragma section,用来控制代码或者全局变量在控制器中存放的地址。

这部分比较复杂,涉及到链接的很多知识,先留个坑,以后有了深入的理解以后再来记录下来。以免现在说错了会误人子弟。

6 总结

MBD中比较常用的预处理指令也就上面这些了。

>>返回个人博客总目录

这篇关于C语言基础:预处理指令的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1132843

相关文章

使用SQL语言查询多个Excel表格的操作方法

《使用SQL语言查询多个Excel表格的操作方法》本文介绍了如何使用SQL语言查询多个Excel表格,通过将所有Excel表格放入一个.xlsx文件中,并使用pandas和pandasql库进行读取和... 目录如何用SQL语言查询多个Excel表格如何使用sql查询excel内容1. 简介2. 实现思路3

Go语言实现将中文转化为拼音功能

《Go语言实现将中文转化为拼音功能》这篇文章主要为大家详细介绍了Go语言中如何实现将中文转化为拼音功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 有这么一个需求:新用户入职 创建一系列账号比较麻烦,打算通过接口传入姓名进行初始化。想把姓名转化成拼音。因为有些账号即需要中文也需要英

Go语言使用Buffer实现高性能处理字节和字符

《Go语言使用Buffer实现高性能处理字节和字符》在Go中,bytes.Buffer是一个非常高效的类型,用于处理字节数据的读写操作,本文将详细介绍一下如何使用Buffer实现高性能处理字节和... 目录1. bytes.Buffer 的基本用法1.1. 创建和初始化 Buffer1.2. 使用 Writ

深入理解C语言的void*

《深入理解C语言的void*》本文主要介绍了C语言的void*,包括它的任意性、编译器对void*的类型检查以及需要显式类型转换的规则,具有一定的参考价值,感兴趣的可以了解一下... 目录一、void* 的类型任意性二、编译器对 void* 的类型检查三、需要显式类型转换占用的字节四、总结一、void* 的

MySQL中my.ini文件的基础配置和优化配置方式

《MySQL中my.ini文件的基础配置和优化配置方式》文章讨论了数据库异步同步的优化思路,包括三个主要方面:幂等性、时序和延迟,作者还分享了MySQL配置文件的优化经验,并鼓励读者提供支持... 目录mysql my.ini文件的配置和优化配置优化思路MySQL配置文件优化总结MySQL my.ini文件

C语言线程池的常见实现方式详解

《C语言线程池的常见实现方式详解》本文介绍了如何使用C语言实现一个基本的线程池,线程池的实现包括工作线程、任务队列、任务调度、线程池的初始化、任务添加、销毁等步骤,感兴趣的朋友跟随小编一起看看吧... 目录1. 线程池的基本结构2. 线程池的实现步骤3. 线程池的核心数据结构4. 线程池的详细实现4.1 初

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验