你应该知道的C语言干货(7)(对数组类指针的区别解析和使用)

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: 也许你正在对某些指针非常困惑,不知道怎么区分和使用,像是数组指针和指针数组,一级指针,二级指针,一维数组,二维数组等,接下来我们来作区分讲解。

也许你正在对某些指针非常困惑,不知道怎么区分和使用,像是数组指针和指针数组,一级指针,二级指针,一维数组,二维数组等,接下来我们来作区分讲解。


#指针数组,数组指针

1:首先我们类比以前学过的知识,整型数组元素都是整型,字符型数组元素都是字符型,浮点型数组元素都是浮点型,所以指针数组就很好理解了,他存储的所有元素都是相同类型的指针,也就是地址。 (别忘了数组的元素类型都相同哦~)

2:同样的,整型指针是指向整型变量的指针,字符指针是指向字符型变量的指针,所以数组指针也就是指向数组的指针。

接下来我们看代码:

#include <stdio.h>
//指针数组的举例
int main()
{
  int arr[10] = { 1,2,3,4,5,6 };
  char str[6] = "haha";
  float farr[3] = { 3.14,6.66 };
  int a = 1;
  int b = 2;
  int c = 3;
  int* parr1[3] = { &a,&b,&c };
  int* pa = &a;
  int* pb = &b;
  int* pc = &c;
  int* parr2[3] = { pa,pb,pc };
  return 0;
}
#include <stdio.h>
//数组指针的举例
int main()
{
  int arr[10] = { 0 };
  int(*parr)[10] = &arr;  
  int(*parr)[5] = &arr;  //error
  int* pa = arr;
  return 0;
}

f13132f1783d4ccdba210a794cc80212.png

*表示parr是一个指针,指向类型为int [10]的数组,如果*parr不加 (),则parr先与[]结合,表示是一个数组,数组元素类型为int*

注意数组指针指向的数组大小要匹配,因为这算是类型,要类型匹配


#一级指针,一维数组


一:一级指针存放普通变量的地址

int main()
{
  int a = 1;
  int b = 2;
  int c = 3;
  int* pa = &a;
  int* pb = &b;
  int* pc = &c;
  return 0;
}

二:一维数组存放普通变量,其数组名在正常情况下表示元素的首地址

1:普遍情况下:

int main()
{
  int arr[10] = { 1,2,3,4,5,6 };
  int* parr = arr;
  return 0;
}

2:两个例外:

(1)当使用sizeof时,若括号里只有数组名,则表示求整个数组的大小。

7942139aa52a4135993d5de09ad7d96d.png

(2)当使用&数组名时,例如&arr

f8f30d30f7994008b8a8b1075b7f543f.png

我们来计算一下大小:

049d213d1aab42fab8f973f29ebe9616.png

DEC是十进制下的数值,HEX是十六进制,OCT是八进制,BIN是二进制

我们可以看的到,arr+1,地址向后走4个字节,刚好一个int的大小

                        &arr+1,地址向后走40个字节, 10个int大小,刚好是我们这个数组的大小。



区分与使用  :

1:首先数组名是地址常量不可以修改,而一级指针可以修改,举例说明:

817a2d12dfcd4411907a51cee9a58398.png

2:两者共同点在于当做数组使用时,用法几乎相同:

180708ca7f0742e0a74bb00c7017b828.png

3:在函数传参上:

这两种方法都可行,本质上都是传指针。

b5975748d9eb4d6dba0e80afbb512d0f.png


#二级指针,二维数组


一:二级指针存放一级指针的地址

432542e2233a4a79bf6706f9c63e9c9b.png

二:二维数组在普遍情况下和两种例外都同于一维数组,对于二维数组,有这样几种理解方式:

1:首先,我们假设有一个二维数组arr[3][3]

dbf68600ae83412da4738cacb8a77c08.png51a7f4ee6c4e41d197ec245f5313b7d1.png

如果我们将二维数组前面的部分想象成一维数组的数组名,arr[3]就是数组名,通过他我们可以找到任意一维数组的每一个元素,就像这样:arr[0]+1,我们就找到了第一行第二列的元素的地址,解引用可得其值,等同于arr[0][1].

这样的话,arr[3]的数组名就是arr,我们画图来理解:

f868a701167d4066a6f9f506c1c14e2c.png

 

arr+1找到第二行,*(arr+1)找到第二行的首元素的地址,**(arr+1)也就是arr[1][0]


值得注意的地方是在定义二维数组时:

7252b8c89b1045bb81c01a963f701666.png

行的大小可以省略,但列的大小绝对不可以省略。

列的大小确定了,那么一行有多少个元素也就确定了,给定元素数目,行的大小也就确定了。

但是行大小确定,列未知,那么一行有多少元素不清楚,在使用arr[1][0]时,是未知的,因为我们不知道第一行有多少元素,第二行可能就没有元素,是未知的。


二者的传参

#include <stdio.h>
void Example1(int arr[3][3]);
void Example2(int (*arr)[3]);
void Example3(int **ppa);
int main()
{
  int a = 1;
  int* pa = &a;
  int** ppa = &pa;
  int arr[3][3] = { 0 };
  Example1(arr);
  Example2(arr);
  Example3(ppa);
  return 0;
}
void Example1(int arr[3][3])
{
  printf("hehe\n");
}
void Example2(int(*arr)[3])
{
  printf("haha\n");
}
void Example3(int** ppa)
{
  printf("gaga\n");
}

看到这里不要懵,二维数组的首地址作为行地址,也就是说他是指向数组的,前面我们说到

arr等同于&arr[0],而arr[0]是第一行元素的首地址,对他整体取地址,熟悉吗?

不就是类似于一维数组的数组名取地址吗?

9639ea4aad1f407382f3c7f4625527eb.png

对于二级指针,int **ppa来说,*ppa表示ppa是个指针,int*表示这个指针指向int*类型的变量。


#字符指针

前面我们也提到过字符指针,也就是指向字符的指针。

int main()
{
  char ch = 'a';
  char* pc = &ch;
  return 0;
}

同时还有一种字符指针:

42848edc338240458fe72f4ded80f8cc.png

字符串常量的首地址交给我们的字符指针,不过要用const修饰,表示*pc不可被修改。

不可以理解成将一个字符串放到了pc里,而是他的首字符的地址。


#下期预告

1.函数指针

2.函数指针数组

3.指向函数指针数组的指针

4.回调函数

以及他们使用的场景


Bingo!!!

目录
相关文章
|
1月前
使用指针访问数组元素
【10月更文挑战第30天】使用指针访问数组元素。
33 3
|
1月前
|
程序员 C语言 开发者
pymalloc 和系统的 malloc 有什么区别
pymalloc 和系统的 malloc 有什么区别
|
29天前
|
程序员 C语言 开发者
pymalloc 和系统的 malloc 有什么区别?
pymalloc 和系统的 malloc 有什么区别?
|
1月前
使用指针访问数组元素
【10月更文挑战第31天】使用指针访问数组元素。
38 2
|
1月前
|
算法 索引
单链表题+数组题(快慢指针和左右指针)
单链表题+数组题(快慢指针和左右指针)
36 1
|
2月前
|
存储 C语言
C语言:普通局部变量、普通全局变量、静态局部变量、静态全局变量的区别
C语言中,普通局部变量在函数内部定义,作用域仅限于该函数;普通全局变量在所有函数外部定义,作用域为整个文件;静态局部变量在函数内部定义但生命周期为整个程序运行期;静态全局变量在所有函数外部定义,但仅在定义它的文件内可见。
91 10
|
2月前
|
存储
如何使用指针数组来实现动态二维数组
指针数组可以用来实现动态二维数组。首先,定义一个指向指针的指针变量,并使用 `malloc` 为它分配内存,然后为每个子数组分配内存。通过这种方式,可以灵活地创建和管理不同大小的二维数组。
|
2月前
|
存储
如何通过指针数组来实现二维数组?
介绍了二维数组和指针数组的概念及其区别,详细讲解了如何使用指针数组模拟二维数组,包括定义与分配内存、访问和赋值元素、以及正确释放内存的步骤,适用于需要动态处理二维数据的场景。
|
2月前
|
存储 C语言
C语言:结构体与共用体的区别
C语言中,结构体(struct)和共用体(union)都用于组合不同类型的数据,但使用方式不同。结构体为每个成员分配独立的内存空间,而共用体的所有成员共享同一段内存,节省空间但需谨慎使用。
|
2月前
|
存储 编译器 C语言
C语言函数的定义与函数的声明的区别
C语言中,函数的定义包含函数的实现,即具体执行的代码块;而函数的声明仅描述函数的名称、返回类型和参数列表,用于告知编译器函数的存在,但不包含实现细节。声明通常放在头文件中,定义则在源文件中。

推荐镜像

更多