C语言-预处理与库

预处理、动态库、静态库

1. 声明与定义分离

一个源文件对应一个头文件

注意:

  • 头文件名以 .h 作为后缀
  • 头文件名要与对应的原文件名 一致

例:

源文件:01_code.c

#include <stdio.h>
int num01 = 10;
int num02 = 20;
void add(int a, int b)
{
    int sum = a + b;
    printf("%d + %d = %d\n", a, b, sum);
}

void mul(int a, int b)
{
    int mul = a + b;
    printf("%d + %d = %d\n", a, b, mul);
}

头文件:01_code.h

extern int num01;
extern int num02;
extern void add(int, int);
extern void mul(int, int);

测试文件:test.c

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

int main(int argc, char const *argv[])
{
    printf("num01 = %d\n", num01);
    printf("num02 = %d\n", num02);
    add(10, 20);
    mul(10, 20);
    return 0;
}

编译:

  • 命令:gcc test.c 01_code.c ./a.out

  • 输出:

    num01 = 10
    num02 = 20
    10 + 20 = 30
    10 + 20 = 30
    

2. 预处理

2.1 c语言编译过程

gcc -E hello.c -o hello.i 	1、预处理
gcc -S hello.i –o hello.s 	2、编译
gcc -c hello.s -o hello.o 	3、汇编
gcc hello.o -o hello_elf 	4、链接

1、预编译

  • .c 中的头文件展开宏展开
  • 生成的文件是 .i 文件

2、编译

  • 将预处理之后的 .i 文件生成 .s 汇编文件

3、汇编

  • .s 汇编文件 生成 .o 文件

4、链接

  • .o 文件 链接成目标文件(即可执行文件)

预编译包含

展开头文件
定义头文件
选择性编译

注意:
	预编译的内容以 # 开头

2.2 include

作用:展开头文件

语法:

#include <> 

尖括号包含的头文件, 在 系统指定的路径下 找头文件

#:表示预编译

#include ""

双引号 包含头文件,先在当前目录下找 头文件,找不到,再到系统指定的路径下找

注意:

1、include 经常用来包含头文件,可以包含 .c 文件,但是大家不要包含 .c 因为 include 包含的文件会在预编译被展开,如果一个.c 被包含多次,展开多次,会导致函数重复定义。所以不要包含.c 文件

2、预处理只是对 include 等预处理操作进行处理,并 不会进行语法检查,这个阶段有语法错误也不会报错,第二个阶段即 编译阶段才进行语法检查

例:

#include "01_code.h"

//等价于 下面, 即在源文件中展开下面代码

extern int num01;
extern int num02;
extern void add(int, int);
extern void mul(int, int);

2.2 宏:define

作用:在预处理 处理定义 类似于 变量函数的东西。即:宏是在预编译的时候进行替换 。

2.2.1 不带参宏

语法:

#define 宏名 值   //宏定义

#undef 宏名       //取消宏定义

注意:

1、如果定义该类型的宏(不带参的宏),值可以省略

2、无需分号结束

3、在 宏定义后,取消定义前 可以使用

4、只能在 当前文件中 使用

例:

#include <stdio.h>
#define PI 3.14
int main(int argc, char const *argv[])
{
    printf("pi = %f\n", PI);
#undef PI  //取消宏定义
    return 0;
}
2.2.2 带参宏

语法:

#define  宏名(形参)  体

注意:

1、形参没有数据类型

2、带参宏带参函数的区别

  • 宏:在预编译时 对其进行 替换,如果一个文件中多次使用宏,那意味着要替换多次,此时就需占用内存,所以占据的内存多

    • 产生的预编译时期
    • 占内存多
    • 速度快
  • 函数:在程序运行时在代码区存储一份,每次调用该函数都需在代码区寻找,将其放入栈内存中(压栈),当函数执行完毕后,从栈中移除(弹栈)

    • 产生在运行时
    • 占内存少

例:

#include <stdio.h>
#define ADD(a, b) a+b
#define MUL(a, b) a*b
#define MUL02(a, b) (a)*(b)
int main(int argc, char const *argv[])
{
    int sum = ADD(20, 30);
    printf("sum=%d\n", sum);

    int mul = MUL(20, 30);
    printf("mul=%d\n", mul);

    int mul02 = MUL(20+10, 30+10);  //20 + 10 * 30 +10
    printf("mul=%d\n", mul02);

    int mul03 = MUL02(20+10, 30+10); //(20 + 10) * (30 + 10)
    printf("mul=%d\n", mul03);

    return 0;
}

