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

C语言——程序环境和预处理(再也不用担心会忘记预处理的知识)

了解程序环境和预处理

  • 前言:
  • 一、程序环境
  • 二、编译+链接
    • 2.1 翻译环境
    • 2.2 编译的几个阶段
    • 2.3 运行环境
  • 三、预处理
    • 3.1 预定义符号
    • 3.2. #define的使用
      • 3.2.1 #define 定义标识符
      • 3.2.2 #define 定义宏
      • 3.2.3 #define 替换规则
      • 3.2.4 #和##的用途
      • 3.2.5 带副作用的宏参数
      • 3.2.6 宏和函数的对比
      • 3.2.7 命名约定
    • 3.3 #undef 的作用
    • 3.4 命令行定义
    • 3.5 条件编译
    • 3.6 文件包含
      • 3.6.1 头文件被包含的方式
      • 3.6.2 嵌套文件包含
  • 四、其他预处理指令

前言:

先简单了解一下程序环境,然后详细总结翻译环境里的编译和链接,然后在总结编译预处理。

一、程序环境

ANSI C的任何一种实现中,存在两个不同的环境

  1. 翻译环境:这个环境中源代码被转换为可执行的机器指令。
  2. 执行环境:执行二进制代码。

计算机如何执行二进制指令?

  • 我们写的C语言代码是文本信息,计算机不能直接理解。
  • 翻译环境: C语言代码——>二进制代码。(预编译、编译、汇编、链接)
  • 执行环境:执行二进制代码。

二、编译+链接

2.1 翻译环境

程序编译+链接过程:
在这里插入图片描述

  • 组成一个程序的每个源文件通过编译过程分别转换成目标代码(object code)。
  • 每个目标文件由链接器(linker)捆绑在一起,形成一个单一而完整的可执行程序
  • 链接器同时也会引入标准C函数库(ANSI C)中任何被该程序所用到的函数,而且它可以搜索程序员个人的程序库,将其需要的函数也链接到程序中。(链接库)
    注意:
    VS2019开发环境集成了:
    编译器( cl.exe )+链接器( link.exe )+调试器

2.2 编译的几个阶段

创建两个源文件:
sum.c

int sum(int num1,int num2)
{
return num1+num2;
}

test.c

#include <stdio.h>
extern int sum(int,int);//声明外部函数
int main()
{
int n=sum(1,2;
printf("%d\n",n);
return 0}

VS2019这样的集成开发环境不方便观察编译细节,这里我使用Linux系统,gcc编译器观察这些细节。
编译分为两个阶段,编译+链接,其中编译可细分为预编译+编译+汇编。(隔离编译,一起链接)
为了查看编译期间的每一步发生了什么,我们可以用命令去编译程序,生成不同文件。
1.预处理

gcc -E test.c -o test.i
预处理完成后就停下来,预处理之后产生的结果都放在test.i文件中。
在这里插入图片描述

2.编译

gcc -s test.c(gcc -s test.i)
编译完成后就停下来,结果保存在test.s中。

3.汇编

gcc -c test.c(gcc -c test.s )
汇编完成后就停下来,结果保存在test.o中。

4.链接

gcc test.o -o test.
链接完成后就停下来,生成test.exe可执行程序。

具体流程如图所示:
在这里插入图片描述

2.3 运行环境

程序执行的过程:
1.程序必须载入内存中,在有操作系统的环境中: 一般这个由操作系统完成。在独立的环境中,程序的载入必须由手工安排,也可以是通过可执行代码置入只读内存来完成。
2.程序的执行便开始。接着便调用main函数
3开始执行程序代码。这个时候程序将使用一个运行时堆栈(stack)(也叫函数栈帧)存储函数的局部变量和返回地址。程序同时也可以使用静态(static) 内存,存储于静态内存中的变量在程序的整个执行过程中一直保留他们的值。
4.终止程序正常终止main函数也有可能是意外终止

三、预处理

3.1 预定义符号

__FILE__        //进行编译的源文件
__LINE__        //文件当前的行号
__DATE__        //文件被编译的日期
__TIME__        //文件被编译的时间
__STDC__        //如果编译器遵循 ANSI C,其值为1,否则未定义(当前VS是不支持ANSI C(标准c))。

这些预定义符号都是语言内置的。
如图:
在这里插入图片描述

3.2. #define的使用

3.2.1 #define 定义标识符

语法:

define name stuff

例子:

#define MAX 1000//给1000定义一个标识符
#define REG register //为register这个关键字,创建一个简短的名字
#define DO_FOREVER for(;;)//用更形象的符号来替换一种实现
#define CASE break;case//在写case的时候,自动把break加上
#define DEBUG_PRINT printf("file:%s\tline:%d\tdatE:%s\ttime:%s\n",__FILE__,\
__LINE__,__DATE__,__TIME__);//如果定义的标识符过长,可以分成几行写,除了最后一行外,每行的后面都加一个‘\’反斜杠,代表连续符

