【C语言入门】解锁核心关键字的终极奥秘与实战应用(三)

目录

一、auto

1.1. 作用

1.2. 特性

1.3. 代码示例

二、register

2.1. 作用

2.2. 特性

2.3. 代码示例

三、static

3.1.  修饰局部变量

3.2. 修饰全局变量

3.3. 修饰函数

四、extern

4.1. 作用

4.2. 特性

4.3. 代码示例

五、volatile

5.1. 作用

5.2. 代码示例

六、总结


接上一篇https://blog.csdn.net/weixin_37800531/article/details/145430862?sharetype=blogdetail&sharerId=145430862&sharerefer=PC&sharesource=weixin_37800531&spm=1011.2480.3001.8118继续分析。

、auto

在C语言中,auto 关键字用于修饰局部变量,尽管在实际编程中,我们往往省略这个关键字,因为局部变量默认就是自动存储类型(auto)的。auto 关键字的主要作用是显式地表明变量的存储类型,并帮助程序员更好地理解代码。

1.1. 作用

  • 修饰局部变量,表明该变量是自动存储类型的。
  • 虽然通常省略,但在某些情况下,显式使用 auto 可以提高代码的可读性。

1.2. 特性

  • 自动分配内存:当函数被调用时,auto 变量会在栈上自动分配内存。
  • 自动释放内存:当函数返回时,auto 变量所占用的内存会自动释放,变量失效。
  • 生命周期auto 变量的生命周期仅限于定义它的函数或代码块内。

1.3. 代码示例

虽然auto关键字通常被省略,但以下示例可以展示其用法:

#include <stdio.h>  void myFunction() {  auto int myAutoVar = 10; // 显式使用 auto 关键字,但通常可以省略  printf("Value of myAutoVar: %d\n", myAutoVar);  // myAutoVar 在这里有效,但函数返回后将自动释放内存  
}  int main() {  myFunction();  // printf("Value of myAutoVar: %d\n", myAutoVar); // 错误:myAutoVar 在这里无效  return 0;  
}

myAutoVar 是一个 auto 类型的局部变量,它在 myFunction 函数内部被定义并初始化。当 myFunction 被调用时,myAutoVar 会在栈上分配内存。当 myFunction 返回时,myAutoVar 所占用的内存会自动释放,变量失效。因此,在 main 函数中尝试访问 myAutoVar 会导致编译错误。

运行结果: 

需要注意的是,由于局部变量默认就是 auto 类型的,所以在实际编程中,我们通常会省略 auto 关键字。

二、register

在C语言中,register 关键字用于向编译器提出建议,希望编译器能够将特定的变量存储在CPU的寄存器中,以便提高对该变量的访问速度。寄存器是CPU内部的一种高速存储单元,其访问速度远快于内存。因此,如果某个变量被频繁地读取,且不会被修改,那么将其存储在寄存器中可以显著提高程序的性能。

2.1. 作用

  • register 关键字的主要作用是向编译器发出建议,希望编译器能够优化变量的存储位置,将其从内存移动到寄存器中。
  • 然而,需要注意的是,这只是一个建议,编译器并不一定会采纳。编译器会根据自身的优化策略、寄存器的可用性以及变量的使用情况来决定是否将变量存储在寄存器中。

2.2. 特性

  • 适用于频繁读取且不会被修改的局部变量:由于寄存器的数量有限,且读写寄存器需要消耗一定的CPU资源,因此register关键字最适合用于那些被频繁读取且不会被修改的局部变量。

  • 编译器可能会忽略此建议:如前所述,register只是一个建议,编译器并不一定会采纳。编译器会根据实际情况来决定是否将变量存储在寄存器中。

  • 不能用于全局变量和静态变量:由于全局变量和静态变量在程序的整个生命周期内都有效,且可能会被多个函数访问和修改,因此它们不适合存储在寄存器中。

2.3. 代码示例

下面是一个使用register关键字的简单示例:

#include <stdio.h>  void count_loops(int n) {  register int i; // 建议编译器将i存储在寄存器中  for (i = 0; i < n; i++) {  // 循环体为空,仅用于演示  }  printf("Loop count: %d\n", i); // 此时i的值应为n  
}  int main() {  count_loops(1000000); // 调用函数,传入一个较大的值以演示效果  return 0;  
}

使用了register关键字来建议编译器将循环变量i存储在寄存器中。然而,需要注意的是,编译器可能会忽略这个建议,并将i存储在内存中。此外,即使编译器采纳了这个建议,由于寄存器的数量有限,如果程序中使用了大量的register变量,那么编译器也可能无法将它们全部存储在寄存器中。 

