C语言 --- 指针(5)

目录

一.sizeof和strlen对比

1.sizeof

2.strlen

3.strlen 和sizeof的对比

二.数组和指针笔试题目详解

 回顾:数组名的理解

1.一维数组

2.字符数组 

代码1:

 代码2:

 代码3:

代码4:

代码5:

代码6:

3.二维数组 

总结 


一.sizeof和strlen对比

1.sizeof

sizeof既是一种关键字也是一个操作符。sizeof的作用是计算变量和类型的所占内存空间的大小,单位是字节,返回size_t类型的值。sizeof只关注所占用内存空间的大小,不在乎内存中存放的什么数据。

比如:

#include<stdio.h>
int main()
{
    int a = 10;
    printf("%d\n",sizeof(a));
    printf("%d\n",sizeof a );
    printf("%d\n",sizeof(int));
    
    return 0;
}

sizeof在使用时,可以不用括号,这证明了它不是一个函数,但是在计算类型比如int就不行,必须加括号。

打印结果都是4,size_t类型的值应该用%zd打印,但是这里也可以使用%d,只不过编译器会发出警告,但是影响不大。

sizeof不关注括号中的数据,只关注存放内存的大小。比如:

#include<stdio.h>
int main()
{
	int a = 10;
	int size = sizeof(a++);
	printf("%d\n",size);
	printf("%d\n",a);
	return 0;
}

输出结果是4  10

很明显sizeof括号中的表达式是没有计算的。 

2.strlen

strlen是C语言库函数,在使用时需要包含头文件<string.h>

函数原型:

size_t strlen(const char *str);

这个函数会返回字符串中字符数,但是不包括\0,也就是返回的是\0之前的字符数。如果没有\0就会出现越界查找的情况。

比如:

#include<stdio.h>
int main()
{
    char arr1[] = {'a','b','c'};
    char arr2[] ="abc" ;
    printf("%d\n",strlen(arr1));
    printf("%d\n",strlen(arr2));
    printf("%d\n",sizeof(arr1));
    printf("%d\n",sizeof(arr2));
    
    return 0;
}

打印结果是 随机值 3 3 4.

第一个是随机值的原因就是没有\0,strlen会一直查找直到遇到\0才终止。

第二个因为是常量字符串,包含\0,所以是3。

第三个计算的是数组arr1中所占内存的大小,只有三个字符,也就是3个字节。

第四个arr2中有\0,所以是4个字节。

3.strlen 和sizeof的对比

sizeofstrlen
  1. sizeof是操作符。
  2. sizeof计算的是操作数所占内存空间的大小,单位是字节。
  3. 不关注内存中存放的什么数据。
  1. strlen是库函数,使用的头文件是<string.h>。
  2. strlen是求字符串长度的,统计的是\0之前的字符个数。
  3. strlen关注内存中是否有\0,如果没有\0,就会继续往前找,可能会出现越界的情况。

二.数组和指针笔试题目详解

 回顾:数组名的理解

 数组名是数组首元素的地址。

但是有两个的特殊情况

  1. sizeof(数组名) ,sizeof中单独放数组名,这里的数组名表示的是整个数组,计算的是整个数组的大小,单位是字节。
  2. &数组名,这里的数组名代表的是整个数组,&取出的是整个数组的地址。(整个数组的地址虽然和数组首元素的地址是相同的,但是指针变量的类型是不同的。)

1.一维数组