在这里插入图片描述

2.2.3 小结
  • 宏就是在预编译时期对其进行替换

  • 不带参宏替换的是一个值

  • 带参宏替换的是一段代码

2.3 选择性编译

作用:选择代码是否被编译

语法:
在这里插入图片描述

例1:判断存在

优点:节省内存,只加载需要的部分

#include <stdio.h>

int main(int argc, char const *argv[])
{
    #ifdef  XXX
        printf("有定义宏名为XXX的宏\n");
    #else
        printf("没定义宏XXX\n");
    #endif

    return 0;
}

在这里插入图片描述
在这里插入图片描述

编译时定义宏:

在这里插入图片描述

例2:判断不存在,和头文件配合使用,防止多次引用头文件

#include <stdio.h>
#include "04_test.h"
#include "04_test.h"
int main(int argc, char const *argv[])
{
    #ifndef YYY
        printf("1111\n");
    #else
        printf("2222\n");
    #endif
    return 0;
}

头文件:04_test.h

#ifndef TEST
#define TEST
extern int num;
//...
#endif

#ifndef 使用含义:

1、第一次引用头文件,没有定义TEST宏,然后定义,再写头文件内容;

2、假如再次引用头文件时,第一次已经定义过TEST宏了,所以直接结束,啥也不干。

源码写法:

在这里插入图片描述

例3:判断是否成立

#include <stdio.h>
int main(int argc, char const *argv[])
{
    #if ScORE > 85
        printf("A\n");
    #elif ScORE > 70
        printf("B\n");
    #elif ScORE >= 60
        printf("c\n");
    #else
        printf("D\n");
    #endif
    return 0;
}

在这里插入图片描述

3. 库

概念:库也叫代码库,可以把一个些目标文件合并在一起方便使用。

3.1 分类

静态库

动态库

静态库、动态库的区别:

在这里插入图片描述

注意:

  • 程序中引入的文件在动态库与静态库同时存在两份
  • 静态编译程序引入静态库中的该文件
  • 动态编译程序引入动态库中的该文件

3.2 编译命令

动态编译:

gcc 源文件名 -o 生成的可执行文件名

静态编译:

gcc -static 源文件名 -o 生成的可执行文件名

3.3 静态库

3.3.1 制作
gcc -c 源文件名.c -o 生成的二进制文件名.o
ar rc lib静态库名称.a 生成的二进制文件名.o

注意:静态库起名的时候必须 lib 开头.a 结尾

步骤:

  • 新建文件夹: 06_code

  • 源文件:myfun.c

    #include <stdio.h>
    
    void add(int a, int b)
    {
        printf("my_sum = %d\n", (a+b));
    }
    
    void mul(int a, int b)
    {
        printf("my_mul = %d\n", (a*b));
    }
    
  • 头文件:myfun.h

    extern void add(int a, int b);
    extern void mul(int a, int b);
    
  • 制作

    在这里插入图片描述

3.3.2 使用

情况1:使用静态库的文件与静态库 在同一文件夹下

命令:

gcc 源文件名 静态库名称 -o 生成的可执行文件名

测试文件:test01.c

#include <stdio.h>
#include "myfun.h"   //可以不写,但是会报警告
int main(int argc, char const *argv[])
{
    add(10, 3);
    return 0;
}

编译:

在这里插入图片描述

情况2:使用静态库的文件与静态库 不在同一文件夹下

注意:

  • 为了让静态库文件与其对应的头文件和使用静态库文件不在同一文件夹下,所以

    • 创建includes与libs文件夹

    • includes文件用于存储头文件

    • libs文件夹存储静态库文件

      mkdir includes
      mkdir libs
      mv myfun.h includes/
      mv libmyfun.a libs/
      
  • 参数

    -L 引用的静态库所在的路径
    -l 静态库名, 去掉lib与.a
    -I 头文件所在路径
    
  • 命令

    gcc 源文件名 -L 静态库所在的路径 -l 静态库名 -I 头文件所在路径 -o 生成的可执行文件名
    

