C语言——数组知识大汇总

1、 定义

数组: 相同类型数据项的集合。

下标变量: 后面跟一个方括号内的下标的变量,指示一个数组元素。

数组下标: 在数组名后的方括号内的数值或表达式,用于指示访问哪个数组元素。


数组是由 ”类型名“ ”标识符“ ”维数“ 组成的复合型数据类型。

”维数“ 表示数组中包含的元素个数,必须大于等于 1 的常量表达式。此表达式只能包含 ”整型字面常量“ ”枚举常量“ ”用常量表达式初始化的整型const对象“。


2、 声明、初始化

语法: 元素类型aname[size];    未初始化

元素类型aname[size] = { 初始化列表 };    初始化


a、静态和自动初始化

存储于静态内存的数组只初始化一次,因此当程序执行时,静态数组已经初始化完毕。

对于自动变量的初始化过程就没有那么浪漫啦,因为自动变量位于运行时堆栈中,执行流每次进入它们所在的代码块时,这类变量每次所处的内存位置可能并不相同。在程序开始之前,编译器没有办法对这些位置进行初始化。如果定义的数组比较大,那么每次初始化的时间是非常可观的。 所以就需要综合考虑是不是要声明为 static 。


b、不完整的初始化

c、自动计算数组长度:根据初始值的长度确定

d、字符数组的初始化