#include<stdio.h>
int main()
{
	int a[] = { 1,2,3,4 };

	printf("%d\n", sizeof(a));//1
	printf("%d\n", sizeof(a + 0));//2
	printf("%d\n", sizeof(*a));//3
	printf("%d\n", sizeof(a + 1));//4
	printf("%d\n", sizeof(a[1]));//5
	printf("%d\n", sizeof(&a));//6
	printf("%d\n", sizeof(*&a));//7
	printf("%d\n", sizeof(&a + 1));//8
	printf("%d\n", sizeof(&a[0]));//9
	printf("%d\n", sizeof(&a[0] + 1));//10

	return 0;
}
序号解析答案
1 a是数组名,sizeof中单独放数组名,计算的是整个数组的大小,单位字节。16
2sizeof中没有单独放数组名a,所以这里的数组名就是数组首元素的地址,地址就是指针变量。在不同环境下大小不同。4 || 8
3a是数组首元素的地址,解引用后得到数组首元素。所以计算的是整形变量的大小。4
4a表示数组首元素的地址,地址就是指针,类型是int*,指针加1跳过4个字节,也就是一个整形元素,也就是第二个元素的地址。4 || 8
5a[1]表示数组第二个元素4
6&a是整个数组的地址,地址就是指针变量.4 || 8
7&a是整个数组的地址,解引用后就是整个数组。所以sizeof计算的是整个数组的大小(&和*可以相互抵消16
8&a是整个数组的地址,+1跳过了整个数组,虽然这个地址并没有指向这个数组了,但是我们并没有越界访问,所以不影响sizeof计算这个指针变量的大小。4 || 8
9&a[0]得到的是数组首元素的地址.4 || 8
10&a[0]+1,跳过一个元素,得到的是数组第二个元素的地址。4 || 8

2.字符数组 

代码1:

char arr[] = {'a','b','c','d','e','f'};
printf("%d\n", sizeof(arr));//1
printf("%d\n", sizeof(arr+0));//2
printf("%d\n", sizeof(*arr));//3
printf("%d\n", sizeof(arr[1]));//4
printf("%d\n", sizeof(&arr));//5
printf("%d\n", sizeof(&arr+1));//6
printf("%d\n", sizeof(&arr[0]+1));//7
序号解析答案
1arr单独放在sizeof中,计算的是整个数组的大小。单位字节6
2arr没有单独放在数组中,arr表示的是数组首元素的地址4 || 8
3arr表示的是数组首元素的地址,解引用后得到的是数组首元素。1
4arr[1]表示数组首元素。1
5&arr表示整个数组的地址,地址就是指针变量4 || 8
6

&a是整个数组的地址,+1跳过了整个数组,虽然这个地址并没有指向这个数组了,但是我们并没有越界访问,所以不影响sizeof计算这个指针变量的大小。

4 || 8
7&arr[0]表示数组首元素的地址,+1跳过一个元素,表示数组第二个元素的地址4 || 8

 代码2:

char arr[] = {'a','b','c','d','e','f'};
printf("%d\n", strlen(arr));//1
printf("%d\n", strlen(arr+0));//2
printf("%d\n", strlen(*arr));//3
printf("%d\n", strlen(arr[1]));//4
printf("%d\n", strlen(&arr));//5
printf("%d\n", strlen(&arr+1));//6
printf("%d\n", strlen(&arr[0]+1));//7
序号解析答案
1arr是数组首元素的地址,strlen计算的是'\0'之前的字符个数,但是这个数组中并没有'\0',所以会出现越界的情况,故无法出现准确的值的情况。随机值
2这里的arr也是数组首元素的地址,+0并没有跳过任何字节,与序号1同理也是不确定情况。(两个值应该是一样的)随机值
3strlen的参数是const char *类型的。*arr得到的是数组首元素,将这个作为参数传给strlen是不行的,这里程序会崩溃。error
4arr[1]表示数组第二个元素,将其作为参数传给strlen是不行的。error
5&arr表示整个数组的地址,虽然是一个数组指针变量,但是传参会进行强制类型转换为char*类型,值和数组首元素的地址相同,所以和序号1和2相同,值一样的。随机值
6

&a是整个数组的地址,+1跳过了整个数组,也就是跳过了6个字节。但是因为没有'\0',也会出现随机的情况

随机值(比序号125小6)
7&arr[0]表示数组首元素的地址,+1跳过一个元素,表示数组第二个元素的地址,只跳过了1个字节。随机值(比序号125小1)

输出结果:(以下是随机值,不是特定情况。)

 

序号3和4已被注释掉了。

 代码3:

char arr[] = "abcdef";
printf("%d\n", sizeof(arr));//1
printf("%d\n", sizeof(arr+0));//2
printf("%d\n", sizeof(*arr));//3
printf("%d\n", sizeof(arr[1]));//4
printf("%d\n", sizeof(&arr));//5
printf("%d\n", sizeof(&arr+1));//6
printf("%d\n", sizeof(&arr[0]+1));//7
序号解析答案
1sizeof(数组名),计算的是整个数组的大小,字符串末尾是有一个'\0'的,只不过没有显示出来,我们可以通过调试观察,所以这个字符串中有7个字符。7
2这里的arr也是数组首元素的地址,+0并没有跳过任何字节,是一个指针变量。4 || 8
3*arr的到的数组首元素。1
4arr[1]表示数组第二个元素.1
5&arr表示整个数组的地址,是一个指针变量。4 || 8
6

&a是整个数组的地址,+1跳过了整个数组,但是还是一个指针变量

4 || 8
7&arr[0]表示数组首元素的地址,+1跳过一个元素,表示数组第二个元素的地址,是一个指针变量4 || 8

序号1:

我们可以看到数组中分别放着不同的字符,字符前面是它们的ASCII码值,arr[6],也就是数组的第7个元素,存放着'\0'字符。

代码4:

char arr[] = "abcdef";
printf("%d\n", strlen(arr));//1
printf("%d\n", strlen(arr+0));//2
printf("%d\n", strlen(*arr));//3
printf("%d\n", strlen(arr[1]));//4
printf("%d\n", strlen(&arr));//5
printf("%d\n", strlen(&arr+1));//6
printf("%d\n", strlen(&arr[0]+1));//7
序号解析答案
1sizeof计算的是'\0'之前的地址,有6个字符6
2sizeof计算的是'\0'之前的地址,有6个字符6
3*arr的到的数组首元素。不应该作为参数传给strlenerror
4arr[1]表示数组第二个元素.不应该作为参数传给strlenerror
5&arr表示整个数组的地址,sizeof计算的是'\0'之前的地址,有6个字符.6
6

&a是整个数组的地址,+1跳过了整个数组,故不知道什么时候会遇到'\0',所以是个随机值

随机值
7&arr[0]表示数组首元素的地址,+1跳过一个元素,表示数组第二个元素的地址,从第二个开始到'\0',有5个字节。5

输出结果:

序号6就是经过了16个字符才遇到'\0',error情况已经注释掉了。

代码5:

char *p = "abcdef";
printf("%d\n", sizeof(p));//1
printf("%d\n", sizeof(p+1));//2
printf("%d\n", sizeof(*p));//3
printf("%d\n", sizeof(p[0]));//4
printf("%d\n", sizeof(&p));//5
printf("%d\n", sizeof(&p+1));//6
printf("%d\n", sizeof(&p[0]+1);//7
序号解析答案
1sizeof计算的是'\0'之前的地址,有6个字符6
2sizeof计算的是'\0'之前的地址,有6个字符6
3*arr的到的数组首元素。不应该作为参数传给strlenerror
4arr[1]表示数组第二个元素.不应该作为参数传给strlenerror
5&arr表示整个数组的地址,sizeof计算的是'\0'之前的地址,有6个字符.6
6

&a是整个数组的地址,+1跳过了整个数组,故不知道什么时候会遇到'\0',所以是个随机值

随机值
7&arr[0]表示数组首元素的地址,+1跳过一个元素,表示数组第二个元素的地址,从第二个开始到'\0',有5个字节。5

代码6:

char *p = "abcdef";
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
序号解析答案
1sizeof计算的是'\0'之前的地址,有6个字符6
2sizeof计算的是'\0'之前的地址,+1后跳过一个字节,也就是一个元素,有5个字符5
3*p的到的数组首元素。不应该作为参数传给strlenerror
4p[0]表示数组第一个元素.不应该作为参数传给strlenerror
5&p表示字符指针的地址,sizeof计算的是'\0'之前的地址,会产生随机值随机值
6

&p是字符指针的地址,加1 跳过4 || 8个字节,故不知道什么时候会遇到'\0',所以是个随机值

随机值
7&p[0]表示首首字符的地址,+1跳过一个元素,表示第二个字符的地址,从第二个开始到'\0',有5个字节。5

输出结果:

3.二维数组 

int a[3][4] = {0};
printf("%d\n",sizeof(a));//1
printf("%d\n",sizeof(a[0][0]));//2
printf("%d\n",sizeof(a[0]));//3
printf("%d\n",sizeof(a[0]+1));//4
printf("%d\n",sizeof(*(a[0]+1)));//5
printf("%d\n",sizeof(a+1));//6
printf("%d\n",sizeof(*(a+1)));//7
printf("%d\n",sizeof(&a[0]+1));//8
printf("%d\n",sizeof(*(&a[0]+1)));//9
printf("%d\n",sizeof(*a));//10
printf("%d\n",sizeof(a[3]));//11
序号解析

答案

1数组名a单独放在sizeof中,计算的是整个数组的大小。48 = 3 * 4 * sizeof(int)
2a[0][0]表示数组第一行第一个元素。4
3
  1. a[0]表示二维数组首元素地址,也就是第一个一维数组的数组名,a[0]就是单独存放在sizeof的数组名,计算的就是第一个一维数组的大小.
  2. a[0]可以写成*(a + 0),a是二维数组的第一个一维数组的地址,解引用就是第一个一维数组了.
4 * sizeof(int) = 16
4a[0]表示第一个一维数组的数组名,并没有单独放在sizeof中,代表首元素的地址,+1,跳过一个元素,也就是&a[0][1].4 || 8
5解引用后得到的是a[0][1]4
6a表示数组首元素的地址,+1就是第二个元素的地址,也就是第二个一维数组的地址,4 || 8
7解引用后就是第二个一维数组。16
8a[0]是第一行的数组名,&a[0]就是第一行的地址,+1后是第二行的地址,地址就是指针4 || 8
9解引用后就是第二行,计算的第二行的大小16
10*a就是第一行,计算的就是第一行的大小16
11a[3]是第四行的数组名,虽然没有第四行,但是sizeof并没有进行访问,所以并不是越界访问,根据类型仍然可以计算第四行的大小,4个整形变量。16

总结 

数组名的意义:

  1.  sizeof(数组名),这⾥的数组名表⽰整个数组,计算的是整个数组的⼤⼩。
  2.  & 数组名,这⾥的数组名表⽰整个数组,取出的是整个数组的地址。
  3. 除此之外所有的数组名都表⽰⾸元素的地址。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

进程伪装详解

前言 当我们获取到一台主机的权限过后&#xff0c;拿到了自己想要搜集的信息&#xff0c;这时候我们就会留一个后门进行权限维持&#xff0c;权限维持的学问其实很深&#xff0c;今天就主要介绍其中一种比较简单的权限维持的方法 -- 进程伪装。 我们知道在windows里面有很多系…

Jetpack Navigation

1.Navigation的诞生与优势 这个留到Compose去学

探索机器学习的无限可能性:从初学者到专家的旅程

探索机器学习的无限可能性&#xff1a;从初学者到专家的旅程 在当今数字时代&#xff0c;机器学习无疑是最引人注目的技术之一。它已经深入到我们生活的方方面面&#xff0c;从个性化推荐到自动驾驶汽车&#xff0c;再到医疗诊断和金融预测。但是&#xff0c;即使我们已经见证…

时间复杂度之大O表示法

一、概念 O表示法&#xff1a; 设T( n)和 g( n)是正整数集到正实数集上的函数。 称T( n) O( g( n)) &#xff0c;当且仅当存在一个正常数 C 和 n0&#xff0c;使得对任意 的 n ≥ n0&#xff0c;有T( n) ≤ C g( n)。 其中&#xff1a;n 是算法输入的规模&#xff0c;如数组的…

【ghost】制作一个DOS启动盘用于备份/恢复系统

常用的DOS启动盘制作工具有USBoot、Ghost及FlashBoot等&#xff0c;本次DOS启动盘使用Ghost工具制作。 制作前准备 装有win10(或win7)系统的PC机&#xff0c;1台&#xff1b;U盘&#xff0c;1个&#xff1b;&#xff08;建议用户选择兼容性较高的金士顿U盘&#xff1b;此次演…

JAVA实战开源项目:快递管理系统(Vue+SpringBoot)

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、研究内容2.1 数据中心模块2.2 快递类型模块2.3 快递区域模块2.4 快递货架模块2.5 快递档案模块 三、界面展示3.1 登录注册3.2 快递类型3.3 快递区域3.4 快递货架3.5 快递档案3.6 系统基础模块 四、免责说明 一、摘要 1.1 项目介绍 …

Linux系统架构----LNMP平台部署中部署wordpress

Linux系统架构----LNMP平台部署中部署wordpress 一、LNMP的概述 LNMP为Linux平台&#xff0c;Nginx web服务软件&#xff0c;mysql数据库软件&#xff0c;PHP编辑语言LNMP系统架构相对于LAMP的优点是LNMP比较节省内存&#xff0c;主要支持静态请求&#xff0c;但在访问量大的…

力扣坑题:加一

注意数组扩容方法 /*** Note: The returned array must be malloced, assume caller calls free().*/ int* plusOne(int* digits, int digitsSize, int* returnSize) {int indexdigitsSize-1,pos1;while(index>0){digits[index]1;if(digits[index]10){digits[index]0;index-…

Django环境下使用Ajax

Django环境下使用Ajax 目录 Django环境下使用Ajax介绍前情提要示例JS实现Ajax实现 传递JSON格式数据传递文件数据Django自带的序列化组件基于jsonresponse序列化数据基于Django自带的serializers 介绍 AJAX 的主要目标是在不刷新整个页面的情况下&#xff0c;通过后台与服务器…

ROS2参数服务的实现

文章目录 1.参数服务的概念及应用场景1.1 概念1.2 应用场景 2.准备工作3.参数服务的实现3.1 参数数据类型的使用3.2 服务端的实现3.3 客户端的实现3.4 编译及运行 1.参数服务的概念及应用场景 1.1 概念 参数服务是以共享的方式实现不同节点之间数据交互的一种通信模式。保存参…

微信小程序-入门

一.通过 Npm方式下载构建 1.下载和安装Npm&#xff1a;Npm https://docs.npmjs.com/downloading-and-installing-node-js-and-npm 或者 https://nodejs.org/en/download/ 未安装npm 提示 以下以安装node安装包为例 按任意键继续 安装完成后 2. 下载和安装小程序开…

每日学习笔记:C++ STL 的Vector

Vector定义 Vector的大小与容量 Vector的函数 操作注意事项 Vector当作C数组 vector<bool>

Sora盈利新路径:基于技术创新与跨界融合

在数字化时代&#xff0c;技术的飞速进步为企业带来了前所未有的盈利机会。Sora作为一款前沿的AI视频生成工具&#xff0c;其盈利新路径可以基于技术创新与跨界融合两个核心策略来探索。 一、技术创新&#xff1a;持续引领行业前沿 Sora学习资料&#xff1a;使用方式完整文档…

AcWing 898. 数字三角形

解题思路 相关代码 import java.util.Scanner;public class Main {public static void main(String[] args){Scanner scanner new Scanner(System.in);int n scanner.nextInt();int a[][] new int[n1][n1];for(int i1;i<n;i)for(int j1;j<i;j)a[i][j] scanner.nextI…

T1 小美的数组询问(15分) - 美团编程题 题解

考试平台&#xff1a; 牛客网 题目类型&#xff1a; 30道单选题&#xff08;60分&#xff09; 2 道编程题 &#xff08;15分 25分&#xff09; 考试时间&#xff1a; 2024-03-09 &#xff08;两小时&#xff09; 题目描述 小美拿到了一个由正整数组成的数组&#xff0c;但其中…

网络安全:OpenEuler 部署 jumpserver 堡垒机

目录 一、实验 1.环境 2.OpenEuler 部署 jumpserver 堡垒机 3.OpenEuler 使用 jumpserver 堡垒机&#xff08;管理Linux&#xff09; 4.OpenEuler 使用 jumpserver 堡垒机&#xff08;管理Windows&#xff09; 二、问题 1.jumpserver 安装报错 一、实验 1.环境 &#x…

Java零基础-多维数组

哈喽&#xff0c;各位小伙伴们&#xff0c;你们好呀&#xff0c;我是喵手。 今天我要给大家分享一些自己日常学习到的一些知识点&#xff0c;并以文字的形式跟大家一起交流&#xff0c;互相学习&#xff0c;一个人虽可以走的更快&#xff0c;但一群人可以走的更远。 我是一名后…

streamlit学习-如何播放HLS视频(streamlit嵌入html)

streamlit学习-如何播放HLS视频 一.效果二.直播环境搭建(仅供演示)1.生成m3u82.搭建http服务器(支持跨域)3.验证hls(VLC播放 http://localhost:8000/playlist.m3u8) 三.streamlit demo 本文演示了streamlit如何实现hls直播[streamlit中嵌入html] 一.效果 二.直播环境搭建(仅供演…

[Spring] IoC 控制反转和DI依赖注入和Spring中的实现以及常见面试题

目录 1. 什么是Spring 2.什么是IoC容器 3.通过实例来深入了解IoC容器的作用 3.1造一量可以定义车辆轮胎尺寸的车出现的问题 3.2解决方法 3.3IoC优势 4.DI介绍 5.Spring中的IoC和DI的实现 5.1.存对象 5.1.2 类注解 5.1.3 方法注解 5.2取对像 (依赖注入) 5.2.1.属性…