当前位置:首页 > > 嵌入式微处理器
[导读]大多数情况下,数组和指针是两个完全不同的C语言特性。

1.指针和数组定义、区别


1.1指针和数组定义


指针和数组都是C语言的精髓所在,对于很多C程序员来说,如果你问这样一个问题:数组和指针有什么区别?他们的答案很可能是:”数组和指针不是同一样东西吗,他们之间有什么区别啊?“;确实在极个别的情况下,数组和指针确实可以”通用“。


但是在却大多数情况下,数组和指针是两个完全不同的C语言特性。


首先,我们分别看一下,指针和数组的定义,以及他们之间的不同;指针表示C语言中某种数据类型的数据存储的内存地址,例如,指向各种整型的指针或者指向某个结构体的指针;数组表示若干个相同C语言数据类型的元素在连续内存中储存的一种形态。


1.2数组和指针的区别


好了,上面即为数组和指针的简单定义。在说明数组和指针之间的区别之前,我们先来看一个实例;

我们定义两个文件:a.c和b.c,其中,


[cpp] view plain copy
a.c文件:
int array[4] = {1, 2, 3, 4};
b.c文件: #include <stdio.h>
extern int *array;
int main(void)
{ 
  int i = 0;
  for (; i <4; i++) 
   {
     printf("*array = %d", *array++);
   }
  return 0;
}


我们编译运行上面的示例程序,不出意外其无法运行。可能有的人很不理解其中原因,他们找不出问题的原因。其实根本原因就是他们认为数组和指针是相同的!其将array定义为一个4个int元素的数组,但是在b.c中却又将其声明为int型指针。其实数组和指针只是在特定的上下文环境下可以认为相同,一般情况下,他们是不同的数据类型,就像float和int类型不同一样。

 

下面逐步分析数组和指针之间的区别:


数组和指针本质上都代表一块内存,数组比较”直接“,数组名即代表这块内存的地址,而指针比较”含蓄“,其本身不代表任何有意义的内容,只有给它赋值后,它才真正的表示一块有意义的内存地址。这就引出了指针和数组的一个区别:定义的时机不同;数组在编译时就已经被确定下来,而指针直到运行时才能被真正的确定到底指向何方。数组就好比生在帝王家一样,一生来就有了属于自己的封地、财富(内存),而指针需要经过编译、链接、运行时等重重考验才能获得属于自己的财富(内存)。但是,数组的这些身份(内存)一旦确定下来就不能轻易的改变了,它们(内存)会伴随数组一生;而指针则有很多的选择,在其一生他可以选择不同的生活方式,比如一个字符指针可以指向单个字符同时也可代表多个字符等。


由数组和指针的上述区别引出了它们的之间的另一个区别:访问方式不同;由于数组名直接代表其身份(数组在内存中的地址),而指针需要间接才能知道自己的身份(通过读取其保存的地址),所以它们的对于自己”财富“的访问方式自然就不同了,数组为直接方式,而指针位间接访问。通过下面几幅图来说明,数组和指针对于自身元素的访问方式的不同:



2.再论数组


2.1数组和指针何时相同


图2-1展示了数组和指针何时相同。


图2-1,中说明了数组除了在作为函数参数或者在表达式中作为右值,其他情况情况下其与指针均不相同。 

 

图2-1指针和数组何时相同


图2-1,中说明了数组除了在作为函数参数或者在表达式中作为右值,其他情况情况下其与指针均不相同。


C语言标准对于数组和指针何时相同定义几条规则:


规则1:表达式中的数组名(与声明不同)被编译器当作一个指向数组第一个元素的指针。


规则2:下标总是与指针的偏移量相同。


规则3:在函数参数的声明中,数组名被编译器当作指向该数组第一个元素的指针。


简而言之,数组和指针的关系颇有点像诗和词关系,它们都是文学性之一,有不少共同之处,但在具体的表现形式上又各有特色。下面具体介绍这几条规则的具体含义。


2.1.1 规则1


组合规则1和规则2,就是对于数组下标的引用总是可以写成“一个指向数组的起始地址的指针加上偏移量”。例如,假如我们声明如下:


int a[10], *p, i = 2;