运行结果:

在使用register关键字时,我们需要保持谨慎,并意识到它只是一个建议,而不是一个强制性的要求。同时,我们还需要通过实际的性能测试来验证编译器是否采纳了我们的建议,并评估其对程序性能的影响。 

三、static

在C语言中,static 关键字有多种用途,包括修饰局部变量、全局变量和函数。

3.1.  修饰局部变量

static修饰局部变量时,它会延长该变量的生命周期至整个程序运行期间,但变量的作用域仍然保持不变,即只能在定义它的函数或代码块内部访问。意味着,即使函数执行完毕,该变量的值也会保留下来,供下次函数调用时使用。

  • 代码示例:
#include <stdio.h>  void functionWithStaticVar() {  static int count = 0; // 静态局部变量,只在第一次调用时初始化  count++;  printf("Count: %d\n", count);  
}  int main() {  functionWithStaticVar(); // 输出:Count: 1  functionWithStaticVar(); // 输出:Count: 2  functionWithStaticVar(); // 输出:Count: 3  return 0;  
}

count是一个静态局部变量。每次调用functionWithStaticVar函数时,count的值都会递增,并且在下次函数调用时保留下来。

  • 实际运行结果: 

3.2. 修饰全局变量

static修饰全局变量时,它会限制该变量的作用域,使其只能在定义它的文件内部访问。有助于避免不同文件之间的命名冲突。

  • 代码示例
// file1.c  
#include <stdio.h>  static int globalVar = 100; // 静态全局变量,只能在file1.c内部访问  void printGlobalVar() {  printf("GlobalVar in file1.c: %d\n", globalVar);  
}  // file2.c  
#include <stdio.h>  // 尝试访问file1.c中的globalVar会导致编译错误  
// extern int globalVar; // 注释掉这行以避免编译错误  // void printGlobalVarFromFile2() {  
//     printf("GlobalVar in file2.c: %d\n", globalVar); // 这行会导致链接错误  
// }  int main() {  // printGlobalVarFromFile2(); // 注释掉这行以避免编译错误  printGlobalVar(); // 调用file1.c中的函数来打印globalVar  return 0;  
}

globalVar是一个静态全局变量,它只能在file1.c内部访问。如果尝试在file2.c中访问它,会导致编译或链接错误。

3.3. 修饰函数

static修饰函数时,它会使该函数只能在定义它的文件内部使用,防止外部链接。有助于隐藏函数的实现细节,减少命名冲突的可能性。

  • 代码示例
// file1.c  
#include <stdio.h>  static void internalFunction() {  printf("This is an internal function in file1.c\n");  
}  void externalFunction() {  internalFunction(); // 调用内部函数  printf("This is an external function in file1.c\n");  
}  // file2.c  
#include <stdio.h>  // 尝试调用file1.c中的internalFunction会导致链接错误  
// void callInternalFunction() {  
//     internalFunction(); // 这行会导致链接错误  
// }  int main() {  externalFunction(); // 调用file1.c中的外部函数  // callInternalFunction(); // 注释掉这行以避免链接错误  return 0;  
}

internalFunction是一个静态函数,它只能在file1.c内部使用。如果尝试在file2.c中调用它,会导致链接错误。而externalFunction是一个外部函数,它可以在其他文件中被调用。

四、extern

在C语言编程中,extern 关键字扮演着至关重要的角色,它允许我们声明在其他文件中定义的变量或函数,从而实现跨文件的资源共享。这是模块化编程的基础,使得我们可以将程序拆分为多个文件,每个文件负责特定的功能或数据结构,然后在需要时通过 extern 声明来访问这些外部定义的资源。

4.1. 作用

extern 关键字的主要作用是声明一个变量或函数是在其他文件中定义的,这样在当前文件中就可以访问到这个变量或函数。它是实现跨文件链接和访问的关键机制。

4.2. 特性

  • 跨文件访问extern 允许我们访问在其他文件中定义的变量或函数。
  • 声明顺序:在使用 extern 声明的变量或函数之前,编译器需要知道它们的存在。因此,extern 声明通常放在文件的开头部分,或者在变量或函数被实际使用之前。
  • 模块化编程extern 是模块化编程的基础,使得我们可以将程序拆分为多个独立的文件,每个文件都可以定义自己的变量和函数,并通过 extern 声明来访问其他文件中的资源。

4.3. 代码示例

下面是一个简单的示例,展示如何使用 extern 关键字来实现跨文件访问变量和函数。

  • file1.c
