×

关于指针数组与数组指针详解(知识点全面)

前端技术网 前端技术网 发表于2024-01-29 05:51:38 浏览1202 评论0

抢沙发发表评论

一、指针数组和数组指针的区别有哪些

指针数组和数组指针的区别主要有以下几点:

1、数组名不可以改变,而指向数组的指针是可以改变的。字符串指针指向的字符串中的字符是不能改变的,而字符数组中的字符是可以改变的。求数组长度时,借用数组名可求得数组长度,而借用指针却得不到数组长度。

关于指针数组与数组指针详解(知识点全面)

2、数组名的指向不可以改变,而指向数组的指针是可以改变的。因为数组名的指向是不可以改变的,使用自增运算符自增就会改变其指向,这是不对的,数组名只能指向数组的开头。

3、指针数组是一个数组,数组的元素都是指针,数组占多少个字节由数组本身决定。数组指针是一个指针,它指向一个数组。在32位系统下永远是占4个字节,至于它指向的数组占多少字节。

4、字符串指针指向的字符串中的字符是不能改变的,而字符数组中的字符是可以改变的。指针数组本身就是个数组,只不过呢,它里面存放的数据类型不是int,也不是char这类基本类型,而是一个指针类型,说白了就是地址。

扩展资料:

对于指针数组,首先,它是一个数组,知道数组的元素是一个指针,换句话说,数组存储一个指针。此外,数组本身的字节数由数组本身决定;

数组指针,首先它是一个指针,数组指针的指针数组,也就是说,它是一个指针指向一个数组,唯一可以确定的是它是4个字节在32位系统中,但多少字节是指向一个数组,这不是确定的,它取决于情况。

关于指针数组与数组指针详解(知识点全面)

二、数组指针和指针数组的区别

数组指针(也称行指针)

定义 int(*p)[n];

()优先级高,首先说明p是一个指针,指向一个整型的一维数组,这个一维数组的长度是n,也可以说是p的步长。也就是说执行p+1时,p要跨过n个整型数据的长度。

如要将二维数组赋给一指针,应这样赋值:

int a[3][4];

int(*p)[4];//该语句是定义一个数组指针,指向含4个元素的一维数组。

p=a;//将该二维数组的首地址赋给p,也就是a[0]或&a[0][0]

p++;//该语句执行过后,也就是p=p+1;p跨过行a[0][]指向了行a[1][]

所以数组指针也称指向一维数组的指针,亦称行指针。

指针数组

定义 int*p[n];

[]

优先级高,先与p结合成为一个数组,再由int*说明这是一个整型指针数组,它有n个指针类型的数组元素。这里执行p+1时,则p指向下一个数组元素,这

样赋值是错误的:p=a;因为p是个不可知的表示,只存在p[0]、p[1]、p[2]...p[n-1],而且它们分别是指针变量可以用来存放变量地

址。但可以这样*p=a;这里*p表示指针数组第一个元素的值,a的首地址的值。

如要将二维数组赋给一指针数组:

int*p[3];

int a[3][4];

p++;//该语句表示p数组指向下一个数组元素。注:此数组每一个元素都是一个指针

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

p[i]=a[i]

这里int*p[3]表示一个一维数组内存放着三个指针变量,分别是p[0]、p[1]、p[2]

所以要分别赋值。

这样两者的区别就豁然开朗了,数组指针只是一个指针变量,似乎是C语言里专门用来指向二维数组的,它占有内存中一个指针的存储空间。指针数组是多个指针变量,以数组形式存在内存当中,占有多个指针的存储空间。

还需要说明的一点就是,同时用来指向二维数组时,其引用和用数组名引用都是一样的。

比如要表示数组中i行j列一个元素:

*(p[i]+j)、*(*(p+i)+j)、(*(p+i))[j]、p[i][j]

优先级:()>[]>*

三、C语言:简述一下“数组和指针的关系”

一、概念

