关于数组和指针的笔试题解析(详解)

news2025/1/19 15:08:18

在这里插入图片描述

文章目录

  • 说明
  • 🚩数组笔试题
    • 💻一维数组
      • 📄练习:
      • 💡解析
    • 💻字符数组
      • 📄练习1:
      • 💡解析
      • 📄练习2:
      • 💡解析
      • 📄练习3:
      • 💡解析
      • 📄练习4:
      • 💡解析
      • 📄练习5:
      • 💡解析
      • 📄练习6:
      • 💡解析
    • 💻二维数组
      • 📄练习:
      • 💡解析
    • 🗞️小结
  • 🚩指针运算笔试题
      • 📄练习1:
      • 💡解析
      • 📄练习2:
      • 💡解析
      • 📄练习3:
      • 💡解析
      • 📄练习4:
      • 💡解析
      • 📄练习5:
      • 💡解析
      • 📄练习6:
      • 💡解析
      • 📄练习7:
      • 💡解析
      • 📄练习8:
      • 💡解析
  • 🚩总结

说明

X64环境下,8个字节
X86环境下,4个字节

小编在运行代码,数据是在VS 2019 X86环境下打印的。

🚩数组笔试题

💻一维数组

📄练习:

code:

#include<stdio.h>

int main()
{
	int a[] = { 1,2,3,4 };
	printf("%zd\n", sizeof(a));
	printf("%zd\n", sizeof(a + 0));
	printf("%zd\n", sizeof(*a));
	printf("%zd\n", sizeof(a + 1));
	printf("%zd\n", sizeof(a[1]));
	printf("%zd\n", sizeof(&a));
	printf("%zd\n", sizeof(*&a));
	printf("%zd\n", sizeof(&a + 1));
	printf("%zd\n", sizeof(&a[0]));
	printf("%zd\n", sizeof(&a[0] + 1));
	return  0;
}

运行结果:

16
4
4
4
4
4
16
4
4
4

在这里插入图片描述

💡解析

printf("%zd\n", sizeof(a));  //16

➿➿数组名的理解:

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

但是有2个例外:

  1. sizeof(数组名),数组名表示整个数组,计算的是整个数组的大小,单位是字节
  2. &数组名,数组名表示整个数组,取出的是数组的地址

⭕所以在此代码中,有四个整型,一个整型占4个字节,故总共占16个字节。


printf("%zd\n", sizeof(a + 0)); //4

数组名a没有单独放在sizeof()中,也没有进行单独取地址&,因此,(a+0)不是数组名,这里的a是数组首元素地址,加上0,相当于没有加
a+0<======>&a[ 0 ]

⭕故,是地址大小,4或者8个字节(X64环境下和X86环境下不一样)


printf("%zd\n", sizeof(*a));   //4

这里没有将单独的一个a放进sizeof()中,也没有取地址&,那么a就是除那两种情况之外,即a就是数组首元素地址,a==>&a[ 0 ]

⭕故,*a 其实就是第一个元素,也就是a[ 0 ] 的大小:4个字节


printf("%zd\n", sizeof(a + 1));  //4

和上面(a+0)一样,a是首元素地址

(a+0)–>&a[ 0 ]
(a+1)–>&a[ 1 ]
⭕故,(a+1)就是第2个元素的地址,大小是4或者8个字节


printf("%zd\n", sizeof(a[1]));    //4

a[1]就是这个数组的第二个元素

⭕故,大小是4个字节


printf("%zd\n", sizeof(&a));

&a➡️取出的是数组的地址,但是数组的地址也是一个地址呀😏,数组的地址可没有高人一等。是地址大小就是4或者8个字节

⭕故,大小是4或者8个字节


printf("%zd\n", sizeof(*&a));     //16

两种解读方式:
1️⃣抵消:
·这里取地址,然后再解引用,抵消掉了,相当于就是a

2️⃣数组指针类型:
· &a类型是一个数组指针,&a<==>int(*p)[ 4 ]

· 我们知道,指针在进行加一或者解引用的时候,跳过多少个字节是取决于指针类型:
*p访问一个数组的大小
p+1是跳过一个数组的大小
·
那么现在p指向一个大小为4,类型为整型的一个数组

取出整个数组的地址,再进行解引用,访问的就是整个数组

因此:

printf("%zd\n", sizeof(*&a));
<==>
printf("%zd\n", sizeof(a));

⭕故,大小是16个字节


printf("%zd\n", sizeof(&a + 1)); //4

在这里插入图片描述

&a+1是跳过整个数组后的地址,是地址大小就是4或者8个字节。

