指针初阶(超详解)

指针初阶

    • 1.指针是什么
    • 2.指针和指针类型
      • 2.1 指针+-整数
      • 2.2 指针的解引用
    • 3.野指针
      • 3.1 野指针成因
      • 3.2如何避免野指针
    • 4.指针运算
      • 4.1 指针+-整数
      • 4.2 指针-指针
      • 4.3 指针的关系运算
    • 5.指针和数组
    • 6.二级指针
    • 7.指针数组

1.指针是什么

指针是什么?

指针理解的2个要点:

  1. 指针是内存中一个最小单元的编号,也就是地址
  2. 平时口语中说的指针,通常指的是指针变量,是用来存放内存地址的变量
    (单元编号 == 地址 == C语言中:指针)

那我们可以这样理解:

内存:
在这里插入图片描述

指针变量
我们可以通过&(取地址操作符)取出变量的内存其实地址,把地址可以存放到一个变量中,这个
变量就是指针变量

int main()
{int a = 10;//是向内存中的栈区空间申请4个字节的空间,这4个字节用来存放10这个数值int* pa = &a;//这里我们对变量a,取出它的地址,可以使用&操作符。//a变量占用4个字节的空间,这里是将a的4个字节的第一个字节的地址存放在p变量中,p就是一个之指针变量。//pa 0x000112233return 0;
}

总结:

指针变量,用来存放地址的变量。(存放在指针中的值都被当成地址处理)。
那这里的问题是:
1.一个小的单元到底是多大?(1个字节)
2.如何编址?
经过仔细的计算和权衡我们发现一个字节给一个对应的地址是比较合适的。
对于32位的机器,假设有32根地址线,那么假设每根地址线在寻址的时候产生高电平(高电压)和低电平(低电压)就是(1或者0);
那么32根地址线产生的地址就会是:
在这里插入图片描述
在这里插入图片描述

  1. 内存被划分为一个个内存单元,每个内存单元的大小是1个字节
  2. 每个字节的内存单元都有一个编号,这个编号就是地址,地址就是C语言中的指针
  3. 地址要储存的话放在指针变量中
  4. 每个内存单元都有唯一的地址来标识
  5. 在32位机器上地址的大小是4个字节,所以指针变量的大小也是4个字节
    同理:在64位机器上地址的大小是8个字节,所以指针变量的大小也是8个字节

2.指针和指针类型

我们看以下代码:

int main()
{printf("%d\n", sizeof(char*));printf("%d\n", sizeof(short*));printf("%d\n", sizeof(int*));printf("%d\n", sizeof(double*));
}

在这里插入图片描述
在这里插入图片描述

2.1 指针±整数

代码演示:

int main()
{int a = 0;int* pa = &a;char* pc = &a;printf("%d\n", pa);printf("%d\n", pa+1);printf("%d\n", pc);printf("%d\n", pc+1);
}

运行结果:
在这里插入图片描述

指针类型是有意义的
指针类型决定了指针+1/-1跳过几个字节
char的指针+1跳过1个字节
short
的指针+1跳过2个字节
int的指针+1跳过4个字节
double
的指针+1跳过8个字节

2.2 指针的解引用

在这里插入图片描述
在这里插入图片描述
指针类型是有意义的
指针类型决定指针解引用时访问几个字节
比如:char* 的指针解引用就只能访问一个字节,而 int* 的指针的解引用就能访问四个字节。

3.野指针

概念: 野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)

3.1 野指针成因

  1. 指针未初始化
    代码演示:
int main()
{int* p;//未初始化*p = 20;return 0;
}
  1. 指针访问越界
    代码演示:
int main()
{int arr[10] = { 0 };int* p = arr;int i = 0;for (i = 0; i <= 10; i++)//指针访问越界{//当指针指向的范围超出数组arr的范围时,p就是野指针*p = -1;p++;}return 0;
}
  1. 指针指向的空间释放
    代码演示:
int* test()
{int a = 10;//0x0040fe44return &a;//地址已返回出函数销毁
}
int main()
{//0x0040fe44int* p = test();//p就是野指针printf("%d", p);return 0;
}

3.2如何避免野指针

  1. 指针初始化
    在这里插入图片描述

  2. 小心指针越界

  3. 指针指向空间释放即使置NULL

  4. 避免返回局部变量的地址

  5. 指针使用之前检查有效性

