C语言学习笔记——程序环境和预处理

news2024/11/15 11:16:00

目录

前言

一、程序环境

1. 翻译环境    

1.1 主要过程  

1.2 编译过程

2. 运行环境

二、预处理

1. 预定义符号

2. #define 

2.1 #define定义标识符

2.2 #define定义宏

2.3 命名约定和移除定义

 3. 条件编译

 4. 文件包含

结束语


前言

       每次我们写完代码运行的时候都会弹出来一个黑框框,这个黑框框实际上是一个可执行程序(.exe文件)。那么代码是如何被变成一个可执行文件的呢?其实这就是编译器所做的事,一起来了解了解吧。


一、程序环境

1. 翻译环境    

1.1 主要过程  

        代码不可能凭空运行,只有可执行程序才能在计算机上运行。因此,在翻译环境下,代码会经过编译,链接形成一个可执行程序。如下图,组成程序的各个源文件经过编译器的编译形成各自的目标文件,再由链接器链接形成一个可执行程序。在链接过程之中,链接器会从C语言标准库中引入程序中所用到的函数。  

1.2 编译过程

       上述过程中的编译过程又可细分为预编译(预处理)、编译。翻译三个阶段。

预处理:这一阶段主要用来执行各种预处理指令,例如#define定义标识符常量,宏。之前在介绍枚举常量时与#define定义的标识符常量进行对比,标识符常量无法进行调试,这就是因为标识符常量在预处理时就已经被替换为常量值。

编译:编译阶段主要对代码的语法,词法,语义进行分析,检查。确保代码无语法错误后,将各个文件中的符号(函数名,全局变量等)进行汇总,便于跨文件调用。

翻译:计算机只能识别二进制的代码,因此在链接之前,编译器需要将C语言代码经汇编代码翻译为二进制代码 ,并形成目标文件。

2. 运行环境

       程序在执行时必须载入内存中,这一步一般由操作系统完成。若在无操作系统的环境中,则需手动完成。程序开始执行会调用main函数,这时候需使用一个运行的堆栈用以存储函数的局部变量和地址。程序也可用静态区存储静态变量和全局变量。最后,程序正常结束(也有可能异常结束)。

二、预处理

1. 预定义符号

__FILE__      //进行编译的源文件
__LINE__     //文件当前的行号
__DATE__    //文件被编译的日期
__TIME__    //文件被编译的时间

       以上列举的预定义符号为C语言内置的符号,使用效果如下图。

2. #define 

2.1 #define定义标识符

       #defien定义标识符的规则为”#define + 标识符名称 + 内容“。

       #define定义的标识符会在预处理阶段直接被替换为其内容,例如下方为一段标识符的定义和使用。在预处理阶段,"MAX"被替换为"100",即将100赋值给变量max。

#define MAX 100

int max = MAX;

        同样地,#define可以以任何数据为内容,因此我们可以对switch语句中的case,break进行如下改造。改造的依据是C语言外的其它语言中部分语言的switch语句不需要使用break结束情况。当其它语言的程序员写C程序时可能会使用如下方式写switch语句。

#define CASE break;case

switch(x)
{
    case 1:printf("%d\n",1);
    CASE 2:printf("%d\n",2);
    CASE 3:printf("%d\n",3);
    CASE 4:printf("%d\n",4);
    default:printf("%d\n",0);
            break;
}

       这段代码在预处理阶段会将CASE替换为break;case,将各个情况分开并且不用在每种情况后手动加上break。

       当然,由于标识符是直接替换内容,使用不熟练可能会造成逻辑错误,例如下方代码

#define A 3+3

int a = 2*A;

       很多新手会认为此时的a的值为12,因为A就是“3+3=6”嘛,那么2*A就是12。这段代码的中的a的值应该为9。由于A为标识符,因此代码中的A直接替换为标识符内容"3+3",即"int a = 2*3+3",结果为9。若想达到预期效果,则需加上括号,如下