⭕故,大小是4或者8个字节


printf("%zd\n", sizeof(&a[0]));    //4

没什么好说的,就是首元素地址🤨,是地址大小就是4或者8个字节。

⭕故,大小是4或者8个字节


printf("%zd\n", sizeof(&a[0] + 1)); //4

&a[0] + 1表示第二个元素的地址(&a[ 1 ])

⭕故,大小是4或者8个字节


💻字符数组

📄练习1:

code:

#include<stdio.h>
int main()
{
	char arr[] = { 'a','b','c','d','e','f' };
	printf("%d\n", sizeof(arr));
	printf("%d\n", sizeof(arr + 0));
	printf("%d\n", sizeof(*arr));
	printf("%d\n", sizeof(arr[1]));
	printf("%d\n", sizeof(&arr));
	printf("%d\n", sizeof(&arr + 1));
	printf("%d\n", sizeof(&arr[0] + 1));
	return 0;
}

运行结果:

6
4
1
1
4
4
4

在这里插入图片描述

💡解析

printf("%d\n", sizeof(arr));  //6

arr表示整个数组,计算的是整个数组的大小

此数组有6个字符,一个字符1个字节

⭕故,一共有6个字节


printf("%d\n", sizeof(arr + 0));   //4

arr是数组首元素的地址,arr+0还是首元素的地址

⭕故,是地址,大小就是4或者8个字节


printf("%d\n", sizeof(*arr)); //1

arr就是首元素地址,*arr解引用,就是首元素,就站一个字符,即1个字节

⭕故,大小就是1个字节


printf("%d\n", sizeof(arr[1]));  //1

arr[ 1 ]表示数组第2个元素,即占一个字节

⭕故,大小就是1个字节


printf("%d\n", sizeof(&arr)); //4

&arr是数组的地址,数组的地址也是地址

⭕故,是地址,大小就是4或者8个字节


printf("%d\n", sizeof(&arr + 1));

&arr+1跳过整个数组,指向的是f后面,

在这里插入图片描述

⭕故,是地址,大小就是4或者8个字节


printf("%d\n", sizeof(&arr[0] + 1)); //4

&arr[0]是首元素地址,+1后变成第二个元素地址

⭕故,是地址,大小就是4或者8个字节


📄练习2:

code:

#include<stdio.h>
#include<string.h>
int main()
{
	char arr[] = { 'a','b','c','d','e','f' };
	printf("%zd\n", strlen(arr));
	printf("%zd\n", strlen(arr + 0));
	printf("%zd\n", strlen(*arr));
	printf("%zd\n", strlen(arr[1]));
	printf("%zd\n", strlen(&arr));
	printf("%zd\n", strlen(&arr + 1));
	printf("%zd\n", strlen(&arr[0] + 1));
	return 0;
}

💡解析

printf("%zd\n", strlen(arr));  //随机值

数组中没有明确给出\0

⭕故,计算出的结果是随机值


printf("%zd\n", strlen(arr + 0)); //随机值

arr+0:首元素地址+0,和没加一样,依然表示arr,数组中也是没有明确给出\0

⭕故,计算出的结果是随机值


printf("%zd\n", strlen(*arr)); //非法访问-err

strlen()函数参数是指针类型

size_t strlen ( const char * str );

而*arr得到是首元素‘a’

这样就意味着将’a’(97)传递给strlen,将97当作地址传递给strlen

⭕故,形成非法访问


printf("%zd\n", strlen(arr[1])); //非法访问-err

arr[1]表示数组第二个元素,‘b’(98)

将98当作地址传递给strlen,依然是非法访问,和上面一样

⭕故,形成非法访问


printf("%zd\n", strlen(&arr)); //随机值

&arr是一个字符数组指针类型–>char (*p)[ 6 ]

对于strlen依然是找到首元素地址,往后读取,但是没有\0

⭕故,计算出的结果是随机值


printf("%zd\n", strlen(&arr + 1)); //随机值

加一后,跳过整个数组

在这里插入图片描述

跳过一个数组后再去往后找,不知道找什么

和上面的随机值是不一样的

差6个字节

⭕故,计算出的结果是随机值


printf("%zd\n", strlen(&arr[0] + 1)); //随机值

&arr[0] + 1表示数组第二个元素,即’b’,从‘b’开始往后数,没有明确给出\0

⭕故,计算出的结果是随机值


📄练习3:

code:

