C语言指针精简版(二)

简介: C语言指针精简版(二)

数组名的理解

对于数组名表示的意义一共有三种情况:

*1、数组名:表示数组⾸元素地址。

*2、sizeof(数组名):表⽰整个数组,计算整个数组的⼤⼩,单位是字节

*3、&数组名,表⽰整个数组,取出的是整个数组的地址

小拓展:

      在实际应用中,*1为一般情况,*2、*3为特殊情况,编译器在遇到arr[i]的时候会将它转为*(arr+i)的形式进行计算,所以理论上有这种写法,但不推荐:

arr[i]   ==   *(arr+i)    ==  *(i+arr)  ==  i[arr]

同时这也是为什么我们有了数组首元素地址arr和一个[i]就可以通过for循环实现遍历数组元素

#include <assert.h>
#include <stdio.h>
int my_strlen(const char* str)    //统计所求字符串长度函数,加上const修饰*str,表示不能修改这个字符串,防止修改str的内容
{
    int count = 0;
    assert(str != NULL);    //确保了指针的有效性,str为空指针就报错了。
    while (*str != '\0')
    {
        count++;     //次数加一
        str++;       //地址加一
    }
    return count;   //返回统计的个数
}
int main()
{
    char arr[] = "hello bit";
    int len = my_strlen(arr);
    printf("%d\n", len);
    return 0;
}

使⽤指针访问数组

#include <stdio.h>
int main()
{
  int arr[10] = { 0 };
  //输⼊
  int i = 0;
  int sz = sizeof(arr) / sizeof(arr[0]);
  //输⼊
  int* p = arr;
  for (i = 0; i < sz; i++)
  {
    scanf_s("%d", p + i);
    //scanf("%d", arr+i);//也可以这样写
        //&arr[i]也行
  }
  //输出
  for (i = 0; i < sz; i++)
  {
    printf("%d ", *(p + i));
        //写成printf("%d ", p[i]);也许
  }
  return 0;
}

⼀维数组传参的本质

       我们之前都是在函数外部计算数组的元素个数,那我们可以把函数传给⼀个函数后,函数内部求数组的元素个数吗?

#include <stdio.h>
void test(int arr[])
{
  int sz2 = sizeof(arr) / sizeof(arr[0]);
  printf("sz2 = %d\n", 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);
  test(arr);
  return 0;
}

       很明显不可以,这是因为我们传递的是数组名,而它又代表数组⾸元素的地址,那么在函数内部我们写sizeof(arr) 计算的其实是⼀个指针变量的⼤⼩⽽不是数组的⼤⼩......

#include <stdio.h>
void test1(int arr[])//参数写成数组形式,本质上还是指针
{
  printf("%d\n", sizeof(arr));//计算⼀个指针变量的⼤⼩
}
}
void test2(int* arr)//参数写成指针形式
{
  printf("%d\n", sizeof(arr));//计算⼀个指针变量的⼤⼩
}
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
  test1(arr);
  test2(arr);
  return 0;
}

所以应将计算数组元素个数的操作放在传参前进行

总结:⼀维数组传参,传递的是数组首元素地址,形参的部分可以写成数组或指针的形式

冒泡排序

核心思想:两两相邻的元素进⾏⽐较

#include <stdio.h>
#include <stdbool.h>
void bubble_sort(int arr[], int sz)
{
  int i = 0;
  for (i = 0; i < sz - 1; i++)
  {
    _Bool flag = false;//flag用于标志是否进行了排序,假设整个序列不需要排序,将flag设为false
    int j = 0;
    for (j = 0; j < sz - i - 1; j++)
    {
      if (arr[j] > arr[j + 1])
      {
        flag = true;//当进入if语句后,证明确实需要排序,此时将flag赋值为true
        int tmp = arr[j];
        arr[j] = arr[j + 1];
        arr[j + 1] = tmp;
      }
    }
    //如果整个数组仍需排循环序,那么在进行完一次内部foe循环后flag的值变为了true(因为进入了if语句)
    //如果整个数组已经排序完成不需要循环了,那么就不会进入if语句中修改flag的值,此时flag为false
    if (flag == false)//当外层for的一次循环执行至此时,如果flag的值仍为flase,则外层for循环break,若为true,则继续下一轮循环
      break;  
  }
}
int main()
{
  int arr[] = { 3,1,7,5,8,9,0,2,4,6 };
  int sz = sizeof(arr) / sizeof(arr[0]);
  bubble_sort(arr, sz);
  //遍历打印冒泡排序后的数组元素
  for (int i = 0; i < sz; i++)
  {
    printf("%d ", arr[i]);
  }
  return 0;
}