int main()
{int* p = NULL;*p = 20;//空指针不允许访问//....if (p != NULL){//...确保非控指针在使用}return 0;
}

4.指针运算

4.1 指针±整数

代码演示1:

int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };// 0 1 2 3 4 5 6 7 8 9 //是指针打印数组内容int* p = arr;int i = 0;//p --> arr//p == arr//p+i == arr+i//*(p+i) == *(arr+i) == arr[i]//*(arr+i) == arr[i]//*(i+arr) ==i[arr]//[]仅为操作符for (i = 0; i < 10; i++){printf("%d ", *(p + i));//printf("%d ", *(arr + i));//printf("%d ", arr[i]);//printf("%d ", i[arr]);//p指向的是数组首元素//p+i是数组下标为i的元素地址//p+i起始时跳过i*sizeof(int)个字节}return 0;
}

运行结果:
在这里插入图片描述
代码演示2:

#define N_VALUES 5
int main()
{float values[N_VALUES];float* vp;for (vp = &values[0]; vp < &values[N_VALUES];){*vp++ = 0;}return 0;
}

分析:
在这里插入图片描述

4.2 指针-指针

案例1:

int main()
{int arr[10] = { 0 };////指针-指针的前提:两个指针指向同一块区域,指针类型时相同的//指针-指针差值的绝对值,指针和指针之间的元素个数//printf("%d\n", &arr[9] - &arr[0]);printf("%d\n", &arr[0] - &arr[9]);return 0;
}

运输结果:
在这里插入图片描述
案例2:

//模拟实现strlen
//1.计算器
//2.递归
size_t my_strlen(char* str)
{char* start = str;while (*str){str++;}return str - start;
}
int main()
{char arr[] = "abcdef";size_t len = my_strlen(arr);printf("%zd\n", len);return 0;
}

运输结果:
在这里插入图片描述

4.3 指针的关系运算

代码演示1:

int main()
{float values[N_VALUES];float* vp;for (vp = &values[N_VALUES]; vp > &values[0];){*--vp = 0;}return 0;
}

分析:
在这里插入图片描述
代码2(代码1修改如下):

#define N_VALUES 5
int main()
{float values[N_VALUES];float* vp;for (vp = &values[N_VALUES - 1]; vp >= &values[0]; vp--){*vp = 0;}return 0;
}

分析:
在这里插入图片描述

实际在绝大部分的编译器上是可以顺利完成任务的,然而我们还是应该避免这样写,因为标准并不保证它可行。
标准规定:
允许指向数组元素的指针与指向数组最后一个元素后面的那个内存位置的指针比较,但是不允许与
指向第一个元素之前的那个内存位置的指针进行比较。

5.指针和数组

  • 指针就是指针,指针变量就是一个变量,存放的地址,指针变量的大小是4/8
  • 数组就是数组,可以存放一组数,数组的大小取决于元素类型和个数
  • 数组的数组名是首元素地址
  • 通过一个指针可以访问数组元素

数组名表示数组首元素地址,但有两个例外

  • sizeof(数组名),数组名单独放在sizeof内部,数组名表示整个数组,计算的是数组的大小,单位是字节
  • &数组名,数组名表示整个数组,取出的事数组的地址,数组的地址和数组首元素地址的值是一样的,但是类型和意义不同

代码案例1:

int main()
{int arr[10] = { 0 };printf("%d\n", arr);printf("%d\n", arr+1);//printf("%d\n", &arr[0]);printf("%d\n", &arr[0]+1);//printf("%d\n", &arr);printf("%d\n", &arr+1);//printf("%d\n", sizeof(arr));
}

在这里插入图片描述
代码案例二:

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
int main()
{int arr[10] = { 0 };int* p = arr;int i = 0;for (i = 0; i < 10; i++){printf("%p===%p\n", p+i, arr+i);}return 0;
}

运行结果:
在这里插入图片描述

6.二级指针

指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪里?
这就是 二级指针
在这里插入图片描述
二级指针变量是存放一级指针变量地址的

代码案例1:

int main()
{int a = 10;int* pa =  &a;//pa是指针变量,一级指针变量int* * ppa = &pa;//ppa指针变量,二级指针变量**ppa = 20;//*ppa得出pa-->**ppa==*pa==aprintf("%d", a);return 0;
}

