深入理解c语言数组_C 语言

一 数组名是什么

数组就是一段连续可用的内存。
比如声明一个 int数组

int array[]={1,2,3};

array代表什么?有的资料说:数组名是指向数组首地址的常量指针。

下面我们可以验证一下。
我都知道sizeof操作符可以返回一个对象或者类型所占的内存字节数。
如:
int i=1;
那么sizeof(i) 的结果就是4(64位机器下的部分编译器是8)

那我们打印sizeof(array)

printf("%d\n",sizeof(array));

结果是:12。

但是我们都知道sizeof(指针变量)==4的。

所有我们得出:数组名不完全是指向数组首地址的常量指针

为什么要用不完全,因为我们利用数组来访问数组元素的时候。它又变得像一个常量指针。

比如
array[0]等效于*(array+0)

这时候 array就是一个指向数组首地址的常量指针,指针类型是指向数组元素类型的指针。这里就是 int*类型

我们可以这样理解:

一个大学的名字叫array。
有人问你array是什么。你会告诉他array是大学,面积多少之类的。
但是有人问你去array怎么走,你会告诉他array的校门(首地址)在哪里。

结论:数组名其实代表的是一个内存区域,但是使用的时候变成了 指向数组首地址的常量指针。

但是这里还有一个小陷阱:

复制代码 代码如下:

#include <stdio.h>

void foo(int a[])
{
  printf("%d\n",sizeof(a));
}

int main(void)
{
  int array[]={1,2,3};
    foo(array);
    return 0;
}

输出的不是12,而是4。

出于效率的考虑,数组传参是引用传参而不是拷贝传参。因为数组长度可能很大,拷贝一份的话太耗资源。
虽然我是这样函数是这样的

复制代码 代码如下:

void foo(int a[])
{
printf("%d\n",sizeof(a));
}

编译器的眼中是这样

复制代码 代码如下:

void foo(int *a)
{
printf("%d\n",sizeof(a));
}

所以sizeof(a)是sizeof(指针变量)肯定是4;

二 字符数组

首先我们看一个简单的程序

复制代码 代码如下:

# include <stdio.h>

int main(void)
{
  char *str1="abc";
  char str2[]="def";

  printf("%s\n",str1+4);
  return 0;
}

输出的结果是 def。

我们要知道c语言中只要用到了 "xxxxx",系统都会自动的把双引号的内容添加到字符常量区。
注意:printf("xxxx");不会把"xxxx"添加到字符常量区。

复制代码 代码如下:

char *str1="abc";         //会把 abc\0 添加到字符常量区,并把首地址赋值给str指针变量。
char str2[]="def";        //会把 def\0 添加到字符常量区,并且函数栈中添加一个字符数组 内容也是 def\0,str2指向的是栈中的数组。
char str[]={'x','y','z'}; //只会在函数栈中添加数组

由于字符常量区是连续的,所以

printf("%s\n",str1+4);

可以打印出str2的值。

三 二维数组

int array[][3]={1,2,3,4,5,6};

前面我们已经说了,使用array访问元素时,array就是一个指针类型为指向数组元素的指针类型,指向数组首地址的指针。
二维数组的元素就是数组,

这样写就更容易理解:
int array[][3]={{1,2,3},{4,5,6}};

所有可以 这样认为 array是这样认为的
int (*const array)[3];
当我访问数组元素时候

array[x][y]在编译器看来就是 *(*(array+x)+y)

*(array+x) 得到的是一个 第x行 类型为 “int[3]”(c语言没有这样的写法) 的数组,

数组名在访问元素的时候当做首地址指针来用,在这里 *(array+x)等同于数组名,
指针类型int *,指向的地址为 array+sizeof(int(*)[3])*x。

访问这个数组的第y个元素的时候,就要用 *(*(array+x)+y)。

这些就是我对c语言数组的理解,如果错误的地方,谢谢指正,轻喷哈。

时间: 2024-08-22 15:23:43

深入理解c语言数组_C 语言的相关文章

详解C++中的对象指针与对象数组_C 语言