情况3:静态库文件与对应的头文件 在系统文件夹下

  • 系统库路径:

    /usr/include 		存储头文件
    /usr/lib 或 /lib 	存储库文件
    
  • 注意:

    # 为了让静态库文件与其对应的头文件和系统文件夹下,所以需要移动
    sudo mv includes/myfun.h /usr/include
    sudo mv libs/libmyfun.a /usr/lib  
    
  • 命令:

    gcc 源文件名 -l 静态库名 -o 生成的可执行文件名  
    

3.4 动态库

3.4.1 制作

命令:

gcc -shared 源文件名 -o 生成的动态库文件名.so
3.4.2 使用

情况1:使用动态库的文件与动态库在同一文件夹下

命令:

gcc 源文件名 动态库名称 -o 生成的可执行文件名  

情况2:使用动态库的文件与动态库不在同一文件夹下

命令:

gcc 源文件名 -L 动态库所在路径 -l 动态库名称 -I 头文件所在路径

注意:

  • 动态库名需要去掉前面的 lib 与后面 .so

情况3:静态库文件与对应的头文件在系统文件夹下

命令:

gcc 源文件名 -l 静态库名 -o 生成的可执行文件名

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

uniapp 使用web-view外接三方

来源 前阵子有个需求是需要在原有的项目上加入一个电子签名的功能&#xff0c;为了兼容性和复用性后面解决方法是将这个电子签名写在一个新的项目中&#xff0c;然后原有的项目使用web-view接入这个电子签名项目&#xff1b; 最近又有一个需求&#xff0c;是需要接入第三方的…

蓝桥杯每日一题2023.11.30

题目描述 九数组分数 - 蓝桥云课 (lanqiao.cn) 题目分析 此题目实际上是使用dfs进行数字确定&#xff0c;每次循环中将当前数字与剩下的数字进行交换 eg.1与2、3、4、、、进行交换 2与3、4、、、进行交换 填空位置将其恢复原来位置即可&#xff0c;也就直接将其交换回去即可…

Linux(CentOS7.5):新增硬盘分区纪实

一、服务器概述 1、既有一块系统硬盘&#xff0c;新增一块100G硬盘。 2、要求&#xff0c;将新插入硬盘分为&#xff1a;20G、30G、50G。 二、操作步骤 1、确认新硬盘是否插入成功&#xff1a; fdisk -l# 红色框出来的&#xff0c;为识别出来的新硬盘信息 # 黄色框出来的&#…

Linux:锁定部分重要文件,防止误操作

一、情景描述 比如root用户或者拥有root权限的用户&#xff0c;登陆系统后&#xff0c;通过useradd指令&#xff0c;新增一个用户。 而我们业务限制&#xff0c;只能某一个人才有权限新增用户。 那么&#xff0c;这个时候&#xff0c;我们就用chattr来锁定/etc/passwd文件&…

一些ab命令

1.ab简介 ab是apache自带的压力测试工具&#xff0c;是apachebench命令的缩写。ab非常实用&#xff0c;它不仅可以对apache服务器进行网站访问压力测试&#xff0c;也可以对或其它类型的服务器如nginx、tomcat、IIS等进行压力测试。 ab的原理&#xff1a;ab命令会创建多个并发…

力扣 790. 多米诺和托米诺平铺(一维dp)

题目描述&#xff1a; 有两种形状的瓷砖&#xff1a;一种是 2 x 1 的多米诺形&#xff0c;另一种是形如 "L" 的托米诺形。两种形状都可以旋转。 给定整数 n &#xff0c;返回可以平铺 2 x n 的面板的方法的数量。返回对 109 7 取模 的值。 平铺指的是每个正方形都…

df新增一列数据,并指定列名

方法1&#xff1a;直接指定df列名赋值为list即可 skill_info_df[age] age_list ps:list的长度要和df对齐 方法二 使用insert&#xff1a;

智能优化算法应用:基于象群算法无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于象群算法无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于象群算法无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.象群算法4.实验参数设定5.算法结果6.参考文献7.MATLAB…

AntDB数据库,通信行业20年变迁的见证者

2000年至今&#xff0c;通信行业发展已过了20多年。面对通信行业巨大的数据信息&#xff0c;数据库在行业发展中发挥了巨大的作用&#xff0c;AntDB数据库便是其中较为知名的一款数据库。在通信行业快速发展的阶段&#xff0c;打破国外产品与技术垄断是产业发展的重点与难点。面…

latex表格中内容过多如何换行【已解决】