运行结果:
在这里插入图片描述

  • *ppa 通过对ppa中的地址进行解引用,这样找到的是 pa , *ppa 其实访问的就是 pa
  • **ppa 先通过 *ppa 找到 pa ,然后对 pa 进行解引用操作: *pa ,那找到的是 a .

代码案例2:

7.指针数组

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

是数组。是存放指针的数组。
数组我们已经知道整形数组,字符数组。
int arr1[5];
char arr2[5]
指针数组:
int* arr3[5];
在这里插入图片描述

代码案例:

int main()
{int arr1[] = { 1,2,3,4,5 };int arr2[] = { 1,2,3,4,5 };int arr3[] = { 1,2,3,4,5 };int* arr[] = { arr1,arr2,arr3 };int i = 0;for (i = 0; i < 3; i++){int j = 0;for (j = 0; j < 5; j++){printf("%d", arr[i][j]);}printf("\n");}return 0;
}

运行结果:
在这里插入图片描述

不知不觉,指针初阶以告一段落。通读全文的你肯定收获满满,让我们继续为C语言学习共同奋进。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/18089.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

maven下载按照及初次使用相关配置

maven下载按照及初次使用相关配置 一、下载 与安装 依赖Java&#xff0c;需要配置JAVA_HOME设置MAVEN自身的运行环境&#xff0c;需要配置MAVEN_HOME测试环境配置结果 MVN测试成功&#xff01;&#xff01;&#xff01; 二、本地仓库配置 Maven启动后&#xff0c;会自动保…

基于Amoeba读写分离(三十六)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 前言 今天要学的是基于Amoeba读写分离。Amoeba是一个开源的关系型数据库管理系统&#xf…

Spark性能调优之数据序列化

前言 在使用Spark进行数据开发的时候,避不开的一个问题就是性能调优。网上一搜一大堆所谓的调优策略很多作者自己都不知所云,导致读者看了后只会更加困惑。我们在研究一个技术的时候第一手资料永远都请参考官网,官网对性能优化不一定是最全甚至最优,但是可以解决大部分问题…

向 Maven 中央仓库上传一个修改过的基于jeecg的autoPOI的 jar包记录

1、注册https://issues.sonatype.org/账号 下面就代表注册好了&#xff0c;同时提交的工单也通过了 2、这里主要是goupId 需要进行认证&#xff0c;需要到域名注册商近一个txt的解析&#xff0c;以便确保这个是你的 通过下面来验证你的域名信息&#xff0c;这里主要是上面的工…

面试题:说一说深拷贝和浅拷贝?

JavaScript中存在两大数据类型&#xff1a; 基本类型 和 引用类型 基本类型数据保存在在栈内存中 引用类型数据保存在堆内存中&#xff0c;引用数据类型的变量是一个指向堆内存中实际对象的引用&#xff0c;存在栈中 深拷贝和浅拷贝都只针对于引用类型。 一、 浅拷贝&#xff1…

Cpp7 — 继承和多态

继承 -------- 面向对象的三大特性之一 面向对象的三大特性&#xff1a;封装、继承、多态 封装&#xff1a;把数据和方法都封装在一起&#xff0c;想给你访问的变成共有&#xff0c;不想给访问的&#xff0c;写成私有。 继承&#xff1a;继承是类设计层次的复用 多态&#…

【AGI】Copilot AI编程辅助工具安装教程

1. 基础激活教程 GitHub和OpenAI联合为程序员们送上了编程神器——GitHub Copilot。 但是&#xff0c;Copilot目前不提供公开使用&#xff0c;需要注册账号通过审核&#xff0c;我也提交了申请&#xff1a;这里第一期记录下&#xff0c;开启教程&#xff0c;欢迎大佬们来讨论…

通向架构师的道路之apache性能调优

一、总结前一天的学习 在前两天的学习中我们知道、了解并掌握了Web Server结合App Server实现单向Https的这样的一个架构。这个架构是一个非常基础的J2ee工程上线布署时的一种架构。在前两天的教程中&#xff0c;还讲述了Http服务 器、App Server的最基本安全配置&#xff08;…

java 数组的使用

