【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,一经查实,立即删除!

相关文章

Kafka分区策略实现

引言 Kafka 的分区策略决定了生产者发送的消息会被分配到哪个分区中&#xff0c;合理的分区策略有助于实现负载均衡、提高消息处理效率以及满足特定的业务需求。 轮询策略&#xff08;默认&#xff09; 轮询策略是 Kafka 默认的分区策略&#xff08;当消息没有指定键时&…

c++ stl 遍历算法和查找算法

概述&#xff1a; 算法主要由头文件<algorithm> <functional> <numeric> 提供 <algorithm> 是所有 STL 头文件中最大的一个&#xff0c;提供了超过 90 个支持各种各样算法的函数&#xff0c;包括排序、合并、搜索、去重、分解、遍历、数值交换、拷贝和…

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…

五子棋对弈

问题描述 "在五子棋的对弈中&#xff0c;友谊的小船说翻就翻&#xff1f;" 不&#xff01;对小蓝和小桥来说&#xff0c;五子棋不仅是棋盘上的较量&#xff0c;更是心与心之间的沟通。这两位挚友秉承着"友谊第一&#xff0c;比赛第二"的宗旨&#xff0c;决…

使用 HTTP::Server::Simple 实现轻量级 HTTP 服务器

在Perl中&#xff0c;HTTP::Server::Simple 模块提供了一种轻量级的方式来实现HTTP服务器。该模块简单易用&#xff0c;适合快速开发和测试HTTP服务。本文将详细介绍如何使用 HTTP::Server::Simple 模块创建和配置一个轻量级HTTP服务器。 安装 HTTP::Server::Simple 首先&…

在AI技术深度渗透的背景下,2025年传媒互联网行业的哪些细分场景和产品形态将迎来爆发式增长?

一、AI技术重构传媒互联网行业版图&#xff1a;从底层逻辑到应用场景 近年来&#xff0c;AI技术已从实验室走向商业化落地&#xff0c;而传媒互联网行业因其庞大的用户基数、高频交互场景和丰富的数据积累&#xff0c;成为AI应用的主战场。根据华源证券最新行业周报&#xff0…

Docker Hub 镜像 Pull 失败的解决方案

目录 引言一、问题二、原因三、解决方法四、参考文献 引言 在云原生技术火热的当下&#xff0c;Docker可谓是其基础&#xff0c;由于其简单以及方便性&#xff0c;让开发人员不必再为环境配置问题而伤脑筋&#xff0c;因为可将其看作一个虚拟机程序去理解。所以掌握好它可谓是…

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;内置增删查改…

求一个数的数根(高精度)

上一期我们讲的是求一个数的数根&#xff0c;和本期唯一不同的是&#xff0c;数据范围不同了&#xff0c;上一期这个数是小于等于10的18次方的&#xff0c;这一期是小于等于10的1000次方的&#xff0c;开一个变量&#xff1f;肯定不行&#xff0c;我们需要再开一个数组&#xf…

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

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

Vue 3 30天精进之旅:Day 13 - 路由守卫

在构建单页面应用时&#xff0c;路由守卫是一个非常重要的概念。它允许我们在路由进入或离开时执行一些操作&#xff0c;比如验证用户权限、处理数据加载、执行导航确认等。Vue Router提供了多种类型的路由守卫&#xff0c;使我们能够灵活地控制路由的行为。在今天的学习中&…

【TypeScript】基础:数据类型

文章目录 TypeScript一、简介二、类型声明三、数据类型anyunknownnervervoidobjecttupleenumType一些特殊情况 TypeScript 是JavaScript的超集&#xff0c;代码量比JavaScript复杂、繁多&#xff1b;但是结构更清晰 一、简介 为什么需要TypeScript&#xff1f; JavaScript的…

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…

三数之和(15)

15. 三数之和 - 力扣&#xff08;LeetCode&#xff09; 可以一起总结的题目&#xff1a;三角形的最大周长&#xff08;976&#xff09;-CSDN博客 解法&#xff1a; class Solution { public:vector<vector<int>> threeSum(vector<int>& nums) {vector…

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

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