C语言---深入指针(2)(一)

简介: C语言---深入指针(2)

1.数组名

//int main()
//{
//    int arr[10] = { 1,2,3,4,5,6,7,8,9,10 }; 
//    //int *p=&arr[0];//把首元素的地址取出来放到p里面去--p就指向了数组里面的1
//    printf("&arr[0]=%p\n", &arr[0]);//首元素的地址
//    printf("&arr=%p\n", &arr);
//    //输出结果是
//    // &arr[0]=00E1F8E0
//    //&arr = 00E1F8E0
//    //可以发现两个地址一样
//
//    
//    return 0;
//}
//其实数组名就是数组首元素的地址
 
 
//int main()
//{
//    int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
//    
//    printf("%d\n",sizeof(arr));//打印的字节是40
//    printf("%d\n", sizeof(arr[0]));//打印的字节是4
//    
//
//    return 0;
//}
//但是有两个例外
//1.sizeof(数组名)--这里的数组表示整个数组,计算是整个数组的大小,单位是字节
//2.&数组名---也表示整个数组,取出的使整个数组的地址
 
 
//除此之外,所有的数组名都是数组首元素的地址
 
//产生疑问:整个数组的地址和数组首元素的地址有什么区别呢?
//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);//也是数组首元素的地址
//    printf("&arr   =%p\n", &arr);//数组的地址
//    //输出结果是
//    // &arr[0]=0000002F78F0FD50
//    //arr     =0000002F78F0FD50
//    //& arr   =0000002F78F0FD50
//    return 0;
//}
//但是这三个地址真的没有差异吗?
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);//数组的地址
    //输出结果是
    /*  &arr[0]      = 000000CA4590FC10
        & arr[0] + 1 = 000000CA4590FC14
        arr          = 000000CA4590FC10
        arr +     1  = 000000CA4590FC14
        & arr        = 000000CA4590FC10
        & arr +   1  = 000000CA4590FC38     */
    //不难发现,一个数组的地址+1跳过的是整个数组
    //单只数组首元素的地址+1跳过的是4个字节
 
    //所以我们&arr,&数组名的时候,取出的就是整个数组的地址
 
    return 0;
}
//以后在做题的时候一定要看看前面有没有sizeof或者前面加没加取地址符号
//数组名的地址就是数组首元素的地址,arr和&arr[0]
//如果是&arr,那么这个就是整个数组的地址了---&数组名
 
 
/*
printf("&arr[0]  =%p\n", &arr[0]);---整型的地址取出来,它的类型是int*
printf("&arr[0]+1=%p\n", &arr[0] + 1);
 
 
printf("arr      =%p\n", arr);-----类型也是int *
printf("arr+1    =%p\n", arr + 1);
 
 
 
printf("&arr     =%p\n", &arr);-----与上面的两种类型不一样
printf("&arr+1   =%p\n", &arr + 1);*/
//类型不一样,指针+1或者解引用就不一样

数组名的地址就是数组首元素的地址,arr和&arr[0]

如果是&arr,那么这个就是整个数组的地址了---&数组名

2.使用指针访问数组

//那我们能使用指针的方式访问数组吗?
//int main()
//{
//    int arr[10] = { 0 };
//    int sz = sizeof(arr) / sizeof(arr[0]);
//    //输入
//
//    int* p = &arr[0];//将数组首元素地址取出来放到指针变量p里面去
//
//    //将首元素的地址存在p里面甚至还能这样写
//    //int* p = arr;
//    for (int i = 0; i < sz; i++)
//    {
//        scanf("%d", p+i);//p放的是首元素的地址,p+i是下标为i的地址
//        //也可以这样写scanf("%d", arr+ i);
//    }
//    //输出
//    for (int i = 0; i < sz; i++)
//    {
//        printf("%d ", *(p+i));//解引用
//        //printf("%d ", *(arr+i));
//        //printf("%d ",i[arr] );
//    }
//    return 0;
//}
 
//*(arr+i)<====>arr[i]
//左边是指针的形式,右边是数组的形式
//arr+i是地址,地址解引用,找到的就是这个地址指向的那个数
//右边的是通过数组下标来访问对应的数字
//arr[i]<====>*(arr + i) <====>*(i+arr)<====> i[arr]
//这里的[]是操作符
 
//1.数组就是数组,是一块连续的空间,是可以存放一个或者多个数组的
//2.指针变量是一个变量,是可以存放地址的变量
//数组和指针不是一个东西
//但是可以使用指针来访问数组的
 