#define A (3+3)

int a = 2*A;

2.2 #define定义宏

       #define定义宏的规则与标识符相似:"#define + 宏名称(宏参数) + 内容"。

       宏的形式与函数相似,都需要传入参数,不同的是,宏的参数是直接替换到宏的内容中,而函数则是使用参数的值。例如下方代码

       这段代码中,宏的计算为"2+1*3=5",而函数的计算为"3*3=9"。

与函数相比宏的优点

1、 函数的调用需要在栈上开辟空间,传参,返回值,销毁空间。这些准备工作可能比实际计算所需的时间要多得多,相比之下宏在处理一些简单计算时的速度更快。

2、 另外,类似于标识符,宏的参数可以为任何内容,这是函数无法做到的,函数的参数必须是声明的指定类型的数据,因此,宏的使用较函数更加灵活,例如向宏中传入一个关键字,这是函数无法做到的。

宏的缺点

1. 每次使用宏的时候,一份宏定义的代码将插入到程序中。除非宏比较短,否则可能大幅度增加程序 的长度。

2. 宏是没法调试的。

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

4. 宏可能会带来运算符优先级的问题,导致程容易出现错。

2.3 命名约定和移除定义

命名约定:由于宏在使用时与函十分相似,因此人们约定在定义宏名称时全部用大写字母而在定义函数名时不全用大写字母。

移除定义:使用"#undef"指令移除一个宏定义,如下图。

 3. 条件编译

       当有些代码需要在特定的条件下执行,其它条件下不执行时,就需要用到条件编译。条件编译类似于if分支语句,区别是条件编译指令是在预处理阶段执行,而分支语句是在函数内执行。这里列举一条常见的条件编译指令

#if 常量表达式
 //满足条件执行内容
#endif

       这段指令就是在常量表达式为真时执行#if与#endif之间的代码。例如

        第一段代码由于常量表达式为真,故定义函数func并成功调用。而第二段代码由于常量表达式为假,函数func未定义,故调用失败报错。

       与if分支语句类似,#if条件编译指令也可出现分支,如下

#if 常量表达式
 //...
#elif 常量表达式
 //...
#else
 //...
#endif

       另外,这里有两组判断表达式是否被定义的条件编译指令。若"symbol"已定义,则上面两段指令的内容被执行,若未定义则下面两段指令的内容被执行。

#if defined(symbol)
#ifdef symbol

#if !defined(symbol)
#ifndef symbol

 4. 文件包含

       当我们需要使用C语言标准库里的函数时,或是使用自己所写的头文件里的内容时,必须包含相应的头文件。包含头文件有以下两种写法(以"stdio.h"为例)

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

       其中使用尖括号"<>"代表直接从C标准库中查找该头文件,而使用引号则代表从程序内部和C标准库中查找。因此,在我们包含C标准库中的头文件时通常使用尖括号以提高效率。

       在大型项目中,多个程序员可能会多次调用同一个头文件,这样容易导致代码的运行效率降低。我们可以使用条件编译的方式来解决这个问题,如下方代码

#ifndef __TEST_H__
#define __TEST_H__
//头文件的内容
#endif   

       这样的写法就可以避免同一个头文件的重复调用,大大提高项目的效率。另外,在头文件开头加上以下代码也可达到相同效果。

#pragma once

结束语

       程序环境和预处理对于新手来说运用不多,但也算是一块比较重要的内容,有助于新手理解程序的编译形成过程,熟悉预处理操作。

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

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

相关文章

刷题28-有效的变位词

32-有效的变位词 解题思路&#xff1a; 注意变位词的条件&#xff0c;当两个字符串完全相等或者长度不等时&#xff0c;就不是变位词。 把字符串中的字符映射成整型数组&#xff0c;统计每个字符出现的次数 注意数组怎么初始化&#xff1a; int [] s1new int[26]代码如下&a…

