前言
今天我们继续学习指针,前面对指针有了初步的认识,对指针的讲解较为详细,但还不是最终的版本。
一、指针是什么?(复习)
- 指针是内存中一个最小单元的编号,也就是地址
- 平时口语中说的指针,通常指的是指针变量,是用来存放内存地址的变量
- 指针就是地址,口语中说的指针通常指的是指针变量。指针变量,就是用来存放地址的变量。
二、文章目录
1.指针类型
我们都知道,变量有不同的类型,整形,浮点型等。那指针有没有类型呢?当然是有的,比如存放整形数据的地址的指针变量当时是整形指针,比如存放浮点型数据的地址的指针变量当时是浮点型指针。
**char *pc = NULL;
int *pc= NULL;
short *pc = NULL;
long *pc = NULL;
float *pc = NULL;
double *pc = NULL;
总结:char* 类型的指针是为了存放 char 类型变量的地址。
short* 类型的指针是为了存放 short 类型变量的地址。
int* 类型的指针是为了存放 int 类型变量的地址。
那么我们为什么要区分指针类型呢,把它们都归为一个类型不行吗?答案肯定是不行的,下面的指针运算会介绍指针类型的意义。
2.野指针
概念: 野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)可以理解为错误的指针,即错误的地址。
野指针可能存在的几种情况:1.指针未初始化。
例如:
#include <stdio.h>
int main()
{
int *p;//局部变量指针未初始化,默认为随机值
*p = 10;
return 0; }`
2.指针越界访问。
#include <stdio.h>
int main()
{
int arr[10] = {0};
int *p = arr;
int i = 0;
for(i=0; i<=11; i++)
{
//每次指针+1后指向下一个内容,当指针指向的范围超出数组arr的范围时,p就是野指针,因为超出了数组的内容,你数组一共10个元素,你指向了第11个内容,这就行不通了,这就是野指针。
*(p++) = i;
}
return 0; }
每次指针+1后指向下一个内容,当指针指向的范围超出数组arr的范围时,p就是野指针,因为超出了数组的内容,你数组一共10个元素,你指向了第11个内容,这就行不通了,这就是野指针。
3.指针运算
指针可以进行许多加减运算:
1.指针加减整数。
#include <stdio.h>
int main() {
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
int* p = arr;
for (int i = 0; i < 10; i++)
{
printf("%d\n", * (p + i));
}
return 0;
}
这段代码是打印数组中的内容,它的的运行结果是:
这里我们可以看到,指针+1跳过的就是一个内容。就是从这个地址蹦到了下一个地址。
这里再给大家介绍一下指针类型的意义,指针类型决定了两个意义:
1.决定指针+1跳过多大的内容。来看下面的代码:
#include <stdio.h>
int main() {
int a = 2;
int* p = &a;
printf("%p\n", p);
printf("%p", p+1);
}
这里我们打印出a的地址后,再打印出p+1的内容,我们来看运行结果:
两个数据正好相差四个字节,我们知道一个整型的大小就是四个字节,我们可以猜测字符型指针+1跳过1个字节,我们来验证一下。
这与我们设想的一致。
2.决定了对指针解引用能访问多大的内容 。
这里先给大家说结论:指针的类型决定了,对指针解引用的时候有多大的权限(能操作几个字节)。比如: char* 的指针解引用就只能访问一个字节,而 int* 的指针的解引用就能访问四个字节。这也很好理解,因为它们都是以它们的视角去看待数据的。
指针还能加减指针。指针加减指针代表的是它们中的元素个数。
4.二级指针
指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪里?存放指针变量的地址就是 二级指针。
int a=0;
int *p=&a;
int**pp=&p;
pp就是二级指针,对二级指针解引用找到的是一级指针。其实还有三级指针,只不过我们不经常用。
总结
我们本篇文章学习了一些稍微复制的指针知识,但对指针还有许多内容,指针的篇章并没有结束,后期会继续学习。
本篇文章内容结束,感谢大家观看。如果意见或建议,可以在评论区留言,您的点赞是我更新的动力。我们下篇文章再见。