比较笨拙的方法: charmessage[ ] = { ‘H', ’e‘, ’l‘,’l‘,’o‘ };

快速初始化字符数组: char message[ ] = ”Hello“;尽管它看上去是一个字符串常量,实际上不是。

区分字符数组与字符串常量

如: char message1[ ] = "Hello";

char *message2 = "Hello";

前者初始化一个字符数组的元素,而后者则是一个真正的字符串常量。这个指针变量被初始化为指向这个字符串常量的存储位置。



数组的集中初始化方式如下:

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

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

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

int a[3] = { [1] = 1, [ 0 ] = 2 }; (知道此种初始化方法即可)


使用循环顺序存取:

int square[10], i;

for (i = 0; i < 10; i++)

{

scanf("%d", &square[i]);

}


初始化一个三维数组:

C语言——数组知识大汇总_第1张图片



3、 下标引用

下标引用实际执行的就是间接访问。


一维数组:

int b[10];

*(b + 3)这个表达式的意思是:首先 b 的值是一个指向整型的指针,所以 3 这个值根据整型值的长度进行调整。加法运算的结果是另一个指向整型的指针,它所指向的是数组第 1 个元素向后移 3 个整数长度的位置。然后,间接访问这个新位置,或者取得那里的值(右值),或者把一个新值存储于该处(左值)。

除了优先级之外,下标引用和间接访问完全相同。

例如下面两个表达是相同的:

array[ subscript ]

*( array + ( subscript ))


下面涉及 ap 的表达式中,写出使用array 的对等表达式

int array[ 10 ];

int *ap = array + 2;


ap array + 2&array[ 2 ]

*ap *(array + 2)array[ 2 ]

ap[ 0 ] 这个表达式对等的表达式是*(ap + (0)),除去括号,与上面的一样。

ap + 6 array + 8&array[ 8 ]

*ap + 6 这里有两个操作符,有优先级,* 的优先级大于 + ,所以对等表达应该为array[ 2 ] + 6

*(ap + 6) array[ 8 ]

ap[ 6 ] array[ 8 ]

&ap 这个表达式是完全合法的,但此时并没有对等的array 的表达式,因为你无法预测编译器会把ap 放在相对于array的什么位置

ap[ -1 ] array[ 1 ]

ap[ 9 ] array[ 11 ] 超过数组的右边界,非法


多维数组:

int matrix[ 3 ][ 10 ]


matrix 它的类型是” 指向包含 10 个整型元素的数组的指针”,它的值是:

C语言——数组知识大汇总_第2张图片

matrix + 1 它的类型也是” 指向包含 10 个整型元素的数组的指针”,它指向matrix的另一行:

C语言——数组知识大汇总_第3张图片

*( matrix + 1)它的类型是“指向整型的指针”,是其中某个值的指针(即某个值的地址),等效于matrix[ 1 ]

C语言——数组知识大汇总_第4张图片


*( matrix + 1)+ 5 它指向的位置比原先那个表达式所指向的位置向后移动了5个整型元素。

C语言——数组知识大汇总_第5张图片


*(*( matrix + 1)+ 5 ) 对上面的值进行间接访问,取出这个元素(右值),如果作为左值则是存储一个新值。等效于matrix[ 1 ][ 5 ]


4、数组和指针

区别:

声明一个数组时,编译器将根据声明所指定的元素数量为数组保留内存空间,然后再创建数组名,它的是一个常量,指向这段空间的起始地址。

声明一个指针变量时,编译器只为指针本身保留内存空间,他并不为任何整型值分配内存空间。而且,指针变量并未初始化为指向任何现有的内存空间,如果它是一个自动变量,它甚至根本不会被初始化。


int a[ 5 ];

int *b;

表达式 *a 是完全合法的,但表达式 *b 却是非法的。 *b 将访问内存中某个不确定的位置,或者导致程序终止。



5、多维数组

如果某个数组的维数不止一个,它就被称为多维数组。

int c[ 6 ] [ 10 ];

把 c 看作是一个包含 6 个元素的向量,只不过它的每个元素本身是一个包含 10 个整型元素的向量。换句话说,c 是一个一维数组的一维数组。


存储顺序:

int array[ 3 ][ 6 ]





6、数组名

一维数组名:它的类型是”指向元素类型的指针“,它指向数组的第一个元素

考虑下面这些声明:

int a;

int b[10];

我们把变量 a 称为标量,因为它是个单一的值,这个变量的类型是一个整数。

我们把变量 b 称为数组,因为它是一些值的集合。

b[4] 的类型是整型,但 b 的类型是什么呢?

在C中,在几乎所有使用数组名的表达式中,数组名的值是一个指针常量,也就是数组第一个元素的地址。

只有在两种场合下,数组名并不用指针常量来表示——就是当数组名作为 sizeof 操作符或单目操作符 & 的操作数时。

sizeof 返回整个数组的长度,而不是指向数组的指针长度。

取一个数组名的地址所产生的是一个指向数组的指针,而不是一个指向某个指针常量值的指针。


多维数组名:

多维数组的数组名与一维数组的差不多,唯一的区别是多维数组第 1 维的元素实际上是另一个数组。

如: int matrix[ 3 ][ 10 ]

创建了 matrix ,它可以看作是一个一维数组,包含三个元素,指示每个元素恰好是包含 10 个整型元素的数组。

matrix 这个名字的值是一个指向它第一个元素的的指针,所以matrix 是一个指向一个包含10 个整型元素的数组的指针。



7、指向数组的指针、指针数组

首先它是一个指针,它指向一个数组。在32位系统下永远是占4个字节,至于它指向的数组占多少字节并不知道。它是“指向数组的指针”的简称;

下面来考考到底哪个是数组指针 、哪个是指针数组?

(A)int  *p1[10];

(B)int  (*p2)[10];

分析:首先我们要明白优先级的问题:

(A)“[  ] ”的优先级比“ * ”要高,p1先与“[ ]”结合,构成了一个数组的定义,数组名为p1,int *修饰的是数组的内容,即数组的每个元素。这个数组,其包含了10个指向int类型的数据指针,即指针数组;

(B)“()”的优先级高于“[ ]”,“*”与p2构成了一个指针的定义,指针的变量名为p2, int 修饰的是数组的内容,即数组的每天元素。数组这里并没有名字,是个匿名数组。现在我们清楚p2是一个指针,它指向一个包含10个int 类型的数据数组,即数组指针。

char (*)[10] p3;   //也是数组指针。char (*)[10]是指针类型,p3是指针变量名。 



8、作为函数参数的数组名、声明数组参数

作为函数参数的数组名:当一个数组名作为参数传递给一个函数时会发生什么情况呢?数组名的值是一个指向指向数组第一个元素的指针,所以此时传递给函数的是一份该指针的拷贝。

函数如果执行了下标引用,实际上是对这个指针执行间接访问操作,并且通过这种间接访问,函数可以访问和修改调用程序的数组元素。

所有的参数传递都是通过传值方式传递的。当然如果你传递了一个指向某个变量的指针,而函数对该指针执行了间接访问操作,那么函数就可以修改那个变量。尽管看上去并不明显,但数组名作为参数时所发生的正是这种情况。这个参数(指针)实际上是通过传值方式传递的,函数得到的是该指针的一份拷贝,它可以被修改,但调用程序所传递的实参不影响。

声明数组参数:

两个原型相等的函数:

一维情况:int strlen(char*string); intstrlen(char string[  ]);(注意使用的上下文环境)

多维情况:void func2(int(*mat)[ 10 ]); = voidfunc2(int mat[  ][ 10 ]);




9、数组的搜索与排序

数组搜索:

C语言——数组知识大汇总_第6张图片

#define NOT_FOUND-1

int search( constint arr[], inttarget, intn)

{

int i, found = 0, where;

i = 0;

while(!found && i < n)

{

if(arr[ i ] == target)

found = 1;

else

++i;

}

if(found)

where = i;

else

where = NOT_FOUND;

return (where);

}



数组的排序:

C语言——数组知识大汇总_第7张图片


int get _min_range(int list[], int first, int last);


void select_sort(int list[] , int n)

{

int fill, /* first element in unsorted subarray */

temp, /* temporary storage */

 index_of_min;/* subscript of next smallest element */


for( fill = 0; fill < n - 1; ++fill )

{

index_of_min = get _min_range(list, fill, n-1);

if( fill != index_of_min)

{

temp = list[index_of_min];

 list[index_of_min] = list[fill];

list[fill] =  list[index_of_min];

}

}

}

你可能感兴趣的:(编程语言【C/C++】)