C语言数据结构(一)—— 数据结构理论、线性表【动态数组、链表(企业版单向链表)】

数据结构理论1.1 数据数据&#xff1a;是描述客观事物的符号&#xff0c;是计算机中可以操作的对象&#xff0c;是能被计算机识别&#xff0c;并输入给计算机处理的符号集合。数据不仅仅包括整型、实型等数值类型&#xff0c;还包括字符及声音、图像、视频等非数值类型。1.2数据…

决策树、随机森林、极端随机树(ERT)

声明&#xff1a;本文仅为个人学习记录所用&#xff0c;参考较多&#xff0c;如有侵权&#xff0c;联系删除 决策树 通俗来说&#xff0c;决策树分类的思想类似于找对象。现想象一个女孩的母亲要给这个女孩介绍男朋友&#xff0c;于是有了下面的对话&#xff1a; 女儿&#x…

C++17 nodiscard标记符

文章目录前言弃值表达式nodiscard标记符函数非弃值声明类/枚举类/结构 非弃值声明返回类引用与类指针前言 在C 17中引入了一个标记符nodiscard&#xff0c;用于声明一个 “非弃值(no-discard)表达式”。那么在开始之前&#xff0c;我们需要了解一下什么是弃值表达式。 弃值表…

LearnOpenGL-入门-着色器

本人刚学OpenGL不久且自学&#xff0c;文中定有代码、术语等错误&#xff0c;欢迎指正 我写的项目地址&#xff1a;https://github.com/liujianjie/LearnOpenGLProject LearnOpenGL中文官网&#xff1a;https://learnopengl-cn.github.io/ 文章目录着色器GLSL数据类型输入与输…

SpringMVC - 12 - 注解配置SpringMVC(完全注解开发)

文章目录注解配置SpringMVC1、创建初始化类WebInit&#xff0c;代替web.xml2、创建SpringConfig配置类&#xff0c;代替spring的配置文件3、创建WebConfig配置类&#xff0c;代替SpringMVC的配置文件4、TestController进行测试注解配置SpringMVC 使用配置类和注解代替web.xml和…

友云生态全球高峰论坛暨长沙首届私董会隆重召开

由友云生态主办的“赢在巅峰决策未来”在友云生态全球新消费、新金融、新资本高峰论坛暨长沙首届私董会于2023年2月22日隆重召开&#xff01;本场私董会以“赢在巅峰 决策未来”为主题&#xff0c;从思维、定位、格局、布局四个角度探讨未来发展的全新动能。作为行业标杆性盛会…

网络应用之HTTP响应报文

HTTP响应报文学习目标能够知道HTTP响应报文的结构1. HTTP响应报文分析HTTP 响应报文效果图:响应报文说明:--- 响应行/状态行 --- HTTP/1.1 200 OK # HTTP协议版本 状态码 状态描述 --- 响应头 --- Server: Tengine # 服务器名称 Content-Type: text/html; charsetUTF-8 # 内容类…

【RabbitMQ笔记06】消息队列RabbitMQ七种模式之Topics主题模式

这篇文章&#xff0c;主要介绍消息队列RabbitMQ七种模式之Topics主题模式。 目录 一、消息队列 1.1、主题模式&#xff08;Topics&#xff09; 1.2、案例代码 &#xff08;1&#xff09;引入依赖 &#xff08;2&#xff09;编写生产者 &#xff08;3&#xff09;编写消费…

MapReduce 性能优化

MapReduce用于大规模数据集的并行运算&#xff0c;所以性能优化也是需要重点关注的内容&#xff0c;下面是我在学习过程中&#xff0c;对于MapReduce 性能优化的点&#xff0c;分享大家学习&#xff0c;enjoy~~ MapReduce的运行流程 以上是MapReduce的运行流程&#xff0c;所以…

Zebec社区上线ZIP-2(地平线升级行动)提案,海量激励将被释放