数组 基本介绍 数组可以存放多个同一类型的数据&#xff0c;数组也是一种数据类型&#xff0c;是引用类型。 即&#xff1a;数组就是一组数据。 数组的使用 1、数组的定义 方法一 -> 单独声明 数据类型[] 数组名 new 数据类型[大小] 说明&#xff1a;int[] a new int…

C/C++算法——散列表

1、散列表介绍 散列表的英文叫Hash Table&#xff0c;我们平时也叫它哈希表或者Hash 表。散列表用的是数组支持按照下标随机访问数据的特性&#xff0c;所以散列表其实就是数组的一种扩展&#xff0c;由数组演化而来。可以说&#xff0c;如果没有数组&#xff0c;就没有散列表。…

iOS——锁与死锁问题

iOS中的锁 什么是锁锁的分类互斥锁1. synchronized2. NSLock3. pthread 递归锁1. NSRecursiveLock2. pthread 信号量Semaphore1. dispatch_semaphore_t2. pthread 条件锁1. NSCodition2. NSCoditionLock3. POSIX Conditions 分布式锁NSDistributedLock 读写锁1. dispatch_barri…

超全整理,Jmeter性能测试-常用Jmeter第三方插件详解(超细)

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 Jmeter作为一个开…

React(4)

1.属性&#xff08;props&#xff09;初始 状态state都是组件内部写的&#xff0c;也就是A组件内的state就只能A组件里面用&#xff0c;其他组件复用不了。因此属性props就可以。 比如一个导航栏&#xff0c;首页有&#xff0c;购物车有&#xff0c;我的有&#xff0c;他们三个…

《吐血整理》进阶系列教程-拿捏Fiddler抓包教程(20)-Fiddler精选插件扩展安装让你的Fiddler开挂到你怀疑人生

1.简介 Fiddler本身的功能其实也已经很强大了&#xff0c;但是Fiddler官方还有很多其他扩展插件功能&#xff0c;可以更好地辅助Fiddler去帮助用户去开发、测试和管理项目上的任务。Fiddler已有的功能已经够我们日常工作中使用了&#xff0c;为了更好的扩展Fiddler&#xff0c…

P4780 Phi的反函数

题目 思路 φ(x)n 当指数均为1时n最小 证明&#xff1a;容斥原理 代码 #include<bits/stdc.h> using namespace std; #define int long long const int maxn1e9; int ansINT_MAX,n; bool f; map<int,bool> mp; bool is_prime(int n){if(n<1) return false;fo…

Spring事务创建与使用

目录 前言Spring中事务的实现声明式事务Transactional 作⽤范围Transactional 参数说明对于事务不回滚的解决方案 前言 在数据库中我们提到了 事务, 事务的定义为, 将一系列操作封装成一个整体去调用 , 要么一起成功, 要么一起失败 Spring中事务的实现 在Spring中事务的操作…

发npm包

重点文件 .github -> workflow -> .yml文件 发自己的包 新建dev分支&#xff0c;合并到master后自动执行 fork别人的包 fork -> base dev新建本地rebase-dev分支 -> 提交push后合并至dev -> dev合并至master后自动执行 值得注意的是&#xff0c;fork别人的…

flask 点赞系统

dianzan.html页面 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>点赞系统</title> </head> <body><h2>这是一个点赞系统</h2><table border"1"><…

【vue】vue-image-lazy图片懒加载使用与介绍【超详细+npm包源代码】

简介 当前插件是基于vue3&#xff0c;写的一个图片懒加载&#xff0c;文章最下方是npm包的源码&#xff0c;你可以自己拿去研究和修改&#xff0c;如有更好的想法可以留言&#xff0c;如果对你有帮助&#xff0c;可以点赞收藏和关注&#xff0c;谢谢。 后续会添加图片放大和切…

蓝桥云课ROS机器人旧版实验报告-07外设

项目名称 实验七 ROS[Kinetic/Melodic/Noetic]外设 成绩 内容&#xff1a;使用游戏手柄、使用RGBD传感器&#xff0c;ROS[Kinetic/Melodic/Noetic]摄像头驱动、ROS[Kinetic/Melodic/Noetic]与OpenCV库、标定摄像头、视觉里程计&#xff0c;点云库、可视化点云、滤波和缩…