就可以通过下面任何一种方式访问a[i]:

p = a;p[i];

p = a;*(p+i);

p = a + i;*p;

    

记住:在表达式中,指针和数组是可以互换的,因为它们在编译器里的最终形式都是指针,并且都是可以进行取下标操作。


2.1.2 规则2


数组下标和指针总是相同的,为什么要这么说呢?大家可能听说过,在编写程序时,对于数组访问应该写成指针的形式,因为这样可以提高效率。对于现代的编译器而言,这个说法一般是错误的!现代编译器对于数组的访问都会自动优化为其对应的指针加偏移量的形式,所以也就没有哪种形式效率更高的说法了。实际上,数组的访问之所以改写为指针加偏移量的方式,是因为其为系统底层最基本的工作方式。


2.1.3 规则3


C语言中函数的参数基本都是“传值”调用的,唯独数组为“引用”调用方式,即数组作为函数参数时,会被编译器自动的转换为指向数组第一个元素的指针,这是编译器自动完成的。之所以这么做,其实是为了系统性能,因为数组结构占用的内存通常比较大,如果“传值”调用的话,内存拷贝会浪费大量的时间和空间,这样做得不偿失,所以数组作为函数参数时,编译器会自动将其转换为指向第一个元素的指针。 


嵌入式ARM

扫描二维码,关注更多精彩内容

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

在嵌入式Linux开发中,快速获取系统状态信息是调试和监控的关键能力。本文整理了7个高频使用的C语言代码片段,涵盖内存、CPU温度、文件操作等核心场景,帮助开发者高效实现系统状态采集。

关键字: 嵌入式Linux C语言

作为当前最广泛应用的对称加密算法,AES-128凭借其128位密钥长度和10轮加密迭代,在保障数据安全的同时保持高效性能。本文将深入解析AES-128的流式实现原理,并提供经过优化的C语言实现方案,特别针对长数据流处理场...

关键字: AES-128 C语言

在C语言的指针宇宙中,函数指针如同一个神秘的传送门,它打破了传统函数调用的静态边界,让程序在运行时能够动态选择执行路径。这种机制不仅赋予代码前所未有的灵活性,更在系统编程、嵌入式开发等场景中扮演着关键角色。

关键字: 函数指针 C语言

在嵌入式系统、数据库开发和多媒体处理等场景中,二进制文件的随机访问是核心需求。C标准库提供的fseek和ftell函数组合,为高效定位文件位置提供了轻量级解决方案。本文通过代码示例和性能对比,解析其实现原理与最佳实践。

关键字: 二进制文件 C语言

结构体作为C/C++中组织异构数据的核心方式,其内存布局直接影响程序性能。本文通过量化实验对比不同对齐策略的内存占用差异,结合编译器指令实现精准优化。

关键字: 结构体 C语言 编译器

在C语言中,字符串操作是程序设计中非常基础且重要的部分。由于C语言本身没有内置的字符串类型,字符串通常以字符数组或字符指针的形式出现。因此,掌握常见的字符串操作函数的实现原理对于深入理解C语言的内存管理、指针操作和字符串...

关键字: C语言

在C语言编程中,循环结构是处理重复任务的核心工具,而break和continue则是控制循环流程的关键指令。虽然两者都用于改变循环的正常执行路径,但它们的行为和适用场景存在本质差异。

关键字: C语言 编程

在C语言编程中,头文件(.h)是代码组织与模块化的核心工具,而宏定义(#define)作为预处理指令,能够显著提升代码的可读性、可移植性和可维护性。

关键字: C语言

在嵌入式实时系统中,多线程编程通过并发执行提升资源利用率,但共享资源访问冲突会引发数据竞争与死锁。锁机制作为核心同步手段,其选择直接影响系统实时性与可靠性。本文从嵌入式场景出发,分析常见锁机制特性,并提出优化策略。

关键字: C语言 多线程编程 嵌入式系统

在C语言编程中,头文件是代码组织和模块化的重要工具。宏定义作为预处理阶段的核心特性,能够显著提升代码的灵活性、可读性和可移植性。一个精心设计的头文件库,配合恰当的宏定义,可以让代码更加优雅高效。

关键字: C语言
关闭