【C语言初阶】指针的运算or数组与指针的关系你了解吗?

简介: 【C语言初阶】指针的运算or数组与指针的关系你了解吗?

📋 前言

   🌈hello! 各位宝子们大家好啊,前面给大大家介绍了指针的基础知识。那么这一章就来给大家介绍下一部分内容!

   ⛳️指针的运算 以及 数组与指针的关系 也是我们在指针的学习过程中非常重要的!

   📚本期文章收录在《C语言初阶篇》,大家有兴趣可以看看呐

  ⛺️ 欢迎铁汁们 ✔️ 点赞 👍 收藏 ⭐留言 📝!

🔥 注:前一章的指针讲解是《指针的概念》

💬 指针运算

💭 指针±整数

其实在上一章我们就见过指针加减整数了对于 type * p的指针想要加 nn 是这样计算的

  • n*sizeof(type)

📚 代码演示:

#include <stdio.h>
int main()
{
  int arr[10] = { 0 };
  int* p = &arr[0];
  //不用下标访问
  int sz = sizeof(arr) /sizeof(arr[0]);
  int i = 0;
  for (i = 0; i < sz; i++)
  {
    *p = i;
     p = p+1;
  }
  p = &arr[0];
  for (i = 0; i < sz; i++)
  {
    printf("%d\n",*(p+i));
  }
  return 0;
}

这里我们就使用指针加减整数的方式来进行访问数组进行赋值

  • 每次让指针p 向前指4个字节,然后进行赋值!
  • 之后再把指针p 归位到最开始的指向位置进行访问
  • 33e077831a9a4808ada3d22d503ef645.png

前面的例子懂了我们再来看这个例子大家看看!

📚 代码演示:

#define N_VALUES 5
float values[N_VALUES];
float *vp;
//指针+-整数;指针的关系运算
for (vp = &values[0]; vp < &values[N_VALUES];)
{
     *vp++ = 0;
}

这个例子其实和前面很像,数组中是从地址值向高地址使开辟的一份连续的空间!这里我们把 数组的第一个地址赋给指针p

  • 所以指针 p 肯定没有 &values[5]的地址大!
  • p++ 每次向前指向下一个元素给每个元素赋值,直到把前五个都赋完!
  • e174cec4a23542ff9bd324d911820040.png

💭 指针-指针

指针加减整数的例子我们了解完了,下面就来了解了解指针加减指针的应用把!

  • 不知道大家了解个库函数strlen 的实现嘛/
  • 今天就用指针-指针的方法实现一下!
    🔥 注:库函数的模拟实现链接,大家感兴趣可以看看呢!《strlen的三种实现方法》

📚 代码演示:

#include <stdio.h>
int my_strlen(char* s)
{
    char* p = s;
    while (*p != '\0')
        p++;
    return p - s;
}
int main()
{
    char arr[] = "abcdef\0";
    int len = my_strlen(arr);
    printf("%d\n", len);
  return 0;
}

⛳️ 诶这里就巧妙的应用了指针里面存放的地址来进行减法运算,用差做取值。刚好能求字符的长度的。

  • 因为一个字符也就只能只能存放一个字节
  • 而每一个字节都有地址连续的空间地址就是递增的!

💭 指针的关系运算

指针减指针的运算我们学了其实指针的运算还有一条规定:

C语言标准规定:

  • 允许指向数组元素的指针与指向数组最后一个元素后面的那个内存位置的指针比较,但是不允许与指向第一个元素之前的那个内存位置的指针进行比较。
  • 我们来看一下这段代码:

📚 代码演示:

for(vp = &values[5-1]; vp >= &values[0];vp--)
{
    *vp = 0;
}

这段代码有什么问题呢?其实这段代码是倒着往回赋值的从数组 values[4],一直向后赋值但是

  • 当最后一个元素 values[0] 赋完值的时候
  • 循环调整部分 vp-- 还会执行一次导致 vp 成为野指针
  • 所以,当我们在次想用 vp 的是就需出现访问越界的问题!