3.2.2 #define 定义宏

#define机制包括了一个规定,允许把参数替换到文本中,这种实现通常称为宏(macro)或者定义宏(define macro)。
#define 宏名{形参表) 字符串
其中形参表是个由逗号隔开的符号表,它可以出现在字符串中
注意:
参数列表的左括号必须与宏名紧领。
如果两者之间有任何空白存在,参数列表就会被解释为字符串的一部分。
举个例子:
定义一个宏

#define SQUARE(s) s*s

观察下面代码

int a=5;
printf(“%d\n”,SQUARE(a+1));
乍一看,你可能觉得这段代码将打印36这个值,实际上,它打印11
因为,宏,在预编译的时候,替换了文本,参数s被替换成a+1,所以这条语句实际是就变成了
printf(“%d\n”,a+1*a+1);

可以看到替换产生的表达式并没有按照预想的次序进行求值,在宏定义上加两个括号,这个问题就解决了。

#define SQUARE(s) (s)*(s)

结果:

printf(“%d\n”,(a+1) (*a+1));

提示:
所以用于对数值表达式进行求值的宏定义都应该用这种方式加上括号,避免在使用宏时由于参数中的操作符或相邻操作符之间不可预料的相互作用。

3.2.3 #define 替换规则

在程序中扩展#define定义符号和宏时,需要涉及几个步骤:
1.在调用宏时,首先对参数进行检查,看看是否包含任何由#definee定义的符号。如果有,它们首先被替换。
2.替换文本随后被插入到程序中原来文本的位置。对于宏,参数名被他们的值所替换。
3.最后,再次对结果文件进行扫描,看看它是否包含任何由#define定义的符号。如果是,就重复上述处理过程。
注意:
1.宏参数和#define定义中可以出现其他#define定义的符号。但是对于宏,不能出现递归。
2.当预处理器搜索#define定义的符号的时候字符串常量的内容并不被搜索。

3.2.4 #和##的用途

“#”的用途:
在这里插入图片描述
观察这个函数和这个结果,我们发现字符串有自动连接的特点。
在这里插入图片描述
观察这个函数和这个结果,我们发现当字符串作为宏参数的时候才可以把字符串放在字符串中
在这里插入图片描述
观察这个函数和这个结果,我们发现使用#,把一个宏参数变成对应的字符串
代码中#value会被预处理器处理为“value”

“##”的用途:
可以把位于它两边的符号合成一个符号。
它允许宏定义从分离的文本片段创建标识符。
在这里插入图片描述
在这里插入图片描述
观察这两个函数和结果,我们发现宏定义时,"##"连接必须产生一个合法的标识符,否则其结果就是未定义。

3.2.5 带副作用的宏参数

当宏参数在宏的定义中出现超过一次的时候,如果参数带有副作用,那么你在使用这个宏的时候就可能出现危险,导致不可预测的后果。副作用就是表达式求值的时候出现的永久性效果
例子:

#include <stdio.h>
#define MAX(a,b) ((a)>(b)?(a):(b))
int main()
{int x = 5;int y = 8;int z = MAX(x++, y++);printf("x=%d\ty=%d\tz=%d\n", x, y, z);return 0;
}