数组:数组是用于储存多个相同类型数据的**。

指针:指针相当于一个变量,但是它和不同变量不一样,它存放的是其它变量在内存中的地址。

二、赋值、存储方式、求sizeof、初始化等

1.赋值

同类型指针变量可以相互赋值,数组不行,只能一个一个元素的赋值或拷贝

2.存储方式

数组:数组在内存中是连续存放的,开辟一块连续的内存空间。数组是根据数组的下进行访问的,多维数组在内存中是按照一维数组存储的,只是在逻辑上是多维的。

数组的存储空间,不是在静态区就是在栈上。

指针:指针很灵活,它可以指向任意类型的数据。指针的类型说明了它所指向地址空间的内存。

指针:由于指针本身就是一个变量,再加上它所存放的也是变量,所以指针的存储空间不能确定。

3.求sizeof

数组:

数组所占存储空间的内存:sizeof(数组名)

数组的大小:sizeof(数组名)/sizeof(数据类型)

指针:

在32位平台下,无论指针的类型是什么,sizeof(指针名)都是4,在64位平台下,无论指针的类型是什么,sizeof(指针名)都是8。

关于指针和数组求sizeof,我在之前的博客中写过,现将连接贴上:

https://blog.csdn.net/cherrydreamsover/article/details/81589838

4.初始化

数组:

(1)char a[]={"Hello"};//按字符串初始化,大小为6.(2)char b[]={'H','e','l','l'};//按字符初始化(错误,输出时将会乱码,没有结束符)(3)char c[]={'H','e','l','l','o','\0'};//按字符初始化1234

这里补充一个大家的误区,就是关于数组的创建和销毁,尤其是多维数组的创建与销毁。

(1)一维数组:

int* arr= new int[n];//创建一维数组

delete[] arr;//销毁

(2)二维数组:

int** arr= new int*[row];//这样相当于创建了数组有多少行

for(int i=0;i<row;i++)

{

arr[i]= new int[col];//到这里才算创建好了

}

//释放

for(int i=0;i<row;i++)

{

delete[] arr[i];

}

delete[] arr;

指针:

//(1)指向对象的指针:(()里面的值是初始化值)int*p=new int(0); delete p;//(2)指向数组的指针:(n表示数组的大小,值不必再编译时确定,可以在运行时确定)int*p=new int[n]; delete[] p;//(3)指向类的指针:(若构造函数有参数,则new Class后面有参数,否则调用默认构造函数,delete调用析构函数)Class*p=new Class; delete p;//(4)指针的指针:(二级指针)int**pp=new(int*)[1];

pp[0]=new int[6];delete[] pp[0];12345678910

这里我们区分两个重要的概念:指针数组、数组指针。

(1)指针数组:它实际上是一个数组,数组的每个元素存放的是一个指针类型的元素。

int* arr[8];//优先级问题:[]的优先级比*高//说明arr是一个数组,而int*是数组里面的内容//这句话的意思就是:arr是一个含有8和int*的数组1234

请点击输入图片描述

(2)数组指针:它实际上是一个指针,该指针指向一个数组。

int(*arr)[8];//由于[]的优先级比*高,因此在写数组指针的时候必须将*arr用括号括起来//arr先和*结合,说明p是一个指针变量//这句话的意思就是:指针arr指向一个大小为8个整型的数组。1234

请点击输入图片描述

三、传参

数组:

数组传参时,会退化为指针,所以我们先来看看什么是退化!

(1)退化的意义:C语言只会以值拷贝的方式传递参数,参数传递时,如果只拷贝整个数组,效率会大大降低,并且在参数位于栈上,太大的数组拷贝将会导致栈溢出。

(2)因此,C语言将数组的传参进行了退化。将整个数组拷贝一份传入函数时,将数组名看做常量指针,传数组首元素的地址。

1.一维数组的传参