#include <stdio.h>  // 定义一个全局变量  
int globalVar = 42;  // 定义一个函数  
void printGlobalVar() {  printf("GlobalVar in file1.c: %d\n", globalVar);  
}
  • file1.h
// 在头文件中使用 extern 来声明 file1.c 中定义的变量和函数  
extern int globalVar;  
extern void printGlobalVar();
  •  file2.c
#include <stdio.h>  
#include "file1.h" // 包含 file1.c 的头文件以访问其声明的变量和函数  int main() {  // 访问 file1.c 中定义的全局变量  printf("Accessing globalVar from file2.c: %d\n", globalVar);  // 调用 file1.c 中定义的函数  printGlobalVar();  return 0;  
}

 

file1.c 定义了一个全局变量 globalVar 和一个函数 printGlobalVar()。然后,在 file1.h 中使用 extern 关键字来声明这些变量和函数,以便在其他文件中访问它们。最后,在 file2.c 中,我们包含了 file1.h 头文件,从而能够访问 file1.c 中定义的变量和函数。

通过这种方式,我们可以将程序拆分为多个文件,每个文件负责特定的功能或数据结构,然后通过 extern 声明和头文件来实现跨文件的资源共享和访问。这是模块化编程的核心思想之一。

五、volatile

volatile 关键字在C/C++等编程语言中用于告诉编译器,某个变量的值可能会在程序的控制流之外被改变。这通常发生在硬件访问、多线程编程或中断服务程序中。使用 volatile 可以防止编译器对该变量进行优化,从而确保每次访问该变量时都能读取其最新的值。

5.1. 作用

  • 防止优化:编译器在优化代码时,可能会将变量的值缓存在寄存器中,以减少对内存的访问。如果变量被声明为 volatile,编译器就不会进行这种优化,而是每次访问该变量时都直接从内存中读取其值。

  • 硬件访问:在嵌入式系统编程中,volatile 常用于访问硬件寄存器的值。这些寄存器的值可能会由硬件本身或其他外部设备改变,因此需要使用 volatile 来确保每次都能读取到最新的值。

  • 多线程编程:在多线程环境中,一个线程可能会修改另一个线程中的变量。虽然C/C++标准并没有将 volatile 定义为线程之间的同步机制,但在某些平台上,使用 volatile 可以防止编译器对共享变量的优化,从而增加线程间通信的可靠性(尽管这不是跨平台或标准的方法,通常应使用同步原语如互斥锁)。

  • 中断服务程序:在中断服务程序中,全局变量的值可能会由中断处理程序改变。使用 volatile 可以确保主程序在访问这些变量时能够读取到最新的值。

5.2. 代码示例

示例1:硬件寄存器访问

#include <stdint.h>  // 假设有一个硬件寄存器的地址是0x40000000  
#define HARDWARE_REGISTER *((volatile uint32_t *)0x40000000)  int main() {  // 读取硬件寄存器的值  uint32_t value = HARDWARE_REGISTER;  // 对寄存器进行写操作  HARDWARE_REGISTER = 0xDEADBEEF;  // 再次读取寄存器的值  value = HARDWARE_REGISTER;  return 0;  
}

HARDWARE_REGISTER 是一个宏,它定义了一个指向硬件寄存器地址的 volatile 指针。确保了每次访问 HARDWARE_REGISTER 时都会直接从硬件寄存器中读取或写入值。

示例2:多线程编程中的共享变量

#include <stdio.h>  
#include <pthread.h>  
#include <unistd.h>  // 注意:这不是跨平台或标准的方法来实现线程间同步  
volatile int shared_variable = 0;  void *thread_function(void *arg) {  // 模拟一些工作  sleep(1);  // 修改共享变量的值  shared_variable = 1;  return NULL;  
}  int main() {  pthread_t thread;  pthread_create(&thread, NULL, thread_function, NULL);  // 等待线程完成工作  while (shared_variable == 0) {  // 这里可能会进行忙等待,但这不是推荐的做法  // 在实际应用中,应该使用条件变量、信号量等同步原语  }  printf("Shared variable has been changed by the thread.\n");  pthread_join(thread, NULL);  return 0;  
}

shared_variable 是一个 volatile 变量,它在多线程环境中被共享。虽然 volatile 在这里可以防止编译器对 shared_variable 的优化,但它并不能保证线程之间的同步。在实际应用中,应该使用互斥锁、条件变量等同步原语来确保线程之间的正确同步。

