C语言学习笔记(二):指针作为参数(入参、反参)
1、指针作为函数的传入参数
字符数组作为形参:
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
void print(int arr[10])
{
printf("arr作为形参sizeof(arr):%d\n", sizeof(arr));
printf("arr[1]作为形参sizeof(arr[1]):%d\n", sizeof(arr[1]));
printf("形参arr = %p\n", arr);
printf("形参&arr[0] = %p\n", &arr[0]);
printf("形参&arr = %p\n", &arr);
printf("形参*arr = %d\n", *arr);
}
int main()
{
int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
printf("arr作为实参sizeof(arr):%d\n", sizeof(arr));
printf("arr[0]作为实参sizeof(arr[0]):%d\n", sizeof(arr[0]));
print(arr);
/*运行结果
arr作为实参sizeof(arr):40
arr[0]作为实参sizeof(arr[0]):4
arr作为形参sizeof(arr):4
arr[1]作为形参sizeof(arr[1]):4
形参arr = 00CFFA04
形参&arr[0] = 00CFFA04
形参&arr = 00CFF930
形参*arr = 1
*/
}
运行结果如下:
可知当数组arr作为参数传递给print函数时 实际上是将数组arr第一个元素的地址给传递了出去,这个指针就存在形参arr所在的内存空间(009BFA30)中,其中arr的值与arr[0]的地址是一样的 。
由运行结果可知 当字符数组作为指针被传入函数当中时 我们无法在函数内部拿到此数组的长度。
因为函数内部只知道此字符数组的首地址,不知道字符数组的末地址。
当数组为字符串时例外,比如char arr[] ="hello world"
,可以通过strlen函数直接拿到此字符串数组的长度,因为字符串都是以\0结尾的。拿到了首地址 又知道末地址里面存的为\0 就算不用strlen函数也可以遍历字符串数组里面的值进行 判断计数,当值为0时字符串结束。
2、指针作为函数的返回参数
test函数的返回值类型为字符指针
char* test()
{
//char arr[] = "hello world";
//const char arr[] = "hello world";
char* arr = "hello world";
return arr;
}
int main()
{
char* p = test();
printf("%p\n",p);
printf("%p\n", &p);
printf("%c\n", *p);
printf("%c\n", p[0]);
printf("%s\n", p);
第一个问题:当arr[]="hello world"
时,arr是存在内存中的栈区会随着函数执行完成而消失,当主函数中指针p接受test函数的返回值arr时 ,p中存储的是arr数组的第一个元素的地址 地址是确确实实的返回了 但地址所指向的内存里面的值已经被回收。下图中第一个地址是数组第一个元素的内存地址,第二个地址为指针p本身内存的地址。
第二个问题:当char* arr = "hello world";
时, arr会被分配到内存的常量区,也就是说返回指针arr(或是说指针p)指向的是常量区的一个字符串首地址,所以在函数结束时 常量区的值不会消失 ,这时候再去读数组里面的值就不会出乱码,如下图。
**第三个问题★★★★★:**当const char arr[] = "hello world";
时。当在函数内使用const修饰arr时 同样也出现了值被释放的问题。但当吧这句定义放在函数外面的时候 就不会出现值被释放的问题,不知是不是函数内部定义的常量没有意义 或是此常量被存放在了栈区从而随着函数结束而释放的问题。这里留个疑问。再放个大佬的文章工大家参考:内存中各个区的划分及特性。