#include<stdio.h>
int main()
{
	char arr[] = "abcdef";
	//[ a b c d e f \0 ]
	printf("%zd\n", sizeof(arr));
	printf("%zd\n", sizeof(arr + 0));
	printf("%zd\n", sizeof(*arr));
	printf("%zd\n", sizeof(arr[1]));
	printf("%zd\n", sizeof(&arr));
	printf("%zd\n", sizeof(&arr + 1)); 
	printf("%zd\n", sizeof(&arr[0] + 1));
	return 0;
}

运行结果:

7
4
1
1
4
4
4

在这里插入图片描述

💡解析

printf("%zd\n", sizeof(arr)); //7

不管有没有\0,只管占了多少字节

有7个元素

⭕故,占7个字节


printf("%zd\n", sizeof(arr + 0)); //4

arr表示首元素地址,即arr+0也表示数组首元素地址,是地址,大小就是4或者8个字节

⭕故,占4或者8个字节


printf("%zd\n", sizeof(*arr)); //1

arr表示首元素地址,*arr就是首元素,大小就是1个字节

⭕故,占1个字节


printf("%zd\n", sizeof(arr[1])); //1

arr[1]表示数组第二个元素,大小1个字节

⭕故,占1个字节


printf("%zd\n", sizeof(&arr)); //4

&arr是数组的地址,是地址,大小就是4或者8个字节

⭕故,占4或者8个字节


printf("%zd\n", sizeof(&arr + 1)); 

&arr是数组地址

&arr+1是跳过整个数组的那个地址

是地址,大小就是4或者8个字节

在这里插入图片描述

⭕故,占4或者8个字节


printf("%zd\n", sizeof(&arr[0] + 1)); //4

&arr[0] + 1表示第二个元素地址

是地址,大小就是4或者8个字节

⭕故,占4或者8个字节


📄练习4:

code:

#include<stdio.h>
#include <string.h>

int main()
{
	char arr[] = "abcdef";
	printf("%zd\n", strlen(arr));
	printf("%zd\n", strlen(arr + 0));
	//printf("%lld\n", strlen(*arr)); 
    //printf("%lld\n", strlen(arr[1])); 
	printf("%zd\n", strlen(&arr)); 
	printf("%zd\n", strlen(&arr + 1));
	printf("%zd\n", strlen(&arr[0] + 1));

	return 0;
}

运行结果:

6
6
6
12
5

在这里插入图片描述

💡解析

printf("%zd\n", strlen(arr)); //6

arr是数组首元素地址,顺着往后读取,读到\0结束

strlen统计的是\0之前的字符串个数

⭕故,计算结果是6


printf("%zd\n", strlen(arr + 0)); //6

arr是数组名,首元素地址

arr+0 还是首元素地址,顺着往后读取,读到\0结束

⭕故,计算结果是6


//printf("%lld\n", strlen(*arr));   //err - 非法访问

*arr表示’a’(97)

将97当作地址传递给strlen

非法访问

⭕故,非法访问


//printf("%lld\n", strlen(arr[1])); //err - 非法访问

arr[1]表示数组第二个元素,即‘b’(98)

将987当作地址传递给strlen

非法访问

⭕故,非法访问


printf("%zd\n", strlen(&arr)); //6

&arr取的是整个数组的地址

但这个地址依然指的是数组的起始位置

传递给strlen后,依然从起始位置往后读取,直到\0停止

⭕故,计算结果是6


printf("%zd\n", strlen(&arr + 1)); //随机值

&arr取的是整个数组的地址

&arr+1跳过整个数组后的地址

从跳过后的地址开始找\0,就是随机值

⭕故,随机值


printf("%zd\n", strlen(&arr[0] + 1)); //5

&arr[0] 表示第一个元素地址

&arr[0] + 1第二个元素地址,即b的地址

从b的地址开始找\0,找到\0后停止

⭕故,计算结果是5


📄练习5:

code:

#include<stdio.h>
int main()
{
	char* p = "abcdef";
	printf("%zd\n", sizeof(p));
	printf("%zd\n", sizeof(p + 1));
	printf("%zd\n", sizeof(*p));
	printf("%zd\n", sizeof(p[0]));
	printf("%zd\n", sizeof(&p));
	printf("%zd\n", sizeof(&p + 1));
	printf("%zd\n", sizeof(&p[0] + 1));

	return 0;
}

运行结果:

4
4
1
1
4
4
4

在这里插入图片描述

💡解析

思路:这里是常量字符串,将常量字符串首字符放在p中。

图示:

在这里插入图片描述


printf("%zd\n", sizeof(p));

p是一个指针变量

sizeof(p)求的就是指针变量的大小

⭕故,大小是4或者8个字节


printf("%zd\n", sizeof(p + 1)); //4