volatile 并不提供原子性保证,即它不能保证对 volatile 变量的读写操作是原子的。在多线程环境中,即使使用了 volatile,也可能需要额外的同步机制来确保对共享变量的正确访问。

六、总结

C语言共有32个关键字。这些关键字根据作用可以分为以下四类:

  • 数据类型关键字(12个):用于声明变量的数据类型。包括char、double、float、int、long、short、signed、unsigned、struct、union、enum、void等。
  • 控制语句关键字(12个):用于控制程序的流程。包括for、do、while、break、continue、if、else、goto、switch、case、default、return等。
  • 存储类型关键字(4个或5个):用于声明变量的存储类型。常见的包括auto、extern、static、register等,有时也将typedef视为存储类型关键字的一种,尽管其主要功能是为数据类型取别名。
  • 其他关键字(3个或4个):包括const(声明只读变量)、sizeof(计算数据类型长度)、volatile(说明变量在程序执行中可被隐含地改变),有时不包括typedef,因其主要功能是数据类型定义。

这些关键字在C语言编程中具有重要的作用,是构成C语言程序的基本元素之一。掌握这些关键字的使用方法和注意事项,对于学习C语言和进行C语言编程至关重要。

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

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

相关文章

2.2 实现双向链表的快速排序

实现一个双向链表的快速排序。 1>程序代码 #include <stdio.h> #include <string.h> #include <unistd.h> #include <stdlib.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <pthread.h>…

力扣动态规划-19【算法学习day.113】

前言 ###我做这类文章一个重要的目的还是记录自己的学习过程&#xff0c;我的解析也不会做的非常详细&#xff0c;只会提供思路和一些关键点&#xff0c;力扣上的大佬们的题解质量是非常非常高滴&#xff01;&#xff01;&#xff01; 习题 1.矩形中移动的最大次数 题目链接…

Gurobi基础语法之 addConstr, addConstrs, addQConstr, addMQConstr

在新版本的 Gurobi 中&#xff0c;向 addConstr 这个方法中传入一个 TempConstr 对象&#xff0c;在模型中就会根据这个对象生成一个约束。更重要的是&#xff1a;TempConstr 对象可以传给所有addConstr系列方法&#xff0c;所以下面先介绍 TempConstr 对象 TempConstr TempC…

neo4j-community-5.26.0 create new database

1.edit neo4j.conf 把 # The name of the default database initial.dbms.default_databasehonglouneo4j # 写上自己的数据库名称 和 # Name of the service #5.0 server.windows_service_nameneo4j #4.0 dbms.default_databaseneo4j #dbms.default_databaseneo4jwind serve…

unity实现回旋镖函数

最近学习unity2D&#xff0c;想实现一个回旋镖武器&#xff0c;发出后就可以在角色周围回旋。 一、目标 1.不是一次性的&#xff0c;扔出去、返回、没有了&#xff1b;而是扔出去&#xff0c;返回到角色后方相同距离&#xff0c;再次返回&#xff1b;再次返回&#xff0c;永远…

【C++基础】字符串/字符读取函数解析

最近在学C以及STL&#xff0c;打个基础 参考&#xff1a; c中的char[] ,char* ,string三种字符串变量转化的兼容原则 c读取字符串和字符的6种函数 字符串结构 首先明确三种字符串结构的兼容关系&#xff1a;string>char*>char [] string最灵活&#xff0c;内置增删查改…

SpringBoot源码解析(九):Bean定义接口体系

SpringBoot源码系列文章 SpringBoot源码解析(一)&#xff1a;SpringApplication构造方法 SpringBoot源码解析(二)&#xff1a;引导上下文DefaultBootstrapContext SpringBoot源码解析(三)&#xff1a;启动开始阶段 SpringBoot源码解析(四)&#xff1a;解析应用参数args Sp…

C++模板编程——可变参函数模板

目录 1. 可变参函数模板基本介绍 2. 参数包展开——通过递归函数 3. 参数包展开——通过编译期间if语句(constexpr if) 4. 重载 5. 后记 进来看的小伙伴们应该对C中的模板有了一定了解&#xff0c;下面给大家介绍一下可变参函数模板。过于基础的概念将不仔细介绍。 1. 可变…

ChatGPT-4o和ChatGPT-4o mini的差异点

在人工智能领域&#xff0c;OpenAI再次引领创新潮流&#xff0c;近日正式发布了其最新模型——ChatGPT-4o及其经济实惠的小型版本ChatGPT-4o Mini。这两款模型虽同属于ChatGPT系列&#xff0c;但在性能、应用场景及成本上展现出显著的差异。本文将通过图文并茂的方式&#xff0…