此前&#xff0c;Zebec社区在上线了投票治理系统Zebec Node后&#xff0c;曾上线了首个提案ZIP-1&#xff0c;对 Nautilus Chain 的推出进行了投票&#xff0c;作为 Zebec Chain 上线前的“先行链”&#xff0c;该链得到了社区用户的欢迎&#xff0c;投通过票的比例高达98.3%。…

负载均衡:LVS 笔记(二)

文章目录LVS 二层负载均衡机制LVS 三层负载均衡机制LVS 四层负载均衡机制LVS 调度算法轮叫调度&#xff08;RR&#xff09;加权轮叫调度&#xff08;WRR&#xff09;最小连接调度&#xff08;LC&#xff09;加权最小连接调度&#xff08;WLC&#xff09;基于局部性的最少链接调…

Apache Hadoop、HDFS介绍

目录Hadoop介绍Hadoop集群HDFS分布式文件系统基础文件系统与分布式文件系统HDFS简介HDFS shell命令行HDFS工作流程与机制HDFS集群角色与职责HDFS写数据流程&#xff08;上传文件&#xff09;HDFS读数据流程&#xff08;下载文件&#xff09;Hadoop介绍 用Java语言实现开源 允许…

SpringBoot:SpringBoot简介与快速入门(1)

SpringBoot快速入门1. SpringBoot简介2. SpringBoot快速入门2.1 创建SpringBoot项目&#xff08;必须联网&#xff0c;要不然创建失败&#xff0c;在模块3会讲到原因&#xff09;2.2 编写对应的Controller类2.3 启动测试3. Spring官网构建工程4. SpringBoot工程快速启动4.1 为什…

自学大数据的第一天

默认跳过基础部分,直接搞集群的部分,期间用到的linux基础默认大伙都会了(不会的话可以现用现查) Hadoop集群搭建 集群特点: 1,逻辑上分离~集群之间没有依赖,互不影响 2,某些进程往往部署在一台服务器上,但是属于不同的集群 3,MapReduce 是计算框架,代码层面的处理逻辑 集群的…

mindspore的MLP模型(多层感知机)

导入模块 import hashlib import os import tarfile import zipfile import requests import numpy as np import pandas as pd import mindspore import mindspore.dataset as ds from mindspore import nn import mindspore.ops as ops import mindspore.numpy as mnp from …

Python 内置函数eval()

Python 内置函数eval() eval(expression, globalsNone, localsNone) 函数用来执行一个字符串表达式&#xff0c;并返回表达式的值。 expression: 字符串表达式。global: 可选&#xff0c;globals必须是一个字典。locals: 可选&#xff0c;locals可以是任何映射对象。 示例 &…

微信小程序开发【壹】

随手拍拍&#x1f481;‍♂️&#x1f4f7; 日期: 2023.02.24 地点: 杭州 介绍: 2023.02.24上午十点&#xff0c;路过学院的教学楼时&#x1f3e2;&#xff0c;突然看见了一团粉红色。走进一看是一排梅花&#x1f338;&#xff0c;赶在它们凋零前&#xff0c;将它们定格在我的相…

QML 第一个应用程序Window

1.创建QML工程 新建文件或者项目-->选择Qt Quick Application 然后生成了一个默认的Window 2.main.cpp中如何加载的qml文件 QQmlApplicationEngine提供了从单个QML文件加载应用程序的便捷方式。 此类结合了QQmlEngine和QQmlComponent&#xff0c;以提供一种方便的方式加载…

用 Python 画如此漂亮的插图 ,So easy

人生苦短&#xff0c;快学Python&#xff01; 今天我们进行一次实战案例分享&#xff0c;以全球预期寿命与人均 GPD数据为例&#xff0c;写一篇 Python 中漂亮散点图的快速指南。除了正常的数据清洗/处理、还会进行简单的统计分析&#xff0c;实现数据处理-统计分析-可视化一条…