0基础学C语言|超详解|指针补充

本文详细探讨了指针在编程中的基本概念,包括指针定义、理解方式、不同类型指针的作用,以及指针运算(如指针加减整数、解引用和关系运算)、指针与数组、二级指针和指针数组的使用。通过实例演示了指针在内存操作中的关键作用和潜在陷阱。
摘要由CSDN通过智能技术生成

指针的定义

​ 在计算机科学中,指针(Pointer)是编程语言中的一个对象,利用地址,它的值直接指向存在电脑存储器中另一个地方的值。由于通过地址能找到所需的变量单元,可以说,地址指向该变量单元。因此,将地址形象化的称为“指针”。意思是通过它能找到以它为地址的内存单元。

一、怎么理解指针?

指针可以有两种理解方式

1>是一个指针变量,变量中存储着地址

2>是一个地址,直接指向一个确定的内存空间,这个地址,被称为指针

\#include <stdio.h>

int main()

{

int a = 10;//在内存中开辟一块空间

int *p = &a;//这里我们对变量a,取出它的地址,可以使用&操作符。

//将a的地址存放在p变量中,p就是一个之指针变量。

return 0;

通过这段代码我们知道了:

指针就是一个变量,可以用来存放地址

并且

存储在指针里面的值 是以地址形式存在的

二、指针和指针类型

什么是指针类型?

就是: type+ *p这种类型的指针变量 其本身的type 数据类型

简单的来讲:

  • 指针的类型决定了-> 指针向前或者向后走一步有多少(字节)
  • 指针的类型决定了-> 对指针解引用的时候有多大的权限(能操作几个字节)

eg:当进行解引用操作,并且进行修改的时候,指针的类型就决定了解引用操作可以修改几个字节的值

附上一段代码,有兴趣的小伙伴可以自己debug调试一下看内存区的变化情况

`#include <stdio.h>`

int main()

{

int n = 0x11223344;

char *pc = (char *)&n;

int *pi = &n;

*pc = 0; //重点在调试的过程中观察内存的变化。

*pi = 0; //重点在调试的过程中观察内存的变化。

return 0;

}

三、指针运算

1.指针 ±整数(了解)

\

#include <stdio.h>

int main()

{

 int n = 10;

 char *pc = (char*)&n;

 int *pi = &n;

 

 printf("%p\n", &n);

 printf("%p\n", pc);

 printf("%p\n", pc+1);

 printf("%p\n", pi);

 printf("%p\n", pi+1);

 return  0; 

}

可以得到**:**指针的类型决定了指针向前或者向后走一步有多大(距离)

2.指针的解引用

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

指针的类型决定了,对指针解引用的时候有多大的权限(能操作几个字节)。

比如: char* 的指针解引用就只能访问一个字节,而 int* 的指针的解引用就能访问四个字节。

3.指针 - 指针

前提:

​      两个指针指向同一块连续区域,

指针+指针?

​     ( 地址+地址) 是没有意义的

    得到数字的绝对值:是指针和指针之间元素的个数,而不是字节的个数

四、指针的关系运算

1. 指针±整数

3.指针的关系运算

这里有一段代码

for(vp = &values[N_VALUES]; vp > &values[0];)

{

   *--vp = 0; 

}

如果简化一下,可以简化为(两者判断条件不同)

for(vp = &values[N_VALUES-1]; vp >= &values[0];vp--) 

{

   *vp = 0; 

}

​ 看起来两种都没问题,但是实际上第二种解法在vp跳出for循环的时候时,指针已经指向了数组名的前一个地址。实际在绝大部分的编译器上是可以顺利完成任务的,然而我们还是应该避免这样写,因为标准并不保证

它可行

C语言的标准规定

允许指向数组元素的指针与指向数组最后一个元素位置的指针进行比较,但是不允许与指向首元素之前的指针进行比较

4.指针的解引用操作

#include <stdio.h>

int main()

{

 int n = 0x11223344;

 char *pc = (char *)&n;

 int *pi = &n;

 *pc = 0;  //重点在调试的过程中观察内存的变化。

 *pi = 0;  //重点在调试的过程中观察内存的变化。

 return 0; 

}

  指针的类型决定了,对指针解引用的时候有多大的权限(能操作几个字节)。

比如: char* 的指针解引用就只能访问一个字节,而 int* 的指针的解引用就能访问四个字节。

五、指针与数组

数组名是什么?

​ 简述:数组名是首元素地址 想要详细了解,挖掘原理的兄弟请移步——> (1条消息) 0基础学C语言(3)—— 超详解–>《数组》_GuLeng______的博客-CSDN博客

​ 1.sizeof(数组名)——这里的数组名不是首元素的地址,是表示整个舒祖德,这里计算的是整个数组的大小,单位还是字节。

​ 2.&数组名 —— 这里的数组名不是首元素的地址,是 表示整个数组的地址

​ 既然可以把数组名当成地址存放到一个指针中,我们使用指针来访问一个数组就成为可能

#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]);
    for(i=0; i<sz; i++)
   {
        printf("&arr[%d] = %p   <====> p+%d = %p\n", i, &arr[i], i, p+i);
   }
    return 0;
}

来看结果

在这里插入图片描述

   所以 p+i 其实计算的是数组 arr 下标为i的地址。

那我们得到结论:

  可以直接通过指针来访问数组

六、二级指针

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

这就是 二级指针

对于二级指针的运算有:

1. ppa 通过对ppa中的地址进行解引用,这样找到的是 pa , *ppa 其实访问的就是 pa .

int b = 20; 

*ppa = &b;//等价于 pa = &b;

#  指针数组

2.ppa 先通过 *ppa 找到 pa ,然后对 pa 进行解引用操作: *pa ,那找到的是 a .

**ppa = 30;
//等价于*pa = 30;
//等价于a = 30;

七、指针数组

定义:是存放指针的数组

  本质是数组。是存放指针的数组。

未完待续~

  • 34
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 12
    评论
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值