C语言学习篇(15)-----函数传参详解
https://m.toutiao.com/is/JpuAcLb/
前面我们已经介绍过什么是指针,指针变量的用法等等,今天我们就来讲讲什么是函数,函数有啥作用,函数的参数有哪些需要注意的地方以及指针与函数的关系。
首先函数是由一些代码块组成,这些代码往往都是为了完成某个特定功能的,使整个程序模块化,便于管理和维护。函数主体好比如是个加工厂,而传入的形参就像是材料,不同的厂加工不同的材料,因此我们有必要探究了下函数形参的本质。
函数传参详解
普通变量作为函数形参
特别说明的是,函数传参,传递是值,而不是变量本身。 经典的例子:两数交换。
以下定义a = 3,b = 5变量,我们想实现a与b值交换,具体代码如下:
#include <stdio.h>void swap1(int a, int b){int tmp;tmp = a;a = b;b = tmp;printf('in swap1, a = %d, b = %d.\n', a, b);}int main(void){int x = 3, y = 5;swap1(x, y);printf('x = %d, y = %d.\n', x, y);}
运行结果:
普通变量作为形参
从上图我们可以看到,在swap1函数中,确实a和b的值是互换了,但是在main主函数中,a和b仍然是3,5,并没有完成交换,因此交换失败。由此可得函数传参只是值的传递,而不是变量本身!!!
数组作为函数形参
以下我们定义数组int a[20], 并将a作为参数传入函数func1中,具体代码如下:
#include <stdio.h>void func1(int a[]){printf('sizeof(a) = %d.\n', sizeof(a));printf('in func1, a = %p.\n', a);}int main(void){int a[20];printf('a = %p.\n', a);func1(a);return 0;}
运行结果:
数组类型作为形参
我们可以看到a(数组的首元素的首地址)与传入函数func1的形参int a[]的a地址完全,这与我们之前的结论:函数传参,只传递值,而不是变量,相一致。 同时我们我们在func1函数中打印了形参int a[]中的a符号的数据长度,结果是8(测试环境是64位Ubuntu),而不是数组的长度(sizeof(int) * 20),因此我们可以得出数组作为形参传参时,实际传递是不是整个数组,而是数组的首元素的首地址。
指针作为函数形参
我们还是两数交换为例子,这次以指针作为形参,看看与上面的结果有何不同:
#include <stdio.h>void swap2(int *a, int *b){int tmp;tmp = *a;*a = *b;*b = tmp;printf('in swap2, *a = %d, *b = %d.\n', *a, *b);}int main(void){int x = 3, y = 5;swap2(&x, &y);printf('x = %d, y = %d.\n', x, y);return 0;}
运行结果:
指针类型作为形参
改用指针类型传参,不仅在函数swap2内实现了两数交换,并且在main主函数中成功交换。感受下与之前普通变量传参的区别?两者方式,本质都是传递的变量的值,但是使用指针方式作为形参,我们可以通过指针指向的地址,实际更改了变量本身的值!
结构体变量作为函数形参
结构体是C语言常用的数据类型,往往我们会将一个事物相关的变量,统一封装成一个结构体,便于管理维护,同时结构体类型变量通常较普通变量来说,数据量大,占内存较多。 那结构体作为函数形参会有什么不一样嘛?
#include <stdio.h>struct my_test{char a;int b;};void func1(struct my_test test1){printf('&test1 = %p.\n', &test1);printf('test1.a = %d.\n', test1.a);printf('test1.b = %d.\n', test1.b);}int main(void){struct my_test test = {.a = 3,.b = 55,};printf('&test = %p.\n', &test);printf('test.a = %d.\n', test.a);printf('test.b = %d.\n', test.b);func1(test);return 0;}
运行结果:
结构体类型作为形参
可以看到形参test1和实参test的结构成员值都是一致, 而test1和test两者的地址却不同,编译器只是将结构体test复制了一份,然后传入函数中,再次验证了函数传参,只传值,而不是变量本身!
如果我们改造下func1,形参使用结构体指针类型,具体如下:
#include <stdio.h>struct my_test{char a;int b;};void func1(struct my_test *test1){printf('test1 = %p.\n', test1);printf('test1->a = %d.\n', test1->a);printf('test1->b = %d.\n', test1->b);}int main(void){struct my_test test = {.a = 3,.b = 55,};printf('&test = %p.\n', &test);printf('test.a = %d.\n', test.a);printf('test.b = %d.\n', test.b);func1(&test);return 0;}
运行结果:
结构体指针类型作为形参
可以看出以结构体指针变量传参,形参test1和test是一样的,其实质都是指向0x7ffe5b29dd80为起始地址的结构体。
总结
上面我们介绍了普通变量传参,数组传参,指针参数,结构体传参等,都总结为一句话:函数传参,传递的是值(不管是普通变量,还是指针(地址)),而不是变量本身!!!
https://m.toutiao.com/is/JpuAcLb/