C语言变量可以定义在任意位置?那么到底放在哪个位置才最好呢?
liebian365 2025-03-11 17:41 5 浏览 0 评论
C语言程序开发不像 Python,若是需要使用变量,必须先定义。仔细想一想,C语言这么要求的原因也是容易理解的,至少C程序可以事先知道需要为该变量分配多少内存,这其实也是C语言程序更可控的原因之一。
C语言程序的变量
虽然“定义再使用”目的是为了使用更小的开销,实现更高的效率,但是很多程序员仍然将它看作是一种落后低效的特性。就代码编写而言的确如此,这增加了整个代码的复杂度。
特别是在 C89 时代,C语言程序员在定义变量时,必须将所有变量定义在语句的开头。下面这段C语言代码是非法的:
printf("hello \n"); int i = 3; printf("i= %d\n", i);
强制所有变量语句的开头定义是非常反人类的做法——若是某个C语言代码块使用的变量稍多,很难写出易读的代码。因此C99抛弃了这种限制,允许程序员混合语句和变量定义表达式。这样一来,我们可以将要使用的变量定义在附近,写出更加简洁易读的C语言代码:
int i; fun1(&i); ... float f; fun2(&f);
问题
C语言程序员在支持C99标准的平台开发程序,可以将变量定义表达式定义在任意位置。那么一个有趣的问题就出现了,请看下面这段C语言代码:
int fun () { if (!somecondition) return false; internalStructure *str1; internalStructure *str2; char *dataPointer; float xyz; }
作为示例,fun() 函数的代码很简单,读者应将注意力放在下面的定义变量上,显然,这几行变量定义代码既可以放在 if() 表达式前面,也可以放在后面。若是 if() 表达式为真,提前让 fun() 函数返回,那么是不是在其后定义的变量就不会被执行了呢?
换句话说,这样的情况下,将变量定义语句放在 if() 表达式之前,和之后,有没有性能上的差异呢?到底应该放在前面还是后面呢?
讨论
首先应该明白的是,C99 标准允许程序员混合语句和变量定义表达式的目的就是为了代码更加清晰易读,基于这一点,当前的C语言程序编码风格是尽可能的将变量声明放在接近其被使用的地方。
实际上,鉴于几乎所有处理器都使用栈指针管理程序栈,从机器的角度来看,在C语言程序中将变量定义在任意位置都是相同的(可能作用域不太一样)。例如下面这两个函数:
int foo() { int x; return 5; } int bar() { int x; int y; return 5; }
如果使用现代C语言编译器处理这段代码,并且不指定优化项,会得到下面这样的指令集(以x64平台为例):
foo: push ebp mov ebp, esp sub esp, 8 mov eax, 5 add esp, 8 ret bar: push ebp mov ebp, esp sub esp, 16 mov eax, 5 add esp, 16 ret
看不懂也没关系,至少我们能够看出,两个函数具有相同数量的操作码。这是因为几乎所有的编译器都会预先分配它们所需的所有空间,机器不能做任何不能确定的事情。
现代编译器大都会做一些特别的处理,以使我们的C语言代码获得更高的效率。例如,编译器可以将一些局部变量优化到寄存器中。
出于同样的原因,编译器将收集所有的局部变量声明,并为它们预先分配空间。C89要求所有变量定义放在最前面,是因为它被设计为成一次通过的编译器,它需要在处理其他代码之前知道所有的变量。
C99则聪明得多,即使程序员将变量定义在代码的后面也是可以的,因为它会向后检查更多的代码,将所有变量定义放在其他语句之前处理,本质上来说,它和C89没有区别,只不过做了更多的工作,方便程序员编写代码而已。
所以,在编写C语言程序时,我们更应该考虑的是怎样让代码更加清晰易懂。就本例而言,将变量定义放在if语句前后,并不会带来效率上的差异。
小结
在C99之后,程序员可以混合变量定义和其他语句。读者应注意,C99做出这样的改进,目的在于方便程序员写出更加清晰易懂的代码,考虑将变量定义放在不同位置以获得效率性能上的提升倒没那么苛刻了。一般来说,将变量定义放在最靠近它被使用的地方是比较推荐的做法,当然了,读者也大可不必局限于此,写出清晰易读,效率优异的程序才是最终目的。
欢迎在评论区一起讨论,质疑。文章都是手打原创,每天最浅显的介绍C语言、linux等嵌入式开发,喜欢我的文章就关注一波吧,可以看到最新更新和之前的文章哦。
未经许可,禁止转载。
相关推荐
- 月薪 4K 到 4W 的运维工程师都经历了什么?
-
运维工程师在前期是一个很苦逼的工作,在这期间可能干着修电脑、掐网线、搬机器的活,显得没地位!时间也很碎片化,各种零碎的琐事围绕着你,很难体现个人价值,渐渐的对行业很迷茫,觉得没什么发展前途。这些枯燥无...
- 计算机专业必须掌握的脚本开发语言—shell
-
提起Shell脚本很多都有了解,因为无论是windows的Dom命令行还是Linux的bash都是它的表现形式,但是很多人不知道它还有一门脚本编程语言,就是ShellScript,我们提起的Shel...
- Linux/Shell:排名第四的计算机关键技能
-
除了编程语言之外,要想找一份计算机相关的工作,还需要很多其他方面的技能。最近,来自美国求职公司Indeed的一份报告显示:在全美工作技能需求中,Linux/Shell技能仅次于SQL、Java、P...
- 使用Flask应用框架在Centos7.8系统上部署机器学习模型
-
安装centos7.8虚拟环境1、镜像链接...
- shell编程
-
简介:Shell是一个用C语言编写的程序,它是用户使用Linux的桥梁。Shell既是一种命令语言,又是一种程序设计语言。...
- 14天shell脚本入门学习-第二天#脚本和参数#排版修正
-
脚本是一种包含一系列命令的文本文件,通常用于自动化任务。Shell脚本是用Shell命令编写的脚本,可以在命令行中执行。掌握脚本的基础知识和变量的使用是编写高效脚本的关键。...
- 嵌入式Linux开发教程:Linux Shell
-
本章重点介绍Linux的常用操作和命令。在介绍命令之前,先对Linux的Shell进行了简单介绍,然后按照大多数用户的使用习惯,对各种操作和相关命令进行了分类介绍。对相关命令的介绍都力求通俗易懂,都给...
- 实现SHELL中的列表和字典效果
-
大家好,我是博哥爱运维。编写代码,很多情况下我们需要有种类型来存储数据,在python中有列表和字典,golang中有切片slice和map,那么在shell中,我们能否实现列表和字典呢,答案是肯定的...
- 14天shell脚本入门学习-第二天#脚本和变量
-
脚本是一种包含一系列命令的文本文件,通常用于自动化任务。Shell脚本是用Shell命令编写的脚本,可以在命令行中执行。掌握脚本的基础知识和变量的使用是编写高效脚本的关键。...
- shell常用命令之awk用法介绍
-
一、awk介绍awk的强大之处,在于能生成强大的格式化报告。数据可以来自标准输入,一个或多个文件,或者其他命令的输出。他支持用户自定义函数和动态正则表达式等先进功能,是Linux/unix一个强大的文...
- Linux编程Shell之入门——Shell数组拼接与合并
-
在Shell中,可以使用不同的方式实现数组拼接和合并。数组拼接指将两个数组中的元素合并成一个数组,而数组合并指将两个数组逐个组合成一个新数组。以下是关于Shell数组拼接和合并的详细介绍:数...
- shell中如何逆序打印数组的内容,或者反转一个数组?
-
章节索引图首先请注意,有序的概念仅适用于索引数组,而不适用于关联数组。如果没有稀疏数组,答案会更简单,但是Bash的数组可以是稀疏的(非连续索引)。因此,我们需要引入一个额外的步骤。...
- 如何学好大数据开发?---shell基本语法
-
昨天我们初步了解到了shell的一些基本知识,比如shell的分类,常用的shell类型。今天就带来大数据开发之shell基本语法,掌握好基础才是最重要的,那接下来就开始学习shell的基本语法。一、...
- Linux编程Shell之入门——Shell关联数组
-
关联数组是Shell中一种特殊的数组类型,它使用字符串作为下标。在关联数组中,每个元素都被标识为一个唯一的字符串键值,也称为关联数组的索引。在Shell中,可以使用declare或typeset命令...
- 从编译器视角看数组和指针
-
虽然有单独的文章描述数组和指针,但二者的关系实在值得再写一篇文章。...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- wireshark怎么抓包 (75)
- qt sleep (64)
- cs1.6指令代码大全 (55)
- factory-method (60)
- sqlite3_bind_blob (52)
- hibernate update (63)
- c++ base64 (70)
- nc 命令 (52)
- wm_close (51)
- epollin (51)
- sqlca.sqlcode (57)
- lua ipairs (60)
- tv_usec (64)
- 命令行进入文件夹 (53)
- postgresql array (57)
- statfs函数 (57)
- .project文件 (54)
- lua require (56)
- for_each (67)
- c#工厂模式 (57)
- wxsqlite3 (66)
- dmesg -c (58)
- fopen参数 (53)
- tar -zxvf -c (55)
- 速递查询 (52)