数组指针

简介: 数组指针

数组指针的定义

1.数组指针是指针还是数组?

指针。

int a = 10;
  int* p = &a;//指向整型数据的指针
char b = 'w';
  char* q = &b;//指向字符变量的指针

所以数组指针应该是指向数组的指针。

2.数组指针应该怎么定义?

int arr[10] = { 0 };
  int(*p)[10] = &arr;

在数组指针定义时区别指针数组的定义,指针数组的定义如下

int* p1[10];

本质还是数组,该数组元素类型为int*,所以每个元素存放的是地址。

3.数组指针定义的解释

int arr[10] = { 0 };
  int(*p)[10] = &arr;

说明:p和*结合,说明p是一个指针变量,该指针指向大小为10个整型的大小的数组,所以p是一个指针,指向一个数组,叫做数组指针。

分析下面的代码

int main()
{
  int arr[10] = { 0 };
  printf("%p\n", &arr[0]);
    printf("%p\n", arr);
    printf("%p\n", &arr);
}

运行结果展示

为什么三个地址一样呢?难道他们三个的意思是一样的?接着看下面的代码

int main()
{
  int arr[10] = { 0 };
  printf("%p\n", &arr[0]);
  printf("%p\n", &arr[0] + 1);
    printf("%p\n", arr);
  printf("%p\n", arr+1);
    printf("%p\n", &arr);
  printf("%p\n", &arr+1);
}

分别给地址做加1操作得到结果如下

可以发现上面两个是地址加一是跳四个字节,而最后一个4->c,差个8,a->c,差了个2,由于十六进制2相当于32,32+8=40,最后一个差了四十个字节,所以其实三个的意思并不一样,

&arr[0]:首元素的地址

arr:数组名表示的是数组的首地址

&arr:表示整个数组的地址;

前两个都可以用整型指针存放他们的地址,最后一个可以用数组指针来存放他的地址

int arr[10] = { 0 };
  int(*p)[10] = &arr;

4.数组指针的使用

void print( int(*p)[10],int sz)
{
  int i = 0;
  for (i = 0; i < sz; i++)
  {
    printf("%d ", *(*p + i));
  }
}
int main()
{
  int arr[10] = { 1 ,2 ,3, 4, 5, 6, 7, 8, 9, 10 };
  int sz = sizeof(arr) / sizeof(arr[0]);
  print(arr, sz);
}

有人会问arr应该用int*去接收,为什么用int(*p)[10]这个也可以呢?可能编译器做了优化,不会报错

void print( int(*p)[10],int sz)
{
  int i = 0;
  for (i = 0; i < sz; i++)
  {
    printf("%d ", *(*p + i));
  }
}
int main()
{
  int arr[10] = { 1 ,2 ,3, 4, 5, 6, 7, 8, 9, 10 };
  int sz = sizeof(arr) / sizeof(arr[0]);
  print(&arr, sz);
}

下面的改成了整个数组的地址,这样理解才和上面说的一致

*(*p + i)

解释一下这一行p代表的是整个数组的地址,对p解引用*p,相当于拿到整个数组,所以*p相当于数组名,而数组名相当于首元素地址,+i操作就是对首元素地址跳四i个字节,然后对(*p+i)解引用得到的*(*p+i)得到对应地址上存的值,一般不用数组指针在一维数组上,下面举例在二维数组上的使用

void print(int(*p)[4], int k, int q)
{
  for (int i = 0; i < k; i++)
  {
    int j = 0;
    for (j = 0; j < q; j++)
    {
      printf("%d ",*(*(p+i)+j));
    }
    printf("\n");
}
}
int main()
{
  int arr[3][4] = { {1,2,3,4},{4,5,6,7},{6,7,8,9} };
  print(arr, 3, 4);
}
*(*(p+i)+j)

这里的p+i,相当于第i行整行数组的地址,*(p+i)相当于拿到了这一行数组,也相当于数组名,也相当于这一行首元素地址,(*(p+i)+j),指的是这一行第j+1元素的地址,然对(*(p+i)+j)解引用得到的是*(*(p+i)+j)就是arr[i][j];

运行结果展示

int(*p)[4]这个数组指针相当于存的是将这个二维数组的整个一行数组的地址。

希望可以帮助到大家,如果有不对的地方希望大佬指教,谢谢大家了

目录
相关文章
|
2月前
使用指针访问数组元素
【10月更文挑战第30天】使用指针访问数组元素。
40 3
|
2月前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
2月前
|
存储 C语言 计算机视觉
在C语言中指针数组和数组指针在动态内存分配中的应用
在C语言中,指针数组和数组指针均可用于动态内存分配。指针数组是数组的每个元素都是指针,可用于指向多个动态分配的内存块;数组指针则指向一个数组,可动态分配和管理大型数据结构。两者结合使用,灵活高效地管理内存。
|
2月前
|
容器
在使用指针数组进行动态内存分配时,如何避免内存泄漏
在使用指针数组进行动态内存分配时,避免内存泄漏的关键在于确保每个分配的内存块都能被正确释放。具体做法包括:1. 分配后立即检查是否成功;2. 使用完成后及时释放内存;3. 避免重复释放同一内存地址;4. 尽量使用智能指针或容器类管理内存。
|
2月前
|
存储 NoSQL 编译器
C 语言中指针数组与数组指针的辨析与应用
在C语言中,指针数组和数组指针是两个容易混淆但用途不同的概念。指针数组是一个数组,其元素是指针类型;而数组指针是指向数组的指针。两者在声明、使用及内存布局上各有特点,正确理解它们有助于更高效地编程。
|
2月前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
61 4
|
2月前
使用指针访问数组元素
【10月更文挑战第31天】使用指针访问数组元素。
53 2
|
2月前
|
算法 索引
单链表题+数组题(快慢指针和左右指针)
单链表题+数组题(快慢指针和左右指针)
41 1
|
3月前
|
存储
如何使用指针数组来实现动态二维数组
指针数组可以用来实现动态二维数组。首先,定义一个指向指针的指针变量,并使用 `malloc` 为它分配内存,然后为每个子数组分配内存。通过这种方式,可以灵活地创建和管理不同大小的二维数组。
|
3月前
|
存储
如何通过指针数组来实现二维数组?
介绍了二维数组和指针数组的概念及其区别,详细讲解了如何使用指针数组模拟二维数组,包括定义与分配内存、访问和赋值元素、以及正确释放内存的步骤,适用于需要动态处理二维数据的场景。