指针的深入理解(七)

指针的深入理解(七)

个人主页:大白的编程日记
个人专栏:C语言学习之路
感谢遇见,我们一起学习进步!


文章目录

  • 指针的深入理解(七)
    • 前言
    • 一.常量字符串指针
      • 1.1常量字符串的理解
      • 1.2常量字符串指针
    • 二.字符指针练习
      • 2.1sizeof
      • 2.2验证
      • 2.3strlen
      • 2.4验证
    • 三. 指针运算笔试题解析
      • 3.1笔试题一
      • 3.2笔试题二
      • 3.3笔试题三
      • 3.4笔试题四
      • 3.5笔试题五
      • 3.6笔试题六
      • 3.7笔试题七
      • 3.8验证
    • 后言

前言

哈喽,大家好!今天继续带大家学习没讲完的指针练习讲解。话不多说,直接进入正题。向着大厂冲锋!杀gigi!在这里插入图片描述


一.常量字符串指针

1.1常量字符串的理解

char* p = "abcdef";//常量字符串,末尾自动补\0

什么是常量字符串?像这种放在双引号内部的字符串就是常量字符串。
那常量字符串有什么不同之处吗?

首先:
常量字符串不可修改。
其次:
常量字符串末尾自动补\0。
这是两个非常重要的知识点,只要掌握好这两个知识点。我们接下来解题就会事半功倍。

  1. 结论一:常量字符串不可修改。
  2. 结论二:常量字符串末尾自动补\0。

1.2常量字符串指针

顾名思义,常量字符串指针就是指向常量字符串的指针。

char* p = "abcdef";//常量字符串,末尾自动补\0

那按理说常量字符指针应该是指向整个字符串的地址。
是不是呢?

其实并非如此。常量字符串指针指向的不是整个字符串的地址,
而是字符串首字符的地址,我们打印时,只需要知道字符串首字符的地址,就能打印整个字符串。

  1. 结论三:常量字符串指针指向的不是整个字符串的地址,
    指向的是首字符的地址。

二.字符指针练习

2.1sizeof

我们知道前面的常量字符串知识点后,来做题巩固一下吧。
看看我们对常量字符串的理解是否深入。

	char* p = "abcdef";//常量字符串,末尾自动补\0printf("%d\n", sizeof(p));//1printf("%d\n", sizeof(p + 1));//2printf("%d\n", sizeof(*p));//3printf("%d\n", sizeof(p[0]));//4printf("%d\n", sizeof(&p));//5printf("%d\n", sizeof(&p + 1));//6printf("%d\n", sizeof(&p[0] + 1));//7
	printf("%d\n", sizeof(p));//1//p是常量字符串指针,指向字符a.//是指针大小就是4/8字节
printf("%d\n", sizeof(p + 1));//2//p指向首字符,p+1跳过一个字符,指向字符b//还是指针,大小就是4/8字节
printf("%d\n", sizeof(*p));//3//p指向首字符,*p就是字符a//大小就是字符大小,就是1字节
	printf("%d\n", sizeof(p[0]));//4//思路一:p[0]--*(p+0)--*p--字符a,大小就是1字节//思路二:把字符串想象成数组,p是数组名,//p[0]就是首元素,大小就是1字节
printf("%d\n", sizeof(&p));//5//指针p也有自己的地址和空间,&p取出的是p指针的地址,//地址就是指针,大小就是4/8字节

printf("%d\n", sizeof(&p + 1));//6//指针p也有自己的地址和空间,&p取出的是p指针的地址,//&p指向的是p变量,&p+1就是跳过p指针后的地址//地址就是指针,是指针大小就是4/8字节
printf("%d\n", sizeof(&p[0] + 1));//7//p[0]就是字符a,&p[0]就是字符a的地址,//&p[0]+1跳过一个字符,指向字符b的地址//还是指针,大小就是4/8

2.2验证

  • 64位环境:
  • 32位环境:

2.3strlen