p放的是‘a’的地址

p+1则放的是’b’的地址

依然是地址

⭕故,大小是4或者8个字节


printf("%zd\n", sizeof(*p)); //1

p指向’a’

*p解引用,是首字符

⭕故,大小是1个字节


printf("%zd\n", sizeof(p[0])); //1

两种解读方式:

1️⃣类似数组:
这里,可以把字符串“abcdef”也当作有个数组
p是数组名
那么,p[0]访问的就是‘a’

2️⃣计算:
p[0]===>*(p+0)

⭕故,大小是1个字节


printf("%zd\n", sizeof(&p));  //4

&p是取出p所占的地址

也是个地址

是地址,大小就是4或者8个字节

⭕故,大小是4或者8个字节


 printf("%zd\n", sizeof(&p + 1));

&p + 1也是地址

跳过p变量的地址,指向的是p后面的地址

在这里插入图片描述

是地址,大小就是4或者8个字节

⭕故,大小是4或者8个字节


 printf("%zd\n", sizeof(&p[0] + 1));

&p[0]是’a’的地址

&p[0] + 1是“b’的地址

是地址,大小就是4或者8个字节

⭕故,大小是4或者8个字节


📄练习6:

code:

#include<stdio.h>
#include<string.h>

int main()
{
	char* p = "abcdef";

	printf("%zd\n", strlen(p));
	printf("%zd\n", strlen(p + 1));
	//printf("%zd\n", strlen(*p));
	//printf("%zd\n", strlen(p[0]));
	printf("%zd\n", strlen(&p));
	printf("%zd\n", strlen(&p + 1));
	printf("%zd\n", strlen(&p[0] + 1));

	return 0;
}

运行结果:

6
5
3
11
5

在这里插入图片描述

💡解析

printf("%zdd\n", strlen(p)); //6

字符串中有\0

p中存放的是a的地址

从a的地址开始向后访问

⭕故,长度是6


printf("%zd\n", strlen(p + 1)); //5

p指向a

p+1指向b

从b开始往后数,直到遇到\0为止

⭕故,长度是5


//printf("%zd\n", strlen(*p));//err

p指向a

*p就是a

将p的值传递给strlen

⭕故,非法访问


//printf("%zd\n", strlen(p[0]));//err

p[0]=>*(p+0)=>*p

⭕故,非法访问


printf("%zd\n", strlen(&p)); //随机值

&p是p的地址

从p所占空间的起始位置开始查找的

完全不可知的

⭕故,随机值


printf("%zd\n", strlen(&p + 1));//随机值

&p+1指向的是p的地址后面

从&p后面的地址开始读取

也是不可知的

⭕故,随机值


printf("%zd\n", strlen(&p[0] + 1)); //5

&p[0] + 1是‘b’的地址

从’b’后面开始数

⭕故,长度是5


💻二维数组

📄练习:

code:

#include<stdio.h>

int main()
{
	int a[3][4] = { 0 };
	printf("%zd\n", sizeof(a));
	printf("%zd\n", sizeof(a[0][0]));
	printf("%zd\n", sizeof(a[0]));
	printf("%zd\n", sizeof(a[0] + 1));
	printf("%zd\n", sizeof(*(a[0] + 1)));
	printf("%zd\n", sizeof(a + 1));
	printf("%zd\n", sizeof(*(a + 1)));
	printf("%zd\n", sizeof(&a[0] + 1));
	printf("%zd\n", sizeof(*(&a[0] + 1)));
	printf("%zd\n", sizeof(*a));
	printf("%zd\n", sizeof(a[3]));
	return 0;
}

运行结果:

48
4
16
4
4
4
16
4
16
16
16

在这里插入图片描述

💡解析

printf("%zd\n", sizeof(a));  //48

计算的是整个二维数组的大小,单位是字节

数组里共有3*4=12个元素

每个元素都是int类型

则344

⭕故,大小是48个字节


printf("%zd\n", sizeof(a[0][0]));  //4

a[0][0]表示第一行第一个元素

⭕故,大小是4个字节


printf("%zd\n", sizeof(a[0]));  //16

a[0]表示第一行数组名

在这里插入图片描述

将第一行数组名单独放在sizeof内部

计算的是第一行大小

⭕故,大小是16个字节


printf("%zd\n", sizeof(a[0] + 1));  //4

a[0]是第一行数组名,但是没有把a[0]单独放在sizeof中

这里的a[0]表示第一行第一个元素的地址

a[0] + 1表示第一行第一个元素地址+1,即第一行第二个元素地址==>a[0][1]的地址