2025最新源支付V7全套开源版+Mac云端+五合一云端

2025最新源支付V7全套开源版Mac云端五合一云端 官方1999元&#xff0c; 最新非网上那种功能不全带BUG开源版&#xff0c;可以自己增加授权或二开 拥有卓越的性能和丰富的功能。它采用全新轻量化的界面UI&#xff0c;让您能更方便快捷地解决知识付费和运营赞助的难题 它基于…

数据分析系列--[12] RapidMiner辨别分析(含数据集)

一、数据准备 二、导入数据 三、数据预处理 四、建模辨别分析 五、导入测试集进行辨别分析 一、数据准备 点击下载数据集 二、导入数据 三、数据预处理 四、建模辨别分析 五、导入测试集进行辨别分析 Ending, congratulations, youre done.

当卷积神经网络遇上AI编译器:TVM自动调优深度解析

从铜线到指令&#xff1a;硬件如何"消化"卷积 在深度学习的世界里&#xff0c;卷积层就像人体中的毛细血管——数量庞大且至关重要。但鲜有人知&#xff0c;一个简单的3x3卷积在CPU上的执行路径&#xff0c;堪比北京地铁线路图般复杂。 卷积的数学本质 对于输入张…

51单片机 02 独立按键

一、独立按键控制LED亮灭 轻触按键&#xff1a;相当于是一种电子开关&#xff0c;按下时开关接通&#xff0c;松开时开关断开&#xff0c;实现原理是通过轻触按键内部的金属弹片受力弹动来实现接通和断开。 #include <STC89C5xRC.H> void main() { // P20xFE;while(1){…

wax到底是什么意思

在很久很久以前&#xff0c;人类还没有诞生文字之前&#xff0c;人类就产生了语言&#xff1b;在诞生文字之前&#xff0c;人类就已经使用了语言很久很久。 没有文字之前&#xff0c;人们的语言其实是相对比较简单的&#xff0c;因为人类的生产和生活水平非常低下&#xff0c;…

SSRF 漏洞利用 Redis 实战全解析:原理、攻击与防范

目录 前言 SSRF 漏洞深度剖析 Redis&#xff1a;强大的内存数据库 Redis 产生漏洞的原因 SSRF 漏洞利用 Redis 实战步骤 准备环境 下载安装 Redis 配置漏洞环境 启动 Redis 攻击机远程连接 Redis 利用 Redis 写 Webshell 防范措施 前言 在网络安全领域&#xff0…

【周易哲学】生辰八字入门讲解(八)

&#x1f60a;你好&#xff0c;我是小航&#xff0c;一个正在变秃、变强的文艺倾年。 &#x1f514;本文讲解【周易哲学】生辰八字入门讲解&#xff0c;期待与你一同探索、学习、进步&#xff0c;一起卷起来叭&#xff01; 目录 一、六亲女命六亲星六亲宫位相互关系 男命六亲星…

大模型训练(5):Zero Redundancy Optimizer(ZeRO零冗余优化器)

0 英文缩写 Large Language Model&#xff08;LLM&#xff09;大型语言模型Data Parallelism&#xff08;DP&#xff09;数据并行Distributed Data Parallelism&#xff08;DDP&#xff09;分布式数据并行Zero Redundancy Optimizer&#xff08;ZeRO&#xff09;零冗余优化器 …

玉米苗和杂草识别分割数据集labelme格式1997张3类别

数据集格式&#xff1a;labelme格式(不包含mask文件&#xff0c;仅仅包含jpg图片和对应的json文件) 图片数量(jpg文件个数)&#xff1a;1997 标注数量(json文件个数)&#xff1a;1997 标注类别数&#xff1a;3 标注类别名称:["corn","weed","Bean…

Streamlit入门

1、Streamlit是什么 Streamlit 是一个用于快速构建数据应用的开源 Python 库&#xff0c;由 Streamlit 公司开发并维护。它极大地简化了从数据脚本到交互式 Web 应用的转化过程&#xff0c;让开发者无需具备前端开发的专业知识&#xff0c;就能轻松创建出美观、实用的交互式应…

机器学习算法在网络安全中的实践

机器学习算法在网络安全中的实践 本文将深入探讨机器学习算法在网络安全领域的应用实践&#xff0c;包括基本概念、常见算法及其应用案例&#xff0c;从而帮助程序员更好地理解和应用这一领域的技术。"> 序言 网络安全一直是信息技术领域的重要议题&#xff0c;随着互联…