char* p = "abcdef";//常量字符串,末尾自动补\0
printf("%d\n", strlen(p));//1
printf("%d\n", strlen(p+1));//2
printf("%d\n", strlen(*p));//3
printf("%d\n", strlen(p[0]));//4
printf("%d\n", strlen(&p));//5
printf("%d\n", strlen(&p+1));//6
printf("%d\n", strlen(&p[0]+1));//7
printf("%d\n", strlen(p));//1
//p是常量字符数组首字符的地址,从a开始统计,6个字符
printf("%d\n", strlen(p+1));//2
//p是常量字符数组首字符的地址,p+1跳过一个字符
//指向b,从b开始统计,5个字符
printf("%d\n", strlen(*p));//3
//p是首字符,*p==字符a,字符a的ascll码值是97
//相当于把97作为地址传给strlen
//strlen接收的是野指针,代码有问题
printf("%d\n", strlen(p[0]));//4
//p[0]也是首字符,也相当于把97作为地址传给strlen
//strlen接收的也是野指针,代码也是有问题的

printf("%d\n", strlen(&p));//5
//p是指向首字符的指针,也有自己独立的空间的地址
//&p取出的是p的地址,和字符串关系不大
//什么时候遇到\0也不知道,是随机值

printf("%d\n", strlen(&p+1));//6
//p是指向首字符的指针,也有自己独立的空间的地址
//&p取出的是p的地址,&p+1跳过就是跳过p后的地址
//什么时候遇到\0也不知道,是随机值


那这两个随机值有没有什么关系呢?

如果\0的位置在p的后面的话,他们会相差4\8。
但是如果\0在p的内部的话,那他们就没有关系。
所以这两个随机值的没有啥太大的关系。
7.

printf("%d\n", strlen(&p[0]+1));//7
//p[0]是首字符,&p[0]取出的是首字符的地址
//&p[0]+1跳过一个字符,指向字符b
//从字符b统计,5个字符

2.4验证

  • 64位环境:
  • 32位环境:

三. 指针运算笔试题解析

接下来我们再来看几道指针运算题(有一道是阿里巴巴的!)

3.1笔试题一

#include <stdio.h>
int main()
{int a[5] = { 1, 2, 3, 4, 5 };int *ptr = (int *)(&a + 1);printf( "%d,%d", *(a + 1), *(ptr - 1));return 0;
}
//程序的结果是什么?

我们先看ptr指针。

 int *ptr = (int *)(&a + 1); 


接下来再来看看*(a+1).

 printf( "%d,%d", *(a + 1), *(ptr - 1));


所以答案应该是2和5.
这里我们来验证一下对不对。


  • 验证:


3.2笔试题二

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>//在X86环境下
//假设结构体的⼤⼩是20个字节
//程序输出的结构是啥?struct Test{int Num;char* pcName;short sDate;char cha[2];short sBa[4];}*p = (struct Test*)0x100000;
int main()
{printf("%p\n", p + 0x1);printf("%p\n", (unsigned long)p + 0x1);printf("%p\n", (unsigned int*)p + 0x1);return 0;
}
//程序的结果是什么?

这道题涉及到我们指针一个非常重要的知识点。
那就是指针加减整除,移动多大距离取决于,指针的类型。

我们一个一个来看

*p = (struct Test*)0x100000;
printf("%p\n", p + 0x1);

p是结构体指针,题目又说,结构体的大小是20,所以p+1指针向前移动20个字节,但是p的地址是十六进制表示,1*16=4=20。
所以第一个结果应该是0x100014.

printf("%p\n", (unsigned long)p + 0x1);

p被强制类型转化为unsigned long,所以此时p不是指针了,
而是整型,整型+1就是+1。所以结果是0x100001。

	printf("%p\n", (unsigned int*)p + 0x1);

p被强制类型转化为unsigned int*,此时p是整形指针,+1跳过一个整型,也就是4个字节。所以结果是0x100004。


  • 验证:


3.3笔试题三

int a[3][2] = { (0, 1), (2, 3), (4, 5) };
int* p;
p = a[0];
printf("%d", p[0]);
return 0;


注意看,数组初始化时用的是小括号,所以数组只存放了1 3 5
a[0]是第一行的数组名,就是第一行首元素的地址。就是1的地址
p[0]==*(p+0)==星号p。所以结果应该是1.


  • 验证:


3.4笔试题四

//假设环境是x86环境,程序输出的结果是啥?
#include <stdio.h>
int main()
{int a[5][5];int(*p)[4];//数组指针p = a;//强制类型转换printf( "%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);return 0;
}


这里我写了思路图解,具体讲解看图就可以了。

  • 验证:

在这里插入图片描述


3.5笔试题五

#include <stdio.h>
int main()
{int aa[2][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };int *ptr1 = (int *)(&aa + 1);int *ptr2 = (int *)(*(aa + 1));printf( "%d,%d", *(ptr1 - 1), *(ptr2 - 1));return 0;
}

  • &aa
    &数组名,是整个二维数组的地址,+1跳过整个二维数组。指向二维数组后面的地址。
    转化为整形指针后,-1向后移动一个整型,指向10.(ptr2-2)
  • aa
    aa是数组名,是第一个一维数组的地址,+1跳过一个一维数组,指向第二个 一维数组的首元素。强制转化为整形指针,-1向后移动一个整型,指向5.(ptr1-1)

  • 验证:


3.6笔试题六

#include <stdio.h>
int main()
{char *a[] = {"work","at","alibaba"};char**pa = a;pa++;printf("%s\n", *pa);return 0;
}

看名字就知道这是一道阿里巴巴的题目。
小伙伴们要竖起耳朵好好听啦!

a是一个指针数组,三个指针分别指向三个字符串。
pa是一个二级指针,指向指针数组首元素a.
pa++跳过一个char*类型的元素,就是一个字符指针。
指向数组第二个元素,*pa就拿到的哥字符串的指针,就是地址。
所以输出的是at。


  • 验证:


3.7笔试题七

int main()
{char *c[] = {"ENTER","NEW","POINT","FIRST"};char**cp[] = {c+3,c+2,c+1,c};char***cpp = cp;printf("%s\n", **++cpp);printf("%s\n", *--*++cpp+3);printf("%s\n", *cpp[-2]+3);printf("%s\n", cpp[-1][-1]+1);return 0;
}

这道题有点难,大家一定要跟上小编的思路,图文结合来看。

  1. 输出一:

    这里大家图文结合看就好了。

  2. 输出二:

  3. 输出三:

  4. 输出四:


3.8验证


后言

今天给大家带来的指针练习,还是有点难度的。但是只要我们记好那几个关键的知识点,一步一步解题,就没有那么难了。回去大家好好消化一下。到这里,我们就是一个指针高手啦!今天就分享到这里,咱们下期见!拜拜~

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

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

相关文章

软考128-上午题-【软件工程】-白盒测试

一、白盒测试&#xff08;结构测试&#xff09; 白盒测试也称为结构测试&#xff0c;根据程序的内部结构和逻辑来设计测试用例&#xff0c;对程序的路径和过程进行测试&#xff0c;检查是否满足设计的需要。 白盒测试常用的技术是&#xff1a;逻辑覆盖、循环覆盖和基本路径测…

Qt控件---多元素类

文章目录 QListWidget&#xff08;纵向列表&#xff09;使用 QTableWidget&#xff08;表格&#xff09;使用 QTreeWidget&#xff08;树形控件&#xff09;QTreeWidgetItem使用 QListWidget&#xff08;纵向列表&#xff09; QListWidget里的每个元素都为 QListWidgetItem 类…

[计算机效率] 时间记录工具:ManicTime