C++对象指针 指向对象的指针 在建立对象时,编译系统会为每一个对象分配一定的存储空间,以存放其成员.对象空间的起始地址就是对象的指针.可以定义一个指针变量,用来存放对象的指针. 如果有一个类: class Time { public : int hour; int minute; int sec; void get_time( ); }; void Time::get_time( ) { cout<<hour<<":"<<minute<<

浅析C语言中的数组及字符数组_C 语言

我们来编写一个程序,以统计各个数字.空白符(包括空格符.制表符及换行符)以及所有其它字符出现的次数.这个程序的实用意义并不大,但我们可以通过该程序讨论 C 语言多方面的问题. 所有的输入字符可以分成 12 类,因此可以用一个数组存放各个数字出现的次数,这样比使用 10 个独立的变量更方便.下面是该程序的一种版本: #include <stdio.h> /* count digits, white space, others */ main() { int c, i, nwhite, nothe

图文详解c/c++中的多级指针与多维数组_C 语言

前言 首先先声明一些常识,如果你对这些常识还不理解,请先去弥补一下基础知识:      1.实际上并不存在多维数组,所谓的多维数组本质上是用一维数组模拟的.      2.数组名是一个常量(意味着不允许对其进行赋值操作),其代表数组首元素的首地址.      3.数组与指针的关系是因为数组下标操作符[],比如,int a[3][2]相当于*(*(a+3)+2) .      4.指针是一种变量,也具有类型,其占用内存空间大小和系统有关,一般32位系统下,sizeof(指针变量)=4.      

C++实例输入多行数字到数组_C 语言

 C++输入多行数字到数组 前天做某公司笔试题的时候,其输入格式是多行数字,每行以空格为分隔符,以换行符号为结束输入到多个数组.在JAVA中有相应的函数直接将一行拆成数组,感觉在C++中这中输入方式还是挺奇怪的,今天想出一种解决方式. 思路: 每次读入一个字符,判断是不是EOF,如果是的话跳出循环: 不是EOF将字符放回缓冲区: 读入换行符,说明一行结束,处理数组: 读入字符,如果不是空格,存入一个临时的字符串: 如果是空格,字符串转换成整数并压入数组: 注意: 对于连续的空格输入要进行判断,否

深入理解C++移位运算符_C 语言

关于逻辑移位.算术移位可参见迅雷深大笔试题部分.的一道题. 以前看到C++标准上说,移位运算符(<<.>>)出界时的行为并不确定: The behavior is undefined if the right operand is negative, orgreater than or equal to the length in bits of the promoted left operand. 我当时也没有深究过这个问题.前几天有个网友来信问起这件事,我才发现,这和Intel

深入理解C++的对象模型_C 语言

何为C++对象模型? C++对象模型可以概括为以下2部分:      1. 语言中直接支持面向对象程序设计的部分      2. 对于各种支持的底层实现机制 引言 现在有一个Point类,声明如下: class Point { public: Point(float xval); virtual ~Point(); float x() const; static int PointCount(); protected: virtual ostream& print(ostream &os)

详解C++中的一维数组和二维数组_C 语言

C++一维数组 定义一维数组 定义一维数组的一般格式为:     类型标识符  数组名[常量表达式]; 例如: int a[10]; 它表示数组名为a,此数组为整型,有10个元素. 关于一维数组的几点说明: 1) 数组名定名规则和变量名相同,遵循标识符定名规则. 2) 用方括号括起来的常量表达式表示下标值,如下面的写法是合法的: int a[10]; int a[2*5]; int a[n*2]; //假设前面已定义了n为常变量 3) 常量表达式的值表示元素的个数,即数组长度.例如,在"int

深入理解堆排序及其分析_C 语言

记得在学习数据结构的时候一味的想用代码实现算法,重视的是写出来的代码有一个正确的输入,然后有一个正确的输出,那么就很满足了.从网上看了许多的代码,看了之后貌似懂了,自己写完之后也正确了,但是不久之后就忘了,因为大脑在回忆的时候,只依稀记得代码中的部分,那么的模糊,根本不能再次写出正确的代码,也许在第一次写的时候是因为参考了别人的代码,看过之后大脑可以进行短暂的高清晰记忆,于是欺骗了我,以为自己写出来的,满足了成就感.可是代码是计算机识别的,而我们更喜欢文字,图像.所以我们在学习算法的时候要注重算

深入理解C++的多态性_C 语言

C++编程语言是一款应用广泛,支持多种程序设计的计算机编程语言.我们今天就会为大家详细介绍其中C++多态性的一些基本知识,以方便大家在学习过程中对此能够有一个充分的掌握. 多态性可以简单地概括为"一个接口,多种方法",程序在运行时才决定调用的函数,它是面向对象编程领域的核心概念.多态(polymorphisn),字面意思多种形状. C++多态性是通过虚函数来实现的,虚函数允许子类重新定义成员函数,而子类重新定义父类的做法称为覆盖(override),或者称为重写.(这里我觉得要补充,重