码农知识堂 - 1000bd
  •   Python
  •   PHP
  •   JS/TS
  •   JAVA
  •   C/C++
  •   C#
  •   GO
  •   Kotlin
  •   Swift
  • C语言指针从基础到进阶


    数组名的理解

    还记得上一篇博客出现了这么一段关于用指针访问数组的代码。

    1. int arr[10] = {1,2,3,4,5,6,7,8,9,10};
    2. int *p = &arr[0];

    这里用&arr[0]拿到了数组首元素的地址,但其实数组名在大部分情况下本来就是首元素的地址

    1. #include
    2. int main()
    3. {
    4. int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
    5. printf("&arr[0] = %p\n", &arr[0]);
    6. printf("arr = %p\n", arr); //这两个printf的输出是一样的
    7. return 0;
    8. }

     关于详细情况可以在我的另一篇博客了解:C语言中关于数组名什么情况为首元素地址,什么情况为整个数组的地址的问题。-CSDN博客但是&arr和&arr+1相差40个字节,这就是因为&arr是数组的地址,+1操作是跳过整个数组的。这⾥我们发现&arr[0]和&arr[0]+1相差4个字节,arr和arr+1相差4个字节,是因为&arr[0]和arr都是。• &数组名,这⾥的数组名表⽰整个数组,取出的是整个数组的地址(整个数组的地址和数组⾸元素的地址是有区别的)• sizeof(数组名),sizeof中单独放数组名,这⾥的数组名表示整个数组,计算的是整个数组的⼤⼩,除此之外,任何地⽅使⽤数组名,数组名都表⽰⾸元素的地址。https://blog.csdn.net/2301_80194476/article/details/136611525?spm=1001.2014.3001.5501

    使用指针访问数组

    有了前面的基础,我们就可以很方便的用指针访问数组了

    1. #include
    2. int main()
    3. {
    4. int arr[10] = {0};
    5. //输⼊
    6. i = 0;
    7. int sz = sizeof(arr)/sizeof(arr[0]);
    8. //输⼊
    9. int* p = arr;
    10. for(i=0; i
    11. {
    12. scanf("%d", p+i);
    13. //scanf("%d", arr+i);//也可以这样写
    14. }
    15. //输出
    16. for(i=0; i
    17. {
    18. printf("%d ", *(p+i));
    19. }
    20. return 0;
    21. }

     在上面的代码的中将p+i换成p[i]也能正常打印本质上p[i]和*(p+i)是一样的,同理arr[i]等价于*(arr+i)。

    一维数组传参的本质

    这里直接给答案:数组名是数组⾸元素的地址;那么在数组传参的时候,传递的是数组名,也就是说一维数组传参的本质就是传递数组首元素的地址。

    1. include
    2. void test1(int arr[])//参数写成数组形式,本质也是指针
    3. {
    4. int sz1 = sizeof(arr)/sizeof(arr[0]);
    5. printf("sz1 = %d\n", sz1);//结果为1
    6. }
    7. void test2(int* arr)//参数写成指针形式
    8. {
    9. int sz2 = sizeof(arr)/sizeof(arr[0]);
    10. printf("sz2 = %d\n", sz2);//计算⼀个指针变量的⼤⼩
    11. }
    12. int main()
    13. {
    14. int arr[10] = {1,2,3,4,5,6,7,8,9,10};
    15. int sz = sizeof(arr)/sizeof(arr[0]);
    16. printf("sz = %d\n", sz);//结果为10
    17. test1(arr);
    18. test2(arr);
    19. return 0;
    20. }

    输出结果:

     

     

    冒泡排序

     后面我会出一期专门的排序算法,手撕常见排序算法!这里主要说关于指针的知识。

    1. void bubble_sort(int arr[], int sz)//参数接收数组元素个数
    2. {
    3. int i = 0;
    4. for(i=0; i-1; i++)
    5. {
    6. int j = 0;
    7. for(j=0; j-1; j++)
    8. {
    9. if(arr[j] > arr[j+1])
    10. {
    11. int tmp = arr[j];
    12. arr[j] = arr[j+1];
    13. arr[j+1] = tmp;
    14. }
    15. }
    16. }
    17. }
    18. int main()
    19. {
    20. int arr[] = {3,1,7,5,8,9,0,2,4,6};
    21. int sz = sizeof(arr)/sizeof(arr[0]);
    22. bubble_sort(arr, sz);
    23. for(i=0; i
    24. {
    25. printf("%d ", arr[i]);
    26. }
    27. return 0;
    28. }
    29. //⽅法2 - 优化
    30. void bubble_sort(int arr[], int sz)//参数接收数组元素个数
    31. {
    32. int i = 0;
    33. for(i=0; i-1; i++)
    34. {
    35. int flag = 1;//假设这⼀趟已经有序了
    36. int j = 0;
    37. for(j=0; j-1; j++)
    38. {
    39. if(arr[j] > arr[j+1])
    40. {
    41. flag = 0;//发⽣交换就说明,⽆序
    42. int tmp = arr[j];
    43. arr[j] = arr[j+1];
    44. arr[j+1] = tmp;
    45. }
    46. }
    47. if(flag == 1)//这⼀趟没交换就说明已经有序,后续⽆序排序了
    48. break;
    49. }
    50. }
    51. int main()
    52. {
    53. int arr[] = {3,1,7,5,8,9,0,2,4,6};
    54. int sz = sizeof(arr)/sizeof(arr[0]);
    55. bubble_sort(arr, sz);
    56. for(i=0; i
    57. {
    58. printf("%d ", arr[i]);
    59. }
    60. return 0;
    61. }

    二级指针

    指针变量也是变量啊,那么指针变量也会有地址,指针变量的地址就储存在二级指针变量里面,二级指针变量也有地址,就存在三级指针变量里面……

    1. int a = 10;
    2. int* pa = &a;
    3. printf("%d\n",*pa);
    4. int** ppa = &pa;//将一级指针pa的地址存到二级指针ppa里面
    5. printf("%d\n",**ppa);//二级指针经过两次解引用找到a

    指针数组

    指针数组是指针还是数组???

    其实指针数组还是数组,只不过这个数组里面放的都是指针。里面的每个元素都是int*类型的。

    我们可以类⽐⼀下,整型数组,是存放整型的数组,字符数组就是存放字符的数组。

    指针数组模拟二维数组

    1. int main()
    2. {
    3. int arr1[] = {1,2,3,4,5};
    4. int arr2[] = {2,3,4,5,6};
    5. int arr3[] = {3,4,5,6,7};
    6. //数组名是数组⾸元素的地址,类型是int*的,就可以存放在parr数组中
    7. int* parr[3] = {arr1, arr2, arr3};
    8. int i = 0;
    9. int j = 0;
    10. for(i=0; i<3; i++)
    11. {
    12. for(j=0; j<5; j++)
    13. {
    14. printf("%d ", parr[i][j]);
    15. }
    16. printf("\n");
    17. }
    18. return 0;
    19. }

     

    parr[i]是访问parr数组的元素,parr[i]找到的数组元素指向了整型⼀维数组,parr[i][j]就是整型⼀维数组中的元素。
    上述的代码模拟出⼆维数组的效果,实际上并⾮完全是⼆维数组,因为每⼀⾏并⾮是连续的。
    本期博客到这里就结束了,如果有什么错误,欢迎指出,如果对你有帮助,请点个赞,谢谢!

  • 相关阅读:
    全景环视前装市场“变天”,这家中国本土供应商首次跻身份额第一
    Linux:命令行参数和环境变量
    uni-app - 城市选择索引列表 / 通过 A-Z 排序的城市列表(uview 组件库 IndexList 索引列表)
    Cesium 展示——读取文件——加载 geojson 文件数据
    GRS认证与TC交易证明的区别
    机器学习(十八):随机搜索和XGBoost
    C++代码转C#代码,制作DLL,以供python调用
    刚考过PMP想问一下怎样才能转行做PM
    Linux基础入门到精通之Linux系统安装日志
    【Java 基础篇】Properties 结合集合类的使用详解
  • 原文地址:https://blog.csdn.net/2301_80194476/article/details/136611385
  • 最新文章
  • 攻防演习之三天拿下官网站群
    数据安全治理学习——前期安全规划和安全管理体系建设
    企业安全 | 企业内一次钓鱼演练准备过程
    内网渗透测试 | Kerberos协议及其部分攻击手法
    0day的产生 | 不懂代码的"代码审计"
    安装scrcpy-client模块av模块异常,环境问题解决方案
    leetcode hot100【LeetCode 279. 完全平方数】java实现
    OpenWrt下安装Mosquitto
    AnatoMask论文汇总
    【AI日记】24.11.01 LangChain、openai api和github copilot
  • 热门文章
  • 十款代码表白小特效 一个比一个浪漫 赶紧收藏起来吧!!!
    奉劝各位学弟学妹们,该打造你的技术影响力了!
    五年了,我在 CSDN 的两个一百万。
    Java俄罗斯方块,老程序员花了一个周末,连接中学年代!
    面试官都震惊,你这网络基础可以啊!
    你真的会用百度吗?我不信 — 那些不为人知的搜索引擎语法
    心情不好的时候,用 Python 画棵樱花树送给自己吧
    通宵一晚做出来的一款类似CS的第一人称射击游戏Demo!原来做游戏也不是很难,连憨憨学妹都学会了!
    13 万字 C 语言从入门到精通保姆级教程2021 年版
    10行代码集2000张美女图,Python爬虫120例,再上征途
Copyright © 2022 侵权请联系2656653265@qq.com    京ICP备2022015340号-1
正则表达式工具 cron表达式工具 密码生成工具

京公网安备 11010502049817号