#include<stdio.h>//传参方式正确//用数组的形式传递参数,不需要指定参数的大小,因为在一维数组传参时,形参不会真实的创建数组,传的只是数组首元素的地址。(如果是变量的值传递,那么形参就是实参的一份拷贝)void test(int arr[])

{}//传参方式正确//不传参数可以,传递参数当然也可以void test(int arr[10])

{}//传参方式正确//一维数组传参退化,用指针进行接收,传的是数组首元素的地址void test(int*arr)

{}//传参方式正确//*arr[20]是指针数组,传过去的是数组名void test2(int*arr[20])

{}//传参方式正确//传过去是指针数组的数组名,代表首元素地址,首元素是个指针向数组的指针,再取地址,就表示二级指针,用二级指针接收void test2(int**arr)

{}int main()

{int arr[10]={0};int*arr2[20]={0};

test(arr);

test2(arr2);

}

2.二维数组的传参

//传参正确//表明二维数组的大小,三行五列void test(int arr[3][5])

{}//传参不正确//二维数组的两个方括号,不能全部为空,也不能第二个为空,只能第一个为空void test(int arr[][])

{}//传参正确//可以写成如下这样传参形式,但是不能写int arr[3][]void test(int arr[][5])

{}//传参不正确//arr是一级指针,可以传给二维数组,但是不能正确读取void test(int*arr)

{}//传参不正确//这里的形参是指针数组,是一维的,可以传参,但是读取的数据不正确void test(int* arr[5])

{}//传参正确//传过去的是二维数组的数组名,即数组首元素的地址,也就是第一行的地址,第一行也是个数组,用一个数组指针接收void test(int(*arr)[5])

{}//传参不正确//可以传参,但是在读取的时候会有级别不同的问题void test(int**arr)

{}int main()

{int arr[3][5]={0};

test(arr);

}

指针:

1.一级指针传参

当函数参数部分是一级指针时,可以接受什么参数例如:test(int*p)

(1)可以是一个整形指针

(2)可以是整型变量地址

(3)可以是一维整型数组数组名

#include<stdio.h>void print(int*p, int sz)

{int i= 0;for(i=0; i<sz; i++)

{printf("%d\n",*(p+i));

}

}int main()

{int arr[10]={1,2,3,4,5,6,7,8,9};int*p= arr;int sz= sizeof(arr)/sizeof(arr[0]);//一级指针p,传给函数print(p, sz);return 0;

}

2.二级指针传参

即当函数参数部分是二级指针时,可以接受什么参数例如:test(int**p)

(1)二级指针变量

(2)一级指针变量地址

(3)一维指针数组的数组名

#include<stdio.h>void test(int** ptr)

{printf("num=%d\n",**ptr);

}int main()

{int num= 10;int*p=&num;int**pp=&p;

test(pp);

test(&p);return 0;

}

四、函数指针、函数指针数组、函数指针数组的指针

1.函数指针

void test()

{printf("hehe\n");

}//pfun能存放test函数的地址void(*pfun)();

函数指针的形式:类型(*)(),例如:int(*p)().它可以存放函数的地址,在平时的应用中也很常见。

2.函数指针数组

形式:例如int(*p[10])();

因为p先和[ ]结合,说明p是数组,数组的内容是一个int(*)()类型的指针

函数指针数组在转换表中应用广泛

3.函数指针数组的指针

指向函数指针数组的一个指针,也就是说,指针指向一个数组,数组的元素都是函数指针

void test(const char* str)

{printf("%s\n", str);

}int main()

{//函数指针pfunvoid(*pfun)(const char*)= test;//函数指针的数组pfunArrvoid(*pfunArr[5])(const char* str);

pfunArr[0]= test;//指向函数指针数组pfunArr的指针ppfunArrvoid(*(*ppfunArr)[10])(const char*)=&pfunArr;return 0;

}

感谢您的阅读!希望本文对解决您关于关于指针数组与数组指针详解(知识点全面)的问题有所帮助。如果您还有其他疑问,欢迎随时向我们提问。