浅学指针(2)数组函数传值调用

简介: 浅学指针(2)数组函数传值调用

前言

1. 指针的使⽤和传址调⽤

学习指针的⽬的是使⽤指针解决问题,那什么问题,⾮指针不可呢?


案例:写⼀个函数,交换两个整型变量的值

代码如下:


#include <stdio.h>
void Swap1(int x, int y)
{
  int tmp = x;
  x = y;
  y = tmp;
}
int main()
{
  int a = 0;
  int b = 0;
  scanf("%d %d", &a, &b);
  printf("交换前:a=%d b=%d\n", a, b);
  Swap1(a, b);
  printf("交换后:a=%d b=%d\n", a, b);
 return 0;
}



输出结果:

ca145c7fc0b949529b8c4debc624dae2.png

这是为什么呢?


0c1b311362e642e39921abe0771597a7.png


我们发现在main函数内部,创建了a和b,a的地址是0x00cffdd0,b的地址是0x00cffdc4,在调⽤Swap1函数时,将a和b传递给了Swap1函数,在Swap1函数内部创建了形参x和y接收a和b的值,但是x的地址是0x00cffcec,y的地址是0x00cffcf0,x和y确实接收到了a和b的值,不过x的地址和a的地址不⼀样,y的地址和b的地址不⼀样,相当于x和y是独⽴的空间,那么在Swap1函数内部交换x和y的值,⾃然不会影响a和b,当Swap1函数调⽤结束后回到main函数,a和b的没法交换。Swap1函数在使⽤的时候,是把变量本⾝直接传递给了函数,这种调⽤函数的⽅式我们之前在函数的时候就知道了,这种叫传值调⽤。


结论:实参传递给形参的时候,形参会单独创建⼀份临时空间来接收实参,对形参的修改不影响实

参。

所以Swap是失败的了。


那么这个时候,就要搬出指针大哥,在main函数中将a和b的地址传递给Swap函数,Swap函数⾥边通过地址间接的操作main函数中的a和b,像是把家(地址)搬到函数中,直接改变家里的东西(a, b的值)。

代码展示:


#include<stdio.h>
void Swap2(int*px, int*py)
{
  int tmp = 0;
  tmp = *px;
  *px = *py;
  *py = tmp;
}
int main()
{
  int a = 0;
  int b = 0;
  scanf("%d %d", &a, &b);
  printf("交换前:a=%d b=%d\n", a, b);
  Swap1(&a, &b);
  printf("交换后:a=%d b=%d\n", a, b);
 return 0;
}


输出结果:

fdd2c8f70cec4dc686eec6f224128396.png

我们可以看到实现成Swap2的⽅式,顺利完成了任务,这⾥调⽤Swap2函数的时候是将变量的地址传递给了函数,这种函数调⽤⽅式叫:传址调⽤。


2 .数组名的理解

如下代码:

#include <stdio.h>
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
  printf("&arr[0] = %p\n", &arr[0]);
  printf("arr = %p\n", arr);
  return 0;
}


输出结果:

e35b39cd0fc145f7b5ae6b04f97496ce.png


所以说:数组名就是数组⾸元素(第⼀个元素)的地址。

但是有两个例外:


• sizeof(数组名),sizeof中单独放数组名,这⾥的数组名表⽰整个数组,计算的是整个数组的⼤⼩,

单位是字节

• &数组名,这⾥的数组名表⽰整个数组,取出的是整个数组的地址(整个数组的地址和数组⾸元素

的地址是有区别的)

除此之外,任何地⽅使⽤数组名,数组名都表⽰⾸元素的地址。


还有一点,arr和&arr有啥区别:


#include <stdio.h>
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
  printf("&arr[0] = %p\n", &arr[0]);
  printf("&arr[0]+1 = %p\n", &arr[0]+1);
  printf("arr = %p\n", arr);
  printf("arr+1 = %p\n", arr+1);
  printf("&arr = %p\n", &arr);
  printf("&arr+1 = %p\n", &arr+1);
 return 0;
}


输出结果:

d5d9c2b571b44101af2969002419be0d.png


这⾥我们发现&arr[0]和&arr[0]+1相差4个字节,arr和arr+1 相差4个字节,是因为&arr[0] 和 arr 都是⾸元素的地址,+1就是跳过⼀个元素。