3.24 时间记录工具&#xff1a;ManicTime ManicTime是一款数据收集软件&#xff0c;主要用于记录电脑上各种软件使用所花费的时间以及电脑闲置的时间。用户还可以定制记录某一时间段内的系统活动。 数据收集&#xff1a;ManicTime能够静默运行于后台&#xff0c;自动跟踪并收…

[MySQL]数据库原理8——喵喵期末不挂科

希望你开心&#xff0c;希望你健康&#xff0c;希望你幸福&#xff0c;希望你点赞&#xff01; 最后的最后&#xff0c;关注喵&#xff0c;关注喵&#xff0c;关注喵&#xff0c;大大会看到更多有趣的博客哦&#xff01;&#xff01;&#xff01; 喵喵喵&#xff0c;你对我真的…

解读《算者生存:商业分析的方法与实践》:构建企业经营分析框架的必备指南

&#x1f482; 个人网站:【 摸鱼游戏】【神级代码资源网站】【工具大全】&#x1f91f; 一站式轻松构建小程序、Web网站、移动应用&#xff1a;&#x1f449;注册地址&#x1f91f; 基于Web端打造的&#xff1a;&#x1f449;轻量化工具创作平台&#x1f485; 想寻找共同学习交…

【Qt 学习笔记】Qt常用控件 | 按钮类控件QPushButton的使用及说明

博客主页&#xff1a;Duck Bro 博客主页系列专栏&#xff1a;Qt 专栏关注博主&#xff0c;后期持续更新系列文章如果有错误感谢请大家批评指出&#xff0c;及时修改感谢大家点赞&#x1f44d;收藏⭐评论✍ Qt常用控件 | 按钮类控件QPushButton的使用及说明 文章编号&#xff1…

Linux数据库自动备份 - 定时任务发到百度云盘、坚果云、邮箱附件

前言 1. 坚果云的webdav云盘最好&#xff01; &#xff08;免费账号每月1G上传流量&#xff09; 2. 不建议数据库备份文件发送到SMTP邮箱&#xff0c;因为对方服务器非常容易当做垃圾邮件处理&#xff0c;而且发信的SMTP账号会被封禁&#xff08;实测163发到QQ邮箱被封&…

new Ammo.btSoftBodyHelpers() 是 Ammo.js 中的一个构造函数,用于创建软体物体的辅助对象,提供了一些方法来创建软体物体

demo案例 new Ammo.btSoftBodyHelpers() 是 Ammo.js 中的一个构造函数&#xff0c;用于创建软体物体的辅助对象&#xff0c;提供了一些方法来创建软体物体。以下是它的一些重要信息&#xff1a; 入参&#xff1a;通常不需要传入参数。 出参&#xff1a;创建的新的软体辅助对…

SENet模型原理及代码介绍

一.模型简介&#xff1a; SENet的全称叫Squeeze-and-Excitation Networks&#xff08;挤压-激励网络&#xff0c;简称SENet&#xff09;&#xff0c;于2017年提出&#xff0c;并拿下了当年的ImageNet分类比赛的冠军。ResNet是2015年ImageNet的冠军&#xff0c;2016年ResNeXt&am…

护眼台灯怎么选看哪些指标?细数五款性价比最高的护眼灯

在日常生活中&#xff0c;越来越多的人开始重视眼部保护。除了日常用眼要合理、阅读写作姿势要正确外&#xff0c;良好的用眼环境同样重要。很多人在写作、学习时都会开启台灯来补充光源&#xff0c;优化用眼环境。但如果台灯不护眼&#xff0c;反而可能加剧眼部疲劳、近视等问…

【服务器部署篇】Linux下Jenkins安装和配置

作者介绍&#xff1a;本人笔名姑苏老陈&#xff0c;从事JAVA开发工作十多年了&#xff0c;带过刚毕业的实习生&#xff0c;也带过技术团队。最近有个朋友的表弟&#xff0c;马上要大学毕业了&#xff0c;想从事JAVA开发工作&#xff0c;但不知道从何处入手。于是&#xff0c;产…