⼆级指针及其解引用

指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪⾥?

答案:二级指针

指针变量前面有几个*就是几级指针,一般来说三级指针就是极限

#include <stdio.h>
int main()
{
  int a = 10;
  int b = 20;
  int* pa= &a;   
  printf("刚开始一级指针pa中存放的地址为:%p\n", pa);  
  *pa = 20;
  printf("对一级指针解引用一次后赋值20后变量a的值由10变为了:%d\n", a);
  int** ppa = &pa;
  **ppa = 100;
  printf("对二级指针解引用两次并赋值100后变量a的值由20变为了:%d\n", a);
  *ppa = &b;
  printf("对二级指针解引用一次并让其指向变量的b的地址后,一级指针pa中存放的地址为:%p\n", pa);
  **ppa = 30;
  printf("%d", b);
  return 0;
}

以下是在程序调试过程中各变量值的变化过程(红色表示此时值发生了改变) :

由于*pa=20的原因,pa于&pa中的值都会发生变化

接下来用一个二级指针ppa获取一级指针pa的地址,该二级指针的地址变为0x0000002c254ff678

对二级指针的两次解引用后赋值为100,可以发现**ppa值的改变也会引起*pa值的改变

对二级指针一次解引用后将b的地址赋值给它,可以发现*ppa值的改变也会引起pa的值改变

最后再更改b的值

关于上述代码的解释:pa为一级指针它中存放的是变量a的地址,ppa为二级指针它中存放的是一个一级指针的地址,对二级指针ppa进行解引用操作,一次解引用后,可修改到指针变量pa在内存中存放的内容,两次解引用后,便可以修改到变量a在内存中存放的内容。甚至你可以这样理解:对二级指针一次解引用后得到的其实就是它所指向的一个一级指针,你也不用关心它地址啊什么的问题你就直接把它当作一级指针用就行,对二级指针的解引用后得到的其实就是那个一级指针指向的某个变量,你把它当作那个变量使用就行

指针数组

学习指针数组之前,我们已经学过了多种数组比如:

  • 整型数组:存放整型的数组  int arr[10]
  • 字符数组:存放字符的数组  char arr[10]

       无论是整型数组还是字符数组,它们当中存放的元素类型都是一样的,而接下来要学的指针数组的也与它们相似......

指针数组的特点:数组指针的每个元素都是⽤来存放地址(指针)的(元素的类型都为指针类型)

指针数组的格式:指针变量类型 数组名[存放的地址(指针)个数]

实例:指针数组模拟⼆维数组

#include <stdio.h>
int main()
{
    int arr1[] = { 1,2,3,4,5 }; //定义三个整型数组
    int arr2[] = { 2,3,4,5,6 };
    int arr3[] = { 3,4,5,6,7 };
    int* parr[3] = { arr1, arr2, arr3 };//将三个整型数组存放在int*类型的指针数组parr中,其中的arr1、arr2和arr3分别代表上述三个整型数组的首元素地址
    int i = 0;
    int j = 0;
    for (i = 0; i < 3; i++)    //i表示存放在int*类型的指针数组parr中的指针的个数
    {
        for (j = 0; j < 5; j++)    //j表示每一个指针指向的数组中的元素个数
        {
            printf("%d ", parr[i][j]);  // parr[i][j]  == >  *(*(parr+i)+j)
        }
        printf("\n");
    }
    return 0;
}

~over~

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

热门文章

最新文章