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

相关文章
|
2月前
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
52 0
|
10天前
|
存储 NoSQL 编译器
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
指针是一个变量,它存储另一个变量的内存地址。换句话说,指针“指向”存储在内存中的某个数据。
58 3
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
|
10天前
|
存储 编译器 C语言
【C语言】指针大小知多少 ?一场探寻C语言深处的冒险 !
在C语言中,指针的大小(即指针变量占用的内存大小)是由计算机的体系结构(例如32位还是64位)和编译器决定的。
35 9
|
10天前
|
安全 程序员 C语言
【C语言】指针的爱恨纠葛:常量指针vs指向常量的指针
在C语言中,“常量指针”和“指向常量的指针”是两个重要的指针概念。它们在控制指针的行为和数据的可修改性方面发挥着关键作用。理解这两个概念有助于编写更安全、有效的代码。本文将深入探讨这两个概念,包括定义、语法、实际应用、复杂示例、最佳实践以及常见问题。
32 7
|
20天前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
81 12
|
13天前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
14天前
|
算法 C语言
C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项
本文深入讲解了C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项,通过实例演示了文件操作的基本流程,帮助读者掌握这一重要技能,提升程序开发能力。
45 3
|
14天前
|
存储 算法 程序员
C 语言指针详解 —— 内存操控的魔法棒
《C 语言指针详解》深入浅出地讲解了指针的概念、使用方法及其在内存操作中的重要作用,被誉为程序员手中的“内存操控魔法棒”。本书适合C语言初学者及希望深化理解指针机制的开发者阅读。
|
20天前
|
存储 C语言 开发者
C 语言指针与内存管理
C语言中的指针与内存管理是编程的核心概念。指针用于存储变量的内存地址,实现数据的间接访问和操作;内存管理涉及动态分配(如malloc、free函数)和释放内存,确保程序高效运行并避免内存泄漏。掌握这两者对于编写高质量的C语言程序至关重要。
42 10
|
14天前
|
程序员 C语言
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门。本文深入探讨了指针的基本概念、声明方式、动态内存分配、函数参数传递、指针运算及与数组和函数的关系,强调了正确使用指针的重要性,并鼓励读者通过实践掌握这一关键技能。
29 1