但是&arr 和 &arr+1相差40个字节,这就是因为&arr是数组的地址+1 操作是跳过整个数组的。到这⾥⼤家应该搞清楚数组名的意义了吧。


3. 使⽤指针访问数组

有了前面知识,我们便可以使用指针访问数组啦!其实数组也相当于指针,如:*(p + 1)= arr[1],扩展

7f290bedf1c647d8a1245e149c9bf8d8.png

数组名arr是数组⾸元素的地址,可以赋值给p,其实数组名arr和p在这⾥是等价的。那我们可以使⽤arr[i]可以访问数组的元素,那p[i]是否也可以访问数组呢?`


#include <stdio.h>
int main()
{
  int arr[10] = {0};
  //输⼊
  int i = 0;
  int sz = sizeof(arr)/sizeof(arr[0]);
  //输⼊
  int* p = arr;
  for(i=0; i<sz; i++){
    scanf("%d", p+i);
    //scanf("%d", arr+i);//也可以这样写
  }
  //输出
  for(i=0; i<sz; i++){
    printf("%d ", p[i]);
  }
 return 0;
}


在第18⾏的地⽅,将*(p+i)换成p[i]也是能够正常打印的,所以本质上p[i] 是等价于 *(p+i)。


4 . 一维数组函数传参形本质

接下来进入重点,⾸先从⼀个问题开始,我们之前都是在函数外部计算数组的元素个数,那我们可以把函数传给⼀个函数后,函数内部求数组的元素个数吗?

展示如下:


#include <stdio.h>
void test(int arr[])
{
  int sz2 = sizeof(arr)/sizeof(arr[0]);
  printf("sz2 = %d\n", sz2);
}
int main()
{
  int arr[10] = {1,2,3,4,5,6,7,8,9,10};
  int sz1 = sizeof(arr)/sizeof(arr[0]);
  printf("sz1 = %d\n", sz1);
  test(arr);
 return 0;
}


输出结果:

58acd46d16ae4c4fb068302119a2a3ac.png


很明显函数是没有正确获得数组元素个数的,上面我们学习了数组名是数组⾸元素的地址;那么在数组传的时候,传递的是数组名,也就是说本质上数组传参本质上传递的是数组⾸元素的地址。

所以函数形参的部分理论上应该使⽤指针变量来接收⾸元素的地址。


那么在函数内部我们写sizeof(arr) 计算的是⼀个地址的⼤⼩(单位字节)⽽不是数组的⼤⼩(单位字节)。正是因为函数的参数部分是本质是指针,所以在函数内部是没办法求的数组元素个数的。


总结:⼀维数组传参,形参的部分可以写成数组的形式,也可以写成指针的形式.


5. ⼆级指针

f3cab896506f40ac87ebca0f4e7c6d15.png

要理解如下:


int main()
{
  int a = 30;
  int * pa = &a;//取出a的地址 * 表示pa是指针,指针的类型是int类型
  int * * ppa = &p;//ppa前的*表示ppa是指针,指针的类型是int *类型
}


2级指针的解引用

对上面案例分析,像是扒白菜一样,一层一层来,如下展示:

136388d8683949ed8451c9058fb92c96.png


最终得到结果


总结:2级指针是存放一级指针的地址

6. 指针数组

指针数组是存放指针的数组

4550bd7a3cad443497f9874665dd3b97.png


结合如上知识,那么指针是可以模拟数组的

扩展:指针数组模拟⼆维数组


#include <stdio.h>
int main()
{
  int arr1[] = {1,2,3,4,5};
  int arr2[] = {2,3,4,5,6};
  int arr3[] = {3,4,5,6,7};
  //数组名是数组⾸元素的地址,类型是int*的,就可以存放在parr数组中
  int* parr[3] = {arr1, arr2, arr3};
  int i = 0;
  int j = 0;
  for(i=0; i<3; i++){
    for(j=0; j<5; j++){
    printf("%d ", parr[i][j]);
    }
    printf("\n");
  }
  return 0;
}



图来展示:

e605160ae2a549f0bc259e7cbf8376ce.png


parr[i]是访问parr数组的元素,parr[i]找到的数组元素指向了整型⼀维数组,parr[i][j]就是整型⼀维数组中的元素。


但是并非是2维数组,原因是模拟的数组并不连续,而数组在内存上是连续存放的。


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