8c627af3b0a9445aa18a97cb7dea9a64.png

📚 代码改进:

for(vp = &values[5-1]; vp > &values[0];)
{
    *--vp = 0;
}

这样当指针 vp 赋完值之后就不会再进行减减的动作了。因为根本就进不来循环里面所以想前面的错误我们就避免了。

🔥 但是:这并不符合C语言的标准,所以我们尽量不要这样写以免有些编译器跑不过去。

💬 指针和数组

📚 代码演示:

#include <stdio.h>
int main()
{
 int arr[10] = {1,2,3,4,5,6,7,8,9,0};
    printf("%p\n", arr);
    printf("%p\n", &arr[0]);
    return 0;

📑 代码结果:

aa81b91fb34e47d19eca3cdea3ca5280.png

  ⛳️ 诶这里就可以看到,其实数组名和数组第一个元素的地址是一样,所以可见数组名和数组首元素的地址是一样的。

  • 那么这样写代码是可行的:
int arr[10] = {1,2,3,4,5,6,7,8,9,0};
int *p = arr;//p存放的是数组首元素的地址
//等同于 *p= &arr[0];

💭 指针和数组变种推演

而指针是用来存放地址的,数组又是一块连续的空间。所以我们可以拿指针变量来接收数组首元素的地址,用来遍历数组!

  • 所以我们对指针解引用 和 数组下标是相等的。
  •     arr[ i ] == *(p+i)
  • 而   *(i+p) == arr[ i ]
  • 所以  *(i+p) == i[ arr ]
  • 而 : p == arr == &arr[ 0 ]
  • 所以  *()操作符 等同于[ ]

📚 代码演示:

#include <stdio.h>
int main()
{
    int arr[] = { 1,2,3,4,5,6,7,8,9,0 };
    int* p = arr; //指针存放数组首元素的地址
    int sz = sizeof(arr) / sizeof(arr[0]);
    int i = 0;
    for (i = 0; i < sz; i++)
    {
        arr[i] = i;
        printf("%d %d %d %d\n",arr[i],i[arr],*(p+i),p[i]);
    }
    return 0;
}

📑 代码结果:

af88a292cc8142eaa7605bfe59254fc2.png

  • 既然这俩个操作符是相等的那么又能推演出:
  • *(arr+i) == arr[ i ] == *(i+arr)
    🔥 注:这里就不给大家演示了,大家可以自己去试着练一下呢。

💭 指针和数组的关系

  ⛳️前面我们看到了,数组和指针的应用也给大家带来不一样的数组写法是不是很有趣!那么他们究竟有什么关系?既然指针和数组使用使用起来一样的效果那么我们可以把指针理解为数组嘛?

  • 其实我们可以这样理解
  • 指针变量就是指针变量,不是数组。指针变量的大小是4/8个字节,是用来专门存放地址的.
  • 数组就是数组,不是指针,数组是一块连续的空间。可以存放一个或,多个类型相同的数据。

🔰 那么他们的联系是什么?

  • 🌱 数组中,数组名其实是数组首元素的地址,数组名 == 地址 == 指针
  • 🌱 当我们知道数组首元素地址的时候,因为数组又是连续存放的,所以通过指针就可以来遍历访问我们的数组。

💬 二级指针

  ⛳️ 指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪里?答案是二级指针,存放指针变量的地址我们叫做二级指针!

📚 代码演示:

#include <stdio.h>
int main()
{
  int a = 10;
  int* p = &a;//p是一级指针变量,指针变量也是变量,
  //变量是在内存中开辟空间的,是变量就有地址
  int** pp = &p;//pp就是二级指针变量,
  //二级指针变量就是用来存放一级指针变量的地址
    return 0;
}

⛳️ 二级指针变量就是用来存放一级指针的地址,那么怎么来理解int**

  • int** 第一个 int* 说明了 pp 是个指针变量
  • 而第二个 * 则说明了 pp 里面存放的是指针变量的地址

9b343fcf2b6e46ea9b7dcc9e08a2bc78.png

💬 二级指针的运算

所以我们对 pp 指针解引用一次是找到一级指针的内容

  • 所以:*pp 等价于 p == &a

所以我们想用二级指针找到一级指针所指向的对象 a 就需要进行俩次解引用

📚 代码演示:

#include <stdio.h>
int main()
{
  int a = 10;
  int* p = &a;
  int** pp = &p;
  **pp = 100;
  printf("%d", a);
    return 0;
}

📑 代码结果:

98d210bab91a4c24a48f4f0169a9f59e.png

📝全篇总结

✅ 归纳:

好了以上就是关于指针的运算 数组和指针的关系 以及 二级指针的概念我们就全讲解完毕了!

  指针的运算

  数组和指针的关系

  二级指针

☁️ 本章的内容就是这么多啦!不知道大家了解了没有呢?有不懂的可以私信评论嗷!

看到这里了还不给博主扣个:
⛳️ 点赞☀️收藏 ⭐️ 关注

💛 💙 💜 ❤️ 💚💓 💗 💕 💞 💘 💖

拜托拜托这个真的很重要!

你们的点赞就是博主更新最大的动力!

有问题可以评论或者私信呢秒回哦。

2fa02dff54c24f75a68f419c86cf25b5.png


目录
相关文章
|
3天前
使用指针访问数组元素
【10月更文挑战第30天】使用指针访问数组元素。
15 3
|
15天前
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
41 0
|
3天前
使用指针访问数组元素
【10月更文挑战第31天】使用指针访问数组元素。
7 2
|
11天前
|
算法 索引
单链表题+数组题(快慢指针和左右指针)
单链表题+数组题(快慢指针和左右指针)
19 1
|
17天前
|
存储 编译器 C语言
【c语言】数组
本文介绍了数组的基本概念及一维和二维数组的创建、初始化、使用方法及其在内存中的存储形式。一维数组通过下标访问元素,支持初始化和动态输入输出。二维数组则通过行和列的下标访问元素,同样支持初始化和动态输入输出。此外,还简要介绍了C99标准中的变长数组,允许在运行时根据变量创建数组,但不能初始化。
33 6
|
14天前
|
C语言
【c语言】指针就该这么学(3)
本文介绍了C语言中的函数指针、typedef关键字及函数指针数组的概念与应用。首先讲解了函数指针的创建与使用,接着通过typedef简化复杂类型定义,最后探讨了函数指针数组及其在转移表中的应用,通过实例展示了如何利用这些特性实现更简洁高效的代码。
11 2
|
15天前
|
C语言
如何避免 C 语言中的野指针问题?
在C语言中,野指针是指向未知内存地址的指针,可能引发程序崩溃或数据损坏。避免野指针的方法包括:初始化指针为NULL、使用完毕后将指针置为NULL、检查指针是否为空以及合理管理动态分配的内存。
|
15天前
|
C语言
C语言:哪些情况下会出现野指针
C语言中,野指针是指指向未知地址的指针,通常由以下情况产生:1) 指针被声明但未初始化;2) 指针指向的内存已被释放或重新分配;3) 指针指向局部变量,而该变量已超出作用域。使用野指针可能导致程序崩溃或不可预测的行为。
|
21天前
|
存储
如何使用指针数组来实现动态二维数组
指针数组可以用来实现动态二维数组。首先,定义一个指向指针的指针变量,并使用 `malloc` 为它分配内存,然后为每个子数组分配内存。通过这种方式,可以灵活地创建和管理不同大小的二维数组。
|
21天前
|
存储
如何通过指针数组来实现二维数组?
介绍了二维数组和指针数组的概念及其区别,详细讲解了如何使用指针数组模拟二维数组,包括定义与分配内存、访问和赋值元素、以及正确释放内存的步骤,适用于需要动态处理二维数据的场景。