是地址,大小是4或者8个字节

在这里插入图片描述
⭕故,大小是4或者8个字节


printf("%zd\n", sizeof(*(a[0] + 1))); //4

a[0] + 1是第一行第二个元素的地址

*(a[0] + 1)解引用,访问的就是第一行第二个元素

⭕故,大小是4个字节


printf("%zd\n", sizeof(a + 1)); //4

这里的a是二维数组的数组名

没有将a单独放在sizeof中

那么,a就是数组首元素地址,也就是第一行的地址😏

a+1跳过一行的地址,就是第二行地址

既然是地址,那么…

⭕故,大小是4或者8个字节


printf("%zd\n", sizeof(*(a + 1))); //16

a+1是第二行地址

*(a + 1)解引用,表示第二行

*(a + 1)==>a[1]

printf("%zd\n", sizeof(*(a + 1)));
<==>
printf("%zd\n", sizeof(a[1]);

两种表示方法,你get到了嘛??😏

⭕故,大小是16个字节


printf("%zd\n", sizeof(&a[0] + 1)); //4

a[0]是第一行数组名

&a[0]取出第一行的地址

&a[0] + 1第一行的地址+1,表示第二行地址

是地址…

⭕故,大小是4或者8个字节


printf("%zd\n", sizeof(*(&a[0] + 1))); //16

&a[0] + 1是第二行的地址

*(&a[0] + 1))对第二行地址解引用,访问第二行数组

⭕故,大小是16个字节


printf("%zd\n", sizeof(*a)); //16

a表示首元素地址,也就是第一行地址

*a对一行地址解引用,就是第一行

⭕故,大小是16个字节


printf("%zd\n", sizeof(a[3])); //16

a[3]是第四行,但是刚开始我们开的数组没有第四行

我们知道,sizeof()是根据类型计算的

实际上不会访问a[3]

a[3]和前面a[0]类型是一样的

⭕故,大小是16个字节


🗞️小结

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

关于sizeof和strlen的介绍,可以看小编的文章《sizeof和strlen的对比》,里面有详细解释!!!

🚩指针运算笔试题

📄练习1:

code:

#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;
}

运行结果:

2,5

💡解析

1️⃣对*(a + 1)的分析:

  • a表示数组首元素地址

  • a+1首元素地址+1,表示第二个元素地址

  • *(a + 1)解引用,表示第二个元素,即2

2️⃣对*(ptr - 1)的分析:

  • 首先分析int* ptr = (int*)(&a + 1);

  • &a表示整个数组地址

  • &a+1表示跳过整个数组地址

  • &a的类型是int(*)[5] ,+1后的类型还是这个类型

  • 现在要将(&a + 1)赋值给ptr

  • ptr是int* 类型,所以需要强制类型转换

在这里插入图片描述

  • 分析*(ptr - 1)

  • ptr-1表示前移动,即5的地址:

在这里插入图片描述

  • *(ptr-1)解引用,表示5

📄练习2:

在X86环境下

假设结构体的⼤⼩是20个字节

code:

#include <stdio.h>

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;
}

运行结果:

00100014
00100001
00100004

💡解析

printf("%p\n", p + 0x1);  //00100014

0x1表示1(在16进制中)

p+0x1即p+1

那么,指针+1到底加几呢??😥

这个取决于指针类型,现在是一个结构体指针

那么加1就是跳过一个结构体大小

结构体大小是20个字节,加的是0x100014

⭕所以,大小变成00100014(16进制中)


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

p的类型被强制转换成unsigned long,无符号整型

此时就不是指针类型啦

那么就是:整型+1

不再是整型指针+1,小小的细节,要清楚

0x100000+1=0x100001

⭕所以,输出结果:00100001


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

p被强制转换成整型指针(unsigned int)

整型指针+1就是+4,即0x100004

⭕所以,输出结果:00100004


📄练习3:

cod:

#include <stdio.h>
int main()
{
	int a[4] = { 1, 2, 3, 4 };
	int* ptr1 = (int*)(&a + 1);
	int* ptr2 = (int*)((int)a + 1);
	printf("%x,%x", ptr1[-1], *ptr2);
	return 0;
}

运行结果:

4,2000000