最近在写论文的时候放了一个表格&#xff0c;但是表格看起来特别大&#xff0c;因为想让某些内容多的单元格完成换行操作 首先在main.tex引入makecell包 \usepackage{makecell} 然后回到表格找到你想换行的单元格&#xff0c;把\makecell{}加进去&#xff0c;然后在需要换行的…

深入浅出强化学习

目录 一、强化学习的概念 二、强化学习的特点 三、强化学习的训练过程 一、强化学习的概念 强化学习是一种机器学习方法&#xff0c;旨在教会算法如何通过与环境的交互来进行学习和决策。与传统的监督学习和无监督学习不同&#xff0c;强化学习侧重于学习与奖励和惩罚&#…

首批量子计算机即将部署!欧盟为波兰提供新算力优势

&#xff08;图片来源&#xff1a;网络&#xff09; 英国量子计算机开发商ORCA公司将为波兰的波兹南超级计算和网络中心&#xff08;PSNC&#xff09;提供两个PT-1光量子系统&#xff0c;以加速其在量子计算领域的研究和应用工作&#xff0c;如生物学、化学和机器学习领域。 …

机器人最优控制开源库 Model-based Optimization for Robotics

系列文章目录 文章目录 系列文章目录前言一、开源的库和工具箱1.1 ACADO1.2 CasADi1.3 Control Toolbox1.4 Crocoddyl1.5 Ipopt1.6 Manopt1.7 LexLS1.8 NLOpt1.9 qpOASES1.10 qpSWIFT1.11 Roboptim 二、其他库和工具箱2.1 MUSCOD2.2 OCPID-DAE12.3 SNOPT 前言 机器人&#xff…

Conductor之动态分叉

Conductor之动态分叉 动态分叉 关于动态分叉&#xff0c;参考1 动态分叉 有时候我们希望在运行时能动态添加分叉任务&#xff08;注意是分叉任务&#xff0c;不是动态任务&#xff09;&#xff0c;Conductor当前版本也是支持的&#xff0c;但是经过试验的版本只支持串行分叉&…

进程间通信方式——管道

进程间通信方式——管道 1、管道2、匿名管道2.1 创建匿名管道2.2 进程间通信 3、有名管道3.1 创建有名管道3.2 进程间通信 4、管道的读写行为 原文链接 1、管道 管道的是进程间通信&#xff08;IPC - InterProcess Communication&#xff09;的一种方式&#xff0c;管道的本质…

进程(process) vs 线程(Thread)

文章目录 前言一、进程&#xff08;process) vs 线程&#xff08;Thread&#xff09;引用自维基百科引用自CSDN INCOE AI引用自 geeksforgeeksOS( Operating System )如何调度线程的线程锁的核心原理是什么? 总结 前言 &#x1f680; 多方面理解进程(process) &#xff0c;线…

Python程序的计时

# -*- coding: UTF-8 -*- import timedef fun():time.sleep(5)sinceTime time.time() print("开始计时时刻&#xff1a;", sinceTime) fun() endTime time.time() print("结束时刻&#xff1a;", endTime) program_time endTime - sinceTime print(&quo…

振南技术干货集:各大平台串口调试软件大赏(5)

注解目录 &#xff08;串口的重要性不言而喻。为什么很多平台把串口称为 tty&#xff0c;比如 Linux、MacOS 等等&#xff0c;振南告诉你。&#xff09; 1、各平台上的串口调试软件 1.1Windows 1.1.1 STCISP &#xff08;感谢 STC 姚老板设计出 STCISP 这个软件。&#xf…

特殊二叉树——堆

&#x1f308;一、堆的基本概念 1.堆&#xff1a;非线性结构&#xff0c;是完全二叉树 2.堆分为大堆和小堆。 大堆&#xff1a;树中任意一个父亲都大于等于孩子&#xff0c;根节点值大于等于其所有子孙节点的值。 小堆&#xff1a;树中任意一个父亲都小于等于孩子&#xff0c;…

【pytorch】深度学习入门一:pytorch的安装与配置(Windows版)

请支持原创&#xff0c;认准DannisTang&#xff08;tangweixuan1995foxmail.com&#xff09; 文章目录 第〇章 阅读前提示第一章 准备工作第一节 Python下载第二节 Python安装第三节 Python配置第四节 Pycharm下载第五节 Pycharm安装第六节 CUDA的安装 第二章 Anaconda安装与配…