//为什么可以使用指针来访问数组呢?
//1.数组在内存中失败连续存放的
//2.指针的元素很方便的可以遍历数组,取出数组的内容

arr[i](arr + i) (i+arr) i[arr]

3.一位数组传参的本质

/*void test(int arr[])
{
    int sz2 = sizeof(arr) / sizeof(arr[0]);
    printf("sz2=%d", 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);//10
 
    test(arr);//arr是数组名,数组名表示的是数组首元素的地址
    return 0;//传过去的是地址,要用指针接收
}*/
//打印结果为
// sz1=10
//sz2 = 2
//为什么二者结果不一样
 
//test(arr)--一维数组传参的本质,传递的是首元素的地址
//所以形参即使写成数组的形式,本质上也是一个指针变量
//所以在函数的开头,我们要用指针接收
 
//数组传参的时候,形参可以写成数组,也可以写成指针‘
//写成数组的形式,最简单,是为了方便,容易接受这种语法
//但是即是写成数组的形式,本质上还是指针
/*void test(int* arr)
{
    int sz2 = sizeof(arr) / sizeof(arr[0]);//传过来的arr是首元素的地址,所以4/4=1
    printf("sz2=%d", sz2);
}
 
int main()
{
    int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
    int sz1 = sizeof(arr) / sizeof(arr[0]);//这里的arr是一整个数组的,所以是40/4=10
 
    printf("sz1=%d\n", sz1);//10
 
    test(arr);
    return 0;
}*/
 
//通常情况下数组传参我们会怎么做呢?
//
//void test(int* arr,int sz)
//{
//    for (int i = 0; i < sz; i++)
//    {
//        printf("%d", arr[i]);//printf("%d", *(arr+i));
//        //*(arr + i)=== arr[i]//i是从0开始的
//    }
//
//}
//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);//10
//
//    test(arr,sz1);//将元素个数传过去
//    return 0;
//}

C语言---深入指针(2)(二)https://developer.aliyun.com/article/1544341

相关文章
|
1月前
|
存储 C语言
【C语言篇】深入理解指针3(附转移表源码)
【C语言篇】深入理解指针3(附转移表源码)
35 1
|
16天前
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
41 0
|
15天前
|
C语言
【c语言】指针就该这么学(3)
本文介绍了C语言中的函数指针、typedef关键字及函数指针数组的概念与应用。首先讲解了函数指针的创建与使用,接着通过typedef简化复杂类型定义,最后探讨了函数指针数组及其在转移表中的应用,通过实例展示了如何利用这些特性实现更简洁高效的代码。
11 2
|
16天前
|
C语言
如何避免 C 语言中的野指针问题?
在C语言中,野指针是指向未知内存地址的指针,可能引发程序崩溃或数据损坏。避免野指针的方法包括:初始化指针为NULL、使用完毕后将指针置为NULL、检查指针是否为空以及合理管理动态分配的内存。
|
16天前
|
C语言
C语言:哪些情况下会出现野指针
C语言中,野指针是指指向未知地址的指针,通常由以下情况产生:1) 指针被声明但未初始化;2) 指针指向的内存已被释放或重新分配;3) 指针指向局部变量,而该变量已超出作用域。使用野指针可能导致程序崩溃或不可预测的行为。
|
22天前
|
存储 C语言
C语言32位或64位平台下指针的大小
在32位平台上,C语言中指针的大小通常为4字节;而在64位平台上,指针的大小通常为8字节。这反映了不同平台对内存地址空间的不同处理方式。
|
22天前
|
存储 算法 C语言
C语言:什么是指针数组,它有什么用
指针数组是C语言中一种特殊的数据结构,每个元素都是一个指针。它用于存储多个内存地址,方便对多个变量或数组进行操作,常用于字符串处理、动态内存分配等场景。
|
22天前
|
存储 C语言
C语言指针与指针变量的区别指针
指针是C语言中的重要概念,用于存储内存地址。指针变量是一种特殊的变量,用于存放其他变量的内存地址,通过指针可以间接访问和修改该变量的值。指针与指针变量的主要区别在于:指针是一个泛指的概念,而指针变量是具体的实现形式。
|
23天前
|
C语言
C语言指针(3)
C语言指针(3)
11 1
|
23天前
|
C语言
C语言指针(2)
C语言指针(2)
12 1