💡解析

  • &a:取数组 a 的地址

  • &a + 1:将指向数组的指针做加法运算,相当于移动了一个整个数组的大小。由于 a 是一个包含4个元素的 int 数组,所以加上1后指向数组之外的内存。

  • (int*)(&a + 1):将指向数组之外的内存地址转换为 int 指针。
    ptr1[-1]:通过指针 ptr1 往前访问前一个位置的值,即指向数组的最后一个元素。

  • (int)a:将数组 a 转换为 int 类型的值,实际上是取数组首元素的地址。

  • (int*)((int)a + 1):将数组首元素地址加上1后再转换为 int 指针。

  • *ptr2:通过指针 ptr2 访问指向的值。

  • 根据上述分析,我们可以预测输出结果。

  • 由于 a 是一个包含4个 int 类型元素的数组,在内存中占用4个 int 大小的连续空间。

  • ptr1[-1]:指向数组最后一个元素,即 a[3] 的值为4。

  • *ptr2:指向数组首元素后面的1字节内存,此内存内容未定义。

  • 因此,我们可以预测输出结果为 4,未定义的值(可能是未初始化的值或者随机值)。

这道题在随着后面的学习,会有更深刻的理解,可以在学习完《数组在内存中的储存》后,再来看这道题


📄练习4:

code:

#include <stdio.h>
int main()
{
	int a[3][2] = { (0, 1), (2, 3), (4, 5) };
	int* p;
	p = a[0];
	printf("%d", p[0]);
	return 0;
}

运行结果:

1

💡解析

这里的数组初始化可不是下面这样:

在这里插入图片描述

int a[3][2] = { (0, 1), (2, 3), (4, 5) };

里面是逗号表达式

逗号表达式:从左向右依次计算,但是整个表达式的结果是最后一个表达式的结果

int a[3][2] = { (0, 1), (2, 3), (4, 5) };
<===>
int a[3][2] = { 1, 3, 5 };

在这里插入图片描述

a[0]表示第一行数组名,是第一行首元素的地址,即a[0]==>&a[0][0]

p[0]==>*(p+0),即1

⭕所以,输出结果:1


📄练习5:

X86环境:

code:

#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;
}

运行结果:

FFFFFFFC,-4

💡解析

int a[5][5];

在这里插入图片描述

int(*p)[4];
	p = a;

在这里插入图片描述


&p[4][2]

在这里插入图片描述


 &p[4][2] - &a[4][2]

在这里插入图片描述


printf("%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);

以%p打印和以%d打印是不一样的

%d打印就直接是-4

%p打印的是-4的补码,即FFFFFFFC


📄练习6:

code:

#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;
}

运行结果:

10,5

💡解析

int* ptr1 = (int*)(&aa + 1);

在这里插入图片描述
将&aa + 1强制转换成int* 型,因为ptr1是int*型

然后赋值给ptr1

在这里插入图片描述
*(ptr1 - 1)是10


int* ptr2 = (int*)(*(aa + 1));

*(aa + 1)<==>aa[1]

aa[1]本来就是一个地址,这里的强制类型转换其实是没有意义的,完全就是迷惑人的

将aa[1]赋值给ptr2

那么*(ptr2 - 1)就是5


📄练习7:

code:

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

运行结果:

at

💡解析

这里是将‘w’,’a’,‘a’的地址存到数组里面去

内存布局:

在这里插入图片描述


char** pa = a;
pa++;

a是数组首元素地址,即char*的地址

pa存放首元素地址

在这里插入图片描述


printf("%s\n", *pa);

对pa解引用,访问的就是第二个元素,即打印at


📄练习8:

code:

#include <stdio.h>
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;
}

运行结果:

POINT
ER
ST
EW

💡解析

本道题是比较难的一道题目!!!笔试原题

char* c[] = { "ENTER","NEW","POINT","FIRST" };

内存布局:

在这里插入图片描述


char** cp[] = { c + 3,c + 2,c + 1,c };

内存布局:

在这里插入图片描述


char*** cpp = cp;

图解:

在这里插入图片描述


printf("%s\n", **++cpp);

++cpp=cpp+1

在这里插入图片描述
*++cpp第一层解引用,访问到的是c+2

通过c+2,得到的是P的地址

则再次解引用,以%s打印,得到POINT


printf("%s\n", *-- * ++cpp + 3);

再次++cpp

从上面基础上再+1

在这里插入图片描述
*++cpp解引用,得到c+1

– * ++cpp:在c+1上减去1,得到c

此时也就不再指向N,而是指向E

*-- * ++cpp:再解引用,得到E的地址

*-- * ++cpp + 3:得到ER


printf("%s\n", *cpp[-2] + 3);

cpp[-2]其实就是*(cpp-2)

*cpp[-2]+3也就是 * *(cpp-2)+3

cpp-2在上面的额基础上减2,cpp不会变
在这里插入图片描述
*(cpp-2)通过解引用,得到c+3

**(cpp-2) 再次解引用,得到F

**(cpp-2)+3 ,得到ST


printf("%s\n", cpp[-1][-1] + 1);

cpp[-1][-1]也就是 * (*(cpp-1)-1)

即, * (*(cpp-1)-1)+1

cpp-1图解:
在这里插入图片描述
*(cpp-1)解引用,得到c+2,也就是P的地址

*(cpp-1)-1,得到N的地址

 *(*(cpp-1)-1)再次解引用,拿到N的值


* (*(cpp-1)-1)+1 ,再加1,得到EW

🚩总结

本节题目来自于公司笔试,前面的题目不算很难,需要我们熟练掌握
后面的题目比较难,需要我们逐个分析
理解完这些题目,相信大家对指针有更深刻理解
😏😏😏😏😏😏😏😏😏😏😏😏😏😏😏😏😏😏😏😏😏😏😏

在这里插入图片描述

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

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

相关文章

科技资讯|三星再申请智能戒指商标,智能穿戴进入更小型化发展

三星正在积极扩展可穿戴设备生态&#xff0c;近日向英国知识产权局提交了名为“Samsung Curio”的新商标&#xff0c;其分类为“Class 9”&#xff0c;可能会用于未来的智能戒指。 智能戒指&#xff1a; 可穿戴计算机本质上的智能手环、智能项链、智能眼镜和智能戒指&#xff1…

新KG视点 | 白硕—大模型时代的知识图谱

OpenKG 大模型专辑 导读 知识图谱和大型语言模型都是用来表示和处理知识的手段。大模型补足了理解语言的能力&#xff0c;知识图谱则丰富了表示知识的方式&#xff0c;两者的深度结合必将为人工智能提供更为全面、可靠、可控的知识处理方法。在这一背景下&#xff0c;OpenKG组织…

全套解决方案:基于pytorch、transformers的中文NLP训练框架,支持大模型训练和文本生成,快速上手,海量训练数据!

全套解决方案&#xff1a;基于pytorch、transformers的中文NLP训练框架&#xff0c;支持大模型训练和文本生成&#xff0c;快速上手&#xff0c;海量训练数据&#xff01; 1.简介 目标&#xff1a;基于pytorch、transformers做中文领域的nlp开箱即用的训练框架&#xff0c;提…

如何满足影视飓风对空间智能化的“挑剔”?智哪儿专访Aqara杭州上城区服务商

最近&#xff0c;商务部、住房城乡建设部等13部门联合印发了关于促进家居消费若干措施的通知&#xff0c;明确提出要促进智能家居设备互联互通&#xff0c;推动单品智能向全屋智能发展。在国家相关政策的支持下&#xff0c;全屋智能引领下的空间智能化面临前所未有的发展机会&a…

“传递信任 服务发展”金融科技标准认证生态大会成功举办 同创永益为支持单位

8月16日下午&#xff0c;北京国家金融科技认证中心&#xff08;以下简称“国金认证”&#xff09;在第十三届农村金融机构信息化发展创新座谈会期间&#xff0c;成功举办“传递信任 服务发展”金融科技标准认证生态大会&#xff0c;邀请管理部门、行业机构、产业机构&#xff0…

亚马逊云科技CEO谈及企业领导力原则的核心:坚持顾客至上

亚马逊云科技首席执行官Adam Selipsky几乎从一开始就在那里&#xff1a;他于2005年加入&#xff0c;在效力亚马逊11年后于2016年离开&#xff0c;转而经营Tableau&#xff0c;并于2021年成为亚马逊云科技首席执行官。当时亚马逊云科技前首席执行官安迪贾西(Andy Jassy)接替杰夫…

基于NXP i.MX 6ULL核心板的物联网模块开发案例(3)

前言 本文主要介绍基于创龙科技TLIMX6U-EVM评估板的物联网模块开发案例&#xff0c;适用开发环境&#xff1a; Windows开发环境&#xff1a;Windows 7 64bit、Windows 10 64bit 虚拟机&#xff1a;VMware15.1.0 Linux开发环境&#xff1a;Ubuntu18.04.4 64bit U-Boot&…

大型企业是否有必要进行数字化转型?

在数字化、信息化、智能化蓬勃发展的今天&#xff0c;初创公司可以很轻易的布局规划数字化发展的路径。而对于大型企业而言&#xff0c;其已经形成了较为成熟稳固的业务及组织架构&#xff0c;是否还有必要根据自身行业发展特点寻求数字化转型&#xff1f;&#xff08;比如制造…

喜讯|思迈特软件入选2023爱分析·数据智能优秀厂商

近期&#xff0c;“2023爱分析数据智能优秀厂商”评选结果于第五届数据智能高峰论坛现场正式公布&#xff0c;思迈特软件凭借在数据智能领域的技术及服务水平&#xff0c;从150家数据智能厂商中脱颖而出&#xff0c;成功入选为“2023爱分析数据智能优秀厂商”。 本次评选是基于…

Python“牵手”1688商品列表数据,关键词搜索1688API接口数据,1688API接口申请指南

1688平台API接口是为开发电商类应用程序而设计的一套完整的、跨浏览器、跨平台的接口规范&#xff0c; 1688API接口是指通过编程的方式&#xff0c;让开发者能够通过HTTP协议直接访问1688平台的数据&#xff0c;包括商品信息、店铺信息、物流信息等&#xff0c;从而实现1688平…

Wi-Fi网络泄露你位置的新恶意软件,如何保护自己?

通常&#xff0c;当黑客用恶意软件感染最好的Windows笔记本电脑时&#xff0c;经济利益是他们的动机。然而&#xff0c;他们也喜欢部署信息窃取恶意软件来获取你的个人数据。 Secureworks反威胁部门的安全研究人员发现了一种神秘的新恶意软件&#xff0c;它完全是在寻找其他东…

【Unity小技巧】在Unity中实现类似书的功能(附git源码)

文章目录 前言本文实现的最终效果素材1. 页面素材2. 卡片内容素材地址 翻页实现1. 配置我们的canvas参数2. 添加封面和页码3. 翻页效果4. 添加按钮5. 脚本控制6. 运行效果 页面内容1. 添加卡片内容2. shader控制卡片背面3. 页面背面显示不同卡片 源码参考完结 前言 欢迎来到游…

【两区域系统的自动发电控制】任何区域突然负荷变化的情况下,如何测量两个区域共享的功率研究(Simulink)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

No message found under code ‘-1‘ for locale ‘zh_CN‘.

导出中的报错&#xff1a;No message found under code -1 for locale zh_CN. 报错原因&#xff1a;页面中展示的数据和后端excel中的数据不一致导致 具体原因&#xff1a;

ray-分布式计算框架-集群与异步Job管理

0. ray 简介 ray是开源分布式计算框架&#xff0c;为并行处理提供计算层&#xff0c;用于扩展AI与Python应用程序&#xff0c;是ML工作负载统一工具包 Ray AI Runtime ML应用程序库集 Ray Core 通用分布式计算库 Task -- Ray允许任意Python函数在单独的Python worker上运行&…

矢量调制分析基础

前言 本文介绍VSA 的矢量调制分析和数字调制分析测量能力。某些扫频调谐频谱分析仪也能通过使用另外的数字无线专用软件来提供数字调制分析。然而&#xff0c;VSA 通常在调制格式和解调算法配置等方面提供更大的测量灵活性&#xff0c;并提供更多的数据结果和轨迹轨迹显示。本…

Codeforces Round #894 (Div.3)

文章目录 前言A. Gift Carpet题目&#xff1a;输入&#xff1a;输出&#xff1a;思路&#xff1a;代码&#xff1a; B. Sequence Game题目&#xff1a;输入&#xff1a;输出&#xff1a;思路&#xff1a;代码&#xff1a; C. Flower City Fence题目&#xff1a;输入&#xff1a…

keepalived双机热备 (四十五)

一、概述 Keepalived 是一个基于 VRRP 协议来实现的 LVS 服务高可用方案&#xff0c;可以解决静态路由出现的单点故障问题。 原理 在一个 LVS 服务集群中通常有主服务器&#xff08;MASTER&#xff09;和备份服务器&#xff08;BACKUP&#xff09;两种角色的服务器…

nuxt2-storybook-vite:环境搭建、基础使用 / nuxt项目组件库

一、创建 nuxt2 项目 安装 - NuxtJS | Nuxt.js 中文网 yarn create nuxt-app <项目名> 二、安装 storybook 2.1、初始化 Storybook pnpm add -g storybook/cli npx -p storybook/cli sb init 命令解释 序号命令命令解释1npx -p storybook/cli sb init是一个命令行…

基于JSP+Servlet+mysql员工权限管理系统

基于JSPServletmysql员工权限管理系统 一、系统介绍二、功能展示四、其他系统实现五、获取源码 一、系统介绍 项目类型&#xff1a;Java web项目 项目名称&#xff1a;基于JSPServlet的员工权限管理系统[qxxt] 项目架构&#xff1a;B/S架构 开发语言&#xff1a;Java语言 …