打印结果:
在这里插入图片描述
这里的MAX宏具有副作用,参数在定义中使用了多次

处理器处理之后的结果:
z=((x++)>(y++)?(x++):(y++));
先x=5,y=8比较,然后都加1;此时,x=6,y=9,因为比较结果为假,所以y赋值给z后再加一,故x=6,y=10,z=9,

3.2.6 宏和函数的对比

宏通常可以执行简单的运算。
比如在两个数中找出较大的一个。

#define MAX(a, b) ((a)>(b)?(a):(b))

这里,我们为什么不用函数来完成这个任务?
宏的优点
1.用于调用函数从函数返回的代码可能比实际执行这个小型计算工作所需要的时间更多。所以宏比函数在程序的规模和速度方面更胜一筹。
2.更为重要的是函数的参数必须声明为特定的类型
所以函数只能在类型合适的表达式上使用。反之可以适用于整形、长整型、浮点型等可以用来比较的类型。
宏是类型无关的。
<font color=red size=>宏的缺点
当然和函数相比宏也有劣势的地方:
1.每次使用宏的时候,一份宏定义的代码将插入到程序中。如果宏较长,可能大幅度增加程序的长度
2.宏是不能调试的。
3.由于与类型无关,也就不够严谨。
4.宏可能会带来运算待优先级的问题,导致程序容易出错。

举个例子:宏可以做到函数做不到的事情,宏的参数可以是类型,函数做不到。

#define MALLOC(num,type) (type*)malloc(num*sizeof(type));
//使用
MALLOC(20, int);//类型做参数
//经过预处理器替换后
(int*)malloc(10 * sizeof(int));

宏和函数的对比:
在这里插入图片描述
补充:
函数栈帧:

  1. 调用函数
  2. 创建函数栈帧
  3. 进行计算
  4. 返回函数

inline(内联函数):
即有宏的优点,也有函数的优点。(没有调用函数返回函数这两个过程)

3.2.7 命名约定

函数和宏的使用语法相似,所以语言本身没有办法区分二者。
程序员默认:
宏名全部大写
函数名不要全部大写

3.3 #undef 的作用

#undef的作用就是移除一个宏定义
如果现存的名字要重新定义,那么它的旧名字首先要被移除

#include <stdio.h>
#define MAX 10
int main()
{printf("%d\n", MAX);
#undef MAX
#define MAX 20printf("%d\n", MAX);return 0;
}

打印结果:
在这里插入图片描述

3.4 命令行定义

许多C的编译器提供了一种能力,允许在命令行中定义符号。用于启动编译过程。
例如:当我们根据同一个源文件要编译出一个程序的不同版本的时候,这个特性有点用处。(假定某个程序中声明了一个某个长度的数组,如果机器内存有限,我们需要一个很小的数组, 但是另外一个机器内存大些,我们需要一个数组能够大些。 )

VS环境无法演示,可以用gcc编译器来演示

#include <stdio.h>
int main()
{
int arr[arr1];
int i =0 ;
for(i = 0; i < arr1; i++)
{
arr[i] = i;
}
for( i = 0; i < arr1; i++)
{
printf("%d ",arr[i]);
}
printf("\n");return 0;
}

在这里插入图片描述

  1. ls 调出当前文件夹的菜单
  2. gcc test.c -D arr1=15 -o test -D指定变量给它赋值 -o编译test.c文件(用gcc编译器编译test.c文件并指定arr1的值为15)(gcc -D arr1=10 test.c)
  3. ./test .当前目录 /test文件(编译当前目录底下的文件)

3.5 条件编译

条件编译的目的和作用:
1.条件编译是指对源程序中某段程序通过条件来控制是否参加编译。
2.根据条件来选取需要的代码进行编译,以便生成不同的应用程序,供不同的用户使用(不同的软硬件平台)。
3.条件编译还可以方便程序的逐段调试,简化调试过程。

在编译一个程序的时候我们如果要将一条语句(一组语句) 编译或者放弃是很方便的。因为我们有条件编译指令。
比如说:
调试性的代码,删除可惜,保留又碍事,所以我们可以选择性的编译。

#include <stdio.h>
//#define __DEBUG__
int main()
{int i = 0;int arr[10] = { 0 };for (i = 0; i < 10; i++){arr[i] = i;
#ifdef __DEBUG__printf("%d\n", arr[i]);
#endif}for (i = 0; i < 10; i++){printf("%d ", arr[i]);}return 0;
}

在这里插入图片描述
常见的条件编译指令:

1、条件编译

#if 常量表达式
//…………
#endif
//常量表达式由预处理器求值
如:
#define DEBUG 1
#if DEBUG//成立
//…………执行
#endif //结束标志
功能:如果常量表达式的结果为真(非0),那么就会执行编译。

2.多分支条件编译

#if 常量表达式1
程序段1//……
#elif 常量表达式2
程序段2//……
#else
程序段3//……
#endif
功能:
如果表达式1的值为真(非0),对程序段1进行编译,如果表达式2的值为真(非0),对程序段2进行编译,否则就编译程序段3,可以使程序在不同的条件下,完成不同的功能。

3.判断是否被定义
//symbol是符号的意思,定义了符号就执行程序段

1、
#if defined(symbol)
程序段
#endif
2、
#ifdef(symbol)
程序段
#endif
3、
#if !defined(symbol)
程序段
#endif
4、
#ifndef(symbol)
程序段
#endif//结束标志
功能:
如果标识符已被#define定义过,则对程序段进行编译。

4、嵌套指令

#if defined(os_UNIX)
#ifdef OPTION1
Unix_option();
#endif
#ifdef OPTION2
Unix_option2();
#endif
#elif defined(os_MISDOS)
#ifdef OPTION2
Misdos_option2();
#endif
#endif
嵌套问题注意配对endif

例子:

#include <stdio.h>
#define MAX 100
#define MIN 1
int main()
{
#if defined(MAX)#ifdef MAXprintf("1\n");#endif#ifdef MINprintf("2\n");#endif
#elif defined(MIN)#ifdef OPTION2printf("3\n");#endif
#endifreturn 0;
}

打印结果:
在这里插入图片描述

注意:
1.条件编译实现的逻辑可以用条件语句来实现,但是条件语句会对整个源程序进行编译,生成的目标代码程序较长,运行时间也较长。
2.采用条件编译可以根据条件只编译程序段1或者程序段2,生成的目标程序较短,运行时间也较短。

3.6 文件包含

#include指令可以使另外一个元素被编译,它的用途是,在#include指令的地方把包含的文件全部替换。(预处理时,先删除这条指令,然后把包含文件的内容全部替换)

3.6.1 头文件被包含的方式

  • 本地文件包含:

#include “myfile”

查找策略:先在源文件目录下查找,如果头文件未找到,编译器就会像库函数头文件一样,在标准路径位置查找头文件;如果找不到就提示编译错误。

  • 库文件包含:

#include <stdio.h>

查找策略:直接去标准路径底下查找头文件,找不到就提示编译错误。

  • 总结:
    对于库文件我们可以使用“”的形式去包含,但是这样查找的效率就低一些,而且这样也不容易区分是库文件还是本地文件。

Linux环境的标准头文件的路径:

/usr/include

VS环境的标准头文件的路径:
vs2013默认路径

c:\program Files (x86)\microsoft Visual studio 12.0\vc\include

3.6.2 嵌套文件包含

嵌套文件包含图:
在这里插入图片描述
可以看到test.h和test.c的公共模块里面包含了两份common.h和common.c公共模块,这样文件内容就重复了。

嵌套文件包含的处理方式:
我们为了解决问题,使用条件编译
两种方法:

#ifndef __TEST_H//根据文件的名字去写
#define __TEST_H
//头文件的内容
#endif//结束标志

或者:

#pragma once//编译提示,头文件只包含一次

四、其他预处理指令

#error
#pragma
#line
……

相关文章:

C语言——程序环境和预处理(再也不用担心会忘记预处理的知识)

了解程序环境和预处理 前言&#xff1a;一、程序环境二、编译链接2.1 翻译环境2.2 编译的几个阶段2.3 运行环境 三、预处理3.1 预定义符号3.2. #define的使用3.2.1 #define 定义标识符3.2.2 #define 定义宏3.2.3 #define 替换规则3.2.4 #和##的用途3.2.5 带副作用的宏参数3.2.6…...

Docker部署EMQX

1、简介 EMQ X (Erlang/Enterprise/Elastic MQTT Broker) 是基于 Erlang/OTP 平台开发的开源物联网 MQTT 消息服务器。 Erlang/OTP是出色的软实时 (Soft-Realtime)、低延时 (Low-Latency)、分布式 (Distributed)的语言平台。 MQTT 是轻量的 (Lightweight)、发布订阅模式 (Pu…...

Spring Cloud(Finchley版本)系列教程(二) 客户端负载均衡Ribbon

Spring Cloud(Finchley版本)系列教程(二) 客户端负载均衡Ribbon 目前主流的负载均衡方案有两种,一种是集中式均衡负载,在消费者与服务提供者之间使用独立的代理方式进行负载,比如F5、Nginx等。另一种则是客户端自己做负载均衡,根据自己的请求做负载,Ribbon就属于客户端自…...

好玩的js特效

记录一些好玩的js特效 1、鱼跳跃特效 引入jquery:https://code.jquery.com/jquery-3.7.1.min.js 源码如下&#xff1a; <!--引入jquery--> <script src"https://code.jquery.com/jquery-3.7.1.min.js"></script> <!--引入跳跃源码--> <s…...

java实现带有html格式和附件的符合RFC822规范的eml格式的信件原文组装

1. 传递html格式的eml信件正文 html传递就是解析成带有< html>标签的字符串在正文中传递即可 From:综合运行平台 to:111qq.com // 重点是格式设置成text/html 编码的话需要设置成UTF-8&#xff0c;不然可能直接在正文中展示html标签&#xff0c;为不是解析成具体的样式…...

如何使用PyTorch训练LLM

推荐&#xff1a;使用 NSDT场景编辑器 快速搭建3D应用场景 像LangChain这样的库促进了上述端到端AI应用程序的实现。我们的教程介绍 LangChain for Data Engineering & Data Applications 概述了您可以使用 Langchain 做什么&#xff0c;包括 LangChain 解决的问题&#xf…...

uniapp 手机 真机测试 ​ 云打包 要是没申请 可以使用云打包 然后采用 测试权限即可​

uniapp 手机 真机测试 打开手机 找到手机的 版本号 点击 知道提示 &#xff08;启动开发者模式&#xff09; 然后 在进行usb的连接打开 运行uniapp 到手机基台 手机确认 即可 四&#xff0c; 云打包 要是没申请 可以使用云打包 然后采用 测试权限即可...

RTSP流媒体服务器EasyNVR视频平台以服务方式启动异常却无报错,该如何解决?

EasyNVR是基于RTSP/Onvif协议的安防视频云服务平台&#xff0c;可实现设备接入、实时直播、录像、检索与回放、云存储、视频分发、级联等视频能力服务&#xff0c;可覆盖全终端平台&#xff08;电脑、手机、平板等终端&#xff09;&#xff0c;在智慧工厂、智慧工地、智慧社区、…...

【List篇】使用Arrays.asList生成的List集合,操作add方法报错

早上到公司&#xff0c;刚到工位&#xff0c;测试同事就跑来说"功能不行了&#xff0c;报服务器异常了&#xff0c;咋回事";我一脸蒙&#xff0c;早饭都顾不上吃&#xff0c;要来了测试账号复现了一下&#xff0c;然后仔细观察测试服务器日志&#xff0c;发现报了一个…...

c++的类模板里,可以直接为静态变量赋值么?

一直以来&#xff0c;咱们学的是&#xff0c;给类模板里的静态变量赋值&#xff0c;要在类外面。但对于类常量&#xff0c;则可以直接在定义时赋值。起因是看STL源码时有这么的写法&#xff0c;又验证了一下。 但是在类模板里直接定义静态活动变量是不可以的&#xff0c;即去…...

【录用案例】CCF-C类,1/2区SCIEI,3个月14天录用,30天见刊,11天检索

计算机科学类SCI&EI 【期刊简介】IF&#xff1a;5.5-6.0&#xff0c;JCR1/2区&#xff0c;中科院2区 【检索情况】SCI&EI 双检&#xff08;CCF-C类&#xff09; 【征稿领域】边缘计算、算法与机器学习的结合研究 录用案例&#xff1a;3个月14天录用&#xff0c;录用…...

qt day 3

1.完成自定义的记事本文件的保存功能 ------------------------------------------------------------------------- widget.cpp ------------------------------------------------------------------------- #include "widget.h" #include "ui_widget.h"…...

SpotBugs检查java代码:在整数上进行没有起任何实际作用的位操作(INT_VACUOUS_BIT_OPERATION)

https://spotbugs.readthedocs.io/en/latest/bugDescriptions.html#int-vacuous-bit-mask-operation-on-integer-value-int-vacuous-bit-operation 在整数上进行无用的与、异或操作&#xff0c;实质上没有做任何有用的工作。 例如&#xff1a;v & 0xffffffff 再例如&…...

javaWeb录入数据异常,mysql显示错误

由于项目,需要输入 电脑的mac地址 ,在web页面中进行录入,但是某个同事录入一直有问题,数据查询时使用 in 或者 都查询不到 通过like %% 可以查询到,非常奇怪,请广大网友不吝赐教. 通过 toHex 进行显示发现 数据开头多了 E2808E...

Vue + Element UI 前端篇(十):动态加载菜单

Vue Element UI 实现权限管理系统 前端篇&#xff08;十&#xff09;&#xff1a;动态加载菜单 动态加载菜单 之前我们的导航树都是写死在页面里的&#xff0c;而实际应用中是需要从后台服务器获取菜单数据之后动态生成的。 我们在这里就用上一篇准备好的数据格式Mock出模…...

图的应用(最小生成树,最短路径,有向无环图)

目录 一.最小生成树 1.生成树 2.无向图的生成树 3.最小生成树算法 二.最短路径 1.单源最短路径---Dijkstra&#xff08;迪杰斯特拉&#xff09;算法 2.所有顶点间的最短路径---Floyd&#xff08;弗洛伊德&#xff09;算法 三.有向无环图的应用 1.AOV网&#xff08;拓扑…...

python正则表达式笔记2

由 \ 和一个字符组成的特殊序列在以下列出。 如果普通字符不是ASCII数位或者ASCII字母&#xff0c;那么正则样式将匹配第二个字符。比如&#xff0c;\$ 匹配字符 $. \number 匹配数字代表的组合。每个括号是一个组合&#xff0c;组合从1开始编号。 比如 (.) \1 匹配 the the 或…...

matplotlib 的默认字体和默认字体系列

matplotlib 的默认字体和默认字体系列 查看默认字体和默认字体系列查看默认字体系列下包含的字体查看 plt.rcParams 设置的所有参数查看所有支持的字体格式设置默认字体方法1&#xff1a;方法2 今天给大家介绍一下 matplotlib 包中的默认字体以及默认字体系列。 查看默认字体和…...

STMCUBEMX_IIC_DMA_AT24C64读取和写入

STMCUBEMX_IIC_DMA_AT24C64读取和写入 说明&#xff1a; 1、此例程只是从硬件IIC升级到DMA读写&#xff0c;因为暂时存储的掉电不丢失数据不多&#xff0c;一页就可以够用&#xff0c;不用担心跨页读写的问题 2、使用DMA后&#xff0c;程序确实是变快了&#xff0c;但是也要注意…...

wsl2相关问题

磁盘空间 wsl 删除相关文件后&#xff0c;如删除docker 无用的容器和镜像&#xff0c;windows上磁盘仍然无法自动回收空间 &#xff08;参考&#xff1a;[microsoft/WSL](https://github.com/microsoft/WSL/issues/4699#issuecomment-627133168)&#xff09; # 如清除无用do…...

调用支付宝接口响应40004 SYSTEM_ERROR问题排查

在对接支付宝API的时候&#xff0c;遇到了一些问题&#xff0c;记录一下排查过程。 Body:{"datadigital_fincloud_generalsaas_face_certify_initialize_response":{"msg":"Business Failed","code":"40004","sub_msg…...

Zustand 状态管理库:极简而强大的解决方案

Zustand 是一个轻量级、快速和可扩展的状态管理库&#xff0c;特别适合 React 应用。它以简洁的 API 和高效的性能解决了 Redux 等状态管理方案中的繁琐问题。 核心优势对比 基本使用指南 1. 创建 Store // store.js import create from zustandconst useStore create((set)…...

day52 ResNet18 CBAM

在深度学习的旅程中&#xff0c;我们不断探索如何提升模型的性能。今天&#xff0c;我将分享我在 ResNet18 模型中插入 CBAM&#xff08;Convolutional Block Attention Module&#xff09;模块&#xff0c;并采用分阶段微调策略的实践过程。通过这个过程&#xff0c;我不仅提升…...

【Linux】C语言执行shell指令

在C语言中执行Shell指令 在C语言中&#xff0c;有几种方法可以执行Shell指令&#xff1a; 1. 使用system()函数 这是最简单的方法&#xff0c;包含在stdlib.h头文件中&#xff1a; #include <stdlib.h>int main() {system("ls -l"); // 执行ls -l命令retu…...

学校招生小程序源码介绍

基于ThinkPHPFastAdminUniApp开发的学校招生小程序源码&#xff0c;专为学校招生场景量身打造&#xff0c;功能实用且操作便捷。 从技术架构来看&#xff0c;ThinkPHP提供稳定可靠的后台服务&#xff0c;FastAdmin加速开发流程&#xff0c;UniApp则保障小程序在多端有良好的兼…...

用docker来安装部署freeswitch记录

今天刚才测试一个callcenter的项目&#xff0c;所以尝试安装freeswitch 1、使用轩辕镜像 - 中国开发者首选的专业 Docker 镜像加速服务平台 编辑下面/etc/docker/daemon.json文件为 {"registry-mirrors": ["https://docker.xuanyuan.me"] }同时可以进入轩…...

Python ROS2【机器人中间件框架】 简介

销量过万TEEIS德国护膝夏天用薄款 优惠券冠生园 百花蜂蜜428g 挤压瓶纯蜂蜜巨奇严选 鞋子除臭剂360ml 多芬身体磨砂膏280g健70%-75%酒精消毒棉片湿巾1418cm 80片/袋3袋大包清洁食品用消毒 优惠券AIMORNY52朵红玫瑰永生香皂花同城配送非鲜花七夕情人节生日礼物送女友 热卖妙洁棉…...

A2A JS SDK 完整教程:快速入门指南

目录 什么是 A2A JS SDK?A2A JS 安装与设置A2A JS 核心概念创建你的第一个 A2A JS 代理A2A JS 服务端开发A2A JS 客户端使用A2A JS 高级特性A2A JS 最佳实践A2A JS 故障排除 什么是 A2A JS SDK? A2A JS SDK 是一个专为 JavaScript/TypeScript 开发者设计的强大库&#xff…...

【C++进阶篇】智能指针

C内存管理终极指南&#xff1a;智能指针从入门到源码剖析 一. 智能指针1.1 auto_ptr1.2 unique_ptr1.3 shared_ptr1.4 make_shared 二. 原理三. shared_ptr循环引用问题三. 线程安全问题四. 内存泄漏4.1 什么是内存泄漏4.2 危害4.3 避免内存泄漏 五. 最后 一. 智能指针 智能指…...

【Linux手册】探秘系统世界:从用户交互到硬件底层的全链路工作之旅

目录 前言 操作系统与驱动程序 是什么&#xff0c;为什么 怎么做 system call 用户操作接口 总结 前言 日常生活中&#xff0c;我们在使用电子设备时&#xff0c;我们所输入执行的每一条指令最终大多都会作用到硬件上&#xff0c;比如下载一款软件最终会下载到硬盘上&am…...