指针进阶(下)指针实操

sizeof 和 strlen

       首先我们来复习一下sizeof 和 strlen 的区别。
sizeof 是操作符,只关注内存中存放的数据的大小,并不会参与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;
}

由于a 是 int 类型的数据,需要的参数为指针(地址),并且 int 占四个字节的空间大小,所以上面三行的 sizeof 计算出来都是 4
我们来看一下运行的结果吧:

strlen 是 string.h 的库函数,主要是计算字符串的长度,当遇到 \0 的时候会停止计算长度。计算出来的是字符个数

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

由于 strlen 遇到 \0 才会停止计算,所以计算 arr1 的长度时,由于不知道什么时候遇到 \0 所以结果是一个不确定的数(也就是随机数)
而计算 arr2 的时候,我们发现 arr2 存放的时一个字符串,也就是说 arr2 实际存储的内容为 a b c \0 这四个字符,由于 strlen 遇到 \0 就会停止计算,也就是说明最后的结果为 3

我们来看一下运行的结果吧:

数组名再深入理解

       我们知道数组名一般来说是指数组的首元素的地址,但是也有两个特殊情况:一个是 sizeof (arr) ,就是一个数组名单独放在sizeof 里面,要注意了,这里是取出的是整个数组,计算的是整个数组的大小;另外一个就是 &arr ,它取出的是整个数组的地址,那么数据类型是什么呢?举个例子:int arr [10] ,我们&arr 取出整个数组的地址,数据类型就是 int (*) [10]

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

由于sizeof(数组名)计算的是整个数组的大小,也就是说我们只需要看数组占多大空间即可,arr1[3]有三个元素,每个元素的大小为 char (1个字节),总大小为 3 * 1 = 3 (字节)

arr2 没有明确说明大小,那我们需要看一下其保存的字符串的大小,也就是 a b c \0 一共有4个元素,每个元素的是 char (1个字节)总大小为 4 * 1 = 4 (字节)

我们来验证一下:

题目剖析

       这里我们通过一些题目来进一步地深入理解数组名与指针地关系。大家可以尝试自己分析一下下面代码的运行结果,然后再看一下解析。

一维数组

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

首先sizeof(a),数组名单独放在 sizeof 内部,计算的是整个数组的大小,也就是 4 * 4 = 16 个字节

sizeof(a+0), a+0说明数组名不是单独放在 sizeof 内部,也就是 arr 指数组首元素的地址,a+0表示跳过0个元素,还是数组首元素的地址,地址就是 4/8 个字节(64位机器地址占8个字节,32位机器地址占4个字节)

sizeof(*a) ,数组名没有单独放在sizeof 内部,就是说 a 代表的是数组首元素的地址,*a数组首元素地址的解引用操作,取出的是数组第一个元素,sizeof 计算的就是数组第一个元素的大小,也就是 4 个字节

sizeof(a+1), 数组名没有单独放在sizeof 内部,就是说 a 代表的是数组首元素的地址,a+1表示跳过一个元素,表示第二个元素的地址,既然是地址,就是 4/8 个字节的大小

sizeof(a[1]), 这个a[1] 表示数组第二个元素,也就是 4 个字节

sizeof(&a), &a 表示取出整个数组的地址,数据类型是 int (*) [4] ,既然是地址,也就是 4/8 个字节了。

我们来看一下运行结果(这里以 64 位机器为例,也就是地址大小为 8 个字节,下面的运行示例也以 64 位机器演示,就不做提醒了)

字符数组

代码一:

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

sizeof(arr), sizeof 内部单独存放数组名,计算的是整个数组的大小,数组有 6 个元素,分别为a b c d e f ;类型是 char 类型(一个字节),一共就是 6 * 1 = 6 (字节)

sizeof(arr+0), arr + 0 显而易见数组名不是单独存放再sizeof 内部,arr 表示首元素的数组名,arr + 0表示跳过0个元素,表示第一个元素的地址,既然是地址,也就是 4/8 个字节

sizeof(*arr), 数组名没有单独存放在 sizeof 内部,arr则表示为数组首元素的地址,*arr 解引用取出数组第一个元素,数据类型为 char ,即一个字节

sizeof(arr[1]),显而易见,arr[1] 取出数组第二个元素,大小为 1 个字节

sizeof(&arr), &arr表示取出整个数组的地址,地址占 4/8 个字节

sizeof(&arr+1),&arr 取出整个数组的地址,数据类型是char (*) [6] , &arr+1跳过一个数组,还是一个地址,就是 4/8 个字节

sizeof(&arr[0]+1), &arr[0] 取出的是数组首元素的地址,+1 表明跳过一个元素,指向数组第二个元素,还是一个地址, 4/8 个字节

我们来看一下运行结果:

代码二:

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

strlen(arr),arr 表示首元素的地址,遇到 \0 才停止计算,是个随机数

strlen(arr+0),arr 表示首元素地址, arr+0表示跳过 0个元素,指向第二个元素,还是一个随机数

strlen(*arr), *arr 取出数组的首元素,strlen 需要传入地址,否则系统会报错,无法运行,硬要访问,a的ascll码值是97,访问的是地址编号为97,这是属于操作系统的是不允许访问的

strlen(arr[1]), arr[1] 取出数组第二个元素,系统报错,硬要访问,a的ascll码值是97,访问的是地址编号为97,这是属于操作系统的是不允许访问的

strlen(&arr),&arr 取出整个数组的地址,,不知道什么时候遇到 \0 ,是个随机数

strlen(&arr+1),&arr +1 取出整个数组的地址,并跳过一个数组,是个随机数

strlen(&arr[0]+1), &arr[0]+1 取出首元素的地址,跳过一个元素,指向第二个元素,是个随机数

代码三

#include <stdio.h>int main()
{char arr[] = "abcdef";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;
}

arr 存放着 a b c d e f \0 这 7 个元素

sizeof(arr),取出整个数组的大小, 7 * 1 = 7 个字节

sizeof(arr+0), arr + 0 跳过 0 个元素,指向数组第一个元素,是个地址,4/8 个字节

sizeof(*arr), *arr 表示数组首元素的解引用,取出第一个元素,1 个字节

sizeof(arr[1]), arr[1] 取出第二个元素,大小为 1 个字节

sizeof(&arr), &arr 取出整个数组的地址,数据类型是 char (*) [7] ,地址为 4/8 个字节

sizeof(&arr+1), 取出整个数组的地址,并跳过一个数组,还是一个地址,4/8 个字节

sizeof(&arr[0]+1), 取出数组首元素的地址,并跳过一个元素,指向数组的第二个元素,还是一个地址,4/8 个字节

下面是代码运行结果:

代码四

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

arr 存放着 a b c d e f \0 这 7 个元素

strlen(arr),传入数组首元素的地址,遇到 \0 停下,一共就是 6

strlen(arr+0),传入数组首元素的地址,并跳过 0 个元素,还是指向数组首元素,一共是 6

strlen(*arr), *arr 取出数组首元素,不是一个地址,系统报错,硬要访问,a的ascll码值是97,访问的是地址编号为97,这是属于操作系统的是不允许访问的

strlen(arr[1]),arr[1] 取出数组第二个元素,不是一个地址,系统报错,硬要访问,b的ascll码值是98,访问的是地址编号为98,这是属于操作系统的是不允许访问的

strlen(&arr), &arr 取出整个数组的地址,还是以数组首元素的地址表示,也就是 6

strlen(&arr+1), &arr + 1 取出整个数组的地址,&arr 的数据类型是 char (*) [7] ,并跳过一个数组,也就是指向一个未知的空间,无法知道什么时候遇到 \0 ,是个随机数

strlen(&arr[0]+1),&arr[0]表示数组首元素的地址,+1 跳过一个元素,也就是从第二个元素开始计算,则为 5

代码五

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

p 是一个字符指针,指向的是一个常量字符串

sizeof§, p 是一个指针变量,也就是地址,4/8 个字节

sizeof(p+1), p + 1 跳过一个元素,指向第二个元素,还是一个地址,4/8 个字节

sizeof(*p), *p 字符串首元素解引用,取出 a,a 是 char 类型,1 个字节

sizeof(p[0]), p[0] 取出字符‘a’,大小为 1 个字节

sizeof(&p), &p 取出的是 p 的地址,是一个二级指针,数据类型为 char** ,也就是 4/8 个字节

sizeof(&p+1), &p + 1,取出 p 的地址,是个二级指针,数据类型为 char** ,再跳过一个数组 ,还是一个地址, 4/8 个字节

sizeof(&p[0]+1), &p[0] 表明取出首元素的地址,跳过一个元素,指向第二个元素的地址,也就是 ‘b’ 的地址,还是一个地址,4/8 个字节

看一下运行结果:

代码六

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

strlen(p) 从第一个元素开始,到 \0 结束,就是6

strlen(p+1), 从第一个元素开始,+1 跳过一个元素,从第二个元素计算,也就是 5

strlen(*p), *p 取出第一个元素,不是地址,系统报错,硬要访问,a的ascll码值是97,访问的是地址编号为97,这是属于操作系统的是不允许访问的

strlen(p[0]), p[0] 取出第一个元素,不是地址,系统报错,硬要访问,a的ascll码值是97,访问的是地址编号为97,这是属于操作系统的是不允许访问的

strlen(&p), &p 指向 p 的地址,未知空间,不知道何时遇到 \0 , 随机值

strlen(&p+1), &p 取出 p 的地址, &p+1 跳过一个 char* , 未知空间,随机值

strlen(&p[0]+1), &p[0] 取出首元素的地址,+1 跳过一个元素,指向第二个元素,也就是 5

课外探讨:&p 和 &p + 1 有关系吗?实际没有,因为你不知道 p 内部是否有 \0 ,如果有,他们的大小就是差 1


二维数组

首先我们来回顾一下二维数组,二维数组可以看出一个个一维数组组成的,举个例子:
int arr [3][3] = {1,2,3,4,5,6,7,8,9};

我们可以将二位数组看成由一个又一个的一维数组组成,则二维数组的首元素地址就是第一行的地址。

arr [0], arr[1], arr[2], 如果单独放在sizeof 内部的话分别表示 数组的第一行 ,第二行,第三行的数组名,否则就是代表数组第一行、第二行、第三行的首元素地址。

我们来看一下下面的题目:

#include <stdio.h>
#include <string.h>int main()
{int a[3][4] = { 0 };printf("%d\n", sizeof(a));printf("%d\n", sizeof(a[0][0]));printf("%d\n", sizeof(a[0]));printf("%d\n", sizeof(a[0] + 1));printf("%d\n", sizeof(*(a[0] + 1)));printf("%d\n", sizeof(a + 1));printf("%d\n", sizeof(*(a + 1)));printf("%d\n", sizeof(&a[0] + 1));printf("%d\n", sizeof(*(&a[0] + 1)));printf("%d\n", sizeof(*a));printf("%d\n", sizeof(a[3]));return 0;
}

sizeof(a), a 单独放在sizeof 内部, 表示整个数组的大小,3 * 4 * (sizeof(int)) = 48

sizeof(a[0][0]), a[0][0] 大小为 4 个字节

sizeof(a[0]), a[0] 单独放在 sizeof 内部,表示第一行的数组大小, 4 * 4 = 16

sizeof(a[0]+1), a[0] 不是单独放在 sizeof 内部,则表示的是第一行的首元素 ,也就是 &a[0][0] ,+1 跳过一个元素,指向 a[0][1] 的地址,4/8 个字节

sizeof(*(a[0]+1)) , a[0] + 1 == &a[0][1], 解引用之后就是a[0][1], 也就是 4 个字节

sizeof(a+1), a 再这里表示数组的首元素也就是第一行的地址,+1跳过一行,就是第二行的地址,地址就是 4/8 个字节

sizeof(*(a+1)), a+1 指向第二行的地址,解引用后就是取出第二行的数组,大小就是 4 * 4 =16

sizeof(&a[0]+1), &a[0] 取出第一行的地址,+ 1 表示跳过一行,指向第二行的地址,4/8 个字节

sizeof(*(&a[0]+1)) , 解引用取出第二行,也就是 4 * 4 = 16 个字节

sizeof(*a), a 表示数组首元素的地址(第一行的地址),解引用取出第一行,大小为 4 * 4 = 16

sizeof(a[3]), a[3] 大小为 4 * 4 = 16 个字节,为什么不会有越界的问题,因为 sizeof 不参与计算,a[3]无需真实存在,仅仅通过类型的推断就能算出长度,a[3] 表示第四行的数组名,单独放在 sizeof 内部,计算的是第四行的大小,4 * 4 = 16

我们来看一下运行结果:


指针运算笔试题解析

题目一

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

我们用图片和文字来双重解析一下:

a表示数组首元素的地址,+1 跳过一个元素,* (a+1) 就是 取出 第二个元素( 2 )
&a 表示取出整个数组的元素,+ 1 跳过一个数组, int* prt = (int*)(&a+1),将&a+1 原先的数据类型(int () [5]) 强制类型转化为 int ,ptr-1 要注意了是向后移动一个元素,为什么呢?因为 ptr 的类型是 int* , - 1 也就是向后移动 int,最终指向 5

来看一下运行结果:

题目二

//在X86环境下
//假设结构体的⼤⼩是20个字节
//程序输出的结构是啥?#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;
}

这里的0x1就是表示1


p的类型是 struct Test * ,+1表示跳过一个结构体,也就是地址+20,由于0x是16进制,所以20换成16 进制就是 0x14, 0x100000 + 0x14 = 0x100014

(unsigned long) p将 p 强制类型转换为 无符号的长整型,+1就是0x100001

(unsigned int*)p 将p 强制类型转换为 unsigned int*,+1跳过 unsigned int 的字节大小,也就是0x100000 + 0x4 = 0x100004

运行结果如下:

题目三

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

要注意了,(,)这是一个逗号表达式,都好表达式的结果去最后一个值,所以{(0,1),(2,3),(4,5)} 的值应该为{1,3,5}

所以数组a[3][2]={1,3,5,0,0,0},p=a[0]表示数组第一行的地址,p[0]相当于a[0][0]也就是取出第一行的首元素,也就是 1

运行结果如下:

题目四

//假设环境是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;
}

int a[5][5],a 的类型是int()[5], 因为这是一个二维数组,二维数组的首元素是二维数组的第一行,所以a的类型是 int () [5]
int (p)[4],无需置疑,p的类型就是int ()[4]

那么&a[4][2]就无需多讲,p[4][2]其实就是*(*(p+4)+2), p+4表示跳过4个int [4],然后+2 表示跳过2个int ,指向图如下图所示。

然后我们要知道指针 - 指针代表就是两个指针之间的元素个数,所以&p[4][2]-&a[4][2] 之间有4个元素,由于是低地址减去高地址,所以是-4

%d打印有符号的整数,就是-4
%p打印的是地址,所以我们需要知道-4在内存中然后存储,那就要将-4的补码写出来(-4的原码是10000000000000000000000000000100,反码就是11111111111111111111111111111011,补码就是11111111111111111111111111111100)也由于是x86环境下,所以是32位的地址,于是将补码转换成16进制的数值(也就是FFFFFFFC)

运行结果:

题目五

#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跳过一个数组,然后强制类型转换为int*, 赋给ptr1
aa表示数组的首元素,+1跳过一个元素,然后强制类型转换为int*, 赋给ptr2

ptr-1 由于ptr 是int*类型,所以跳过一个int ,同理ptr2-1也是跳过一个int

运行结果:

题目六

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

a是个字符指针数组,存放的是work,at,alibaba的首元素的地址
pa = a意味着pa是a的首元素的地址,pa++跳过一个char*,也就是指向a 的第二个元素,*pa 取出a的第二个元素,也就是at的首元素的地址,所以打印at

运行结果:

题目七

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

我们先将图画出来:

在这里+的优先级最低

** ++ cpp ,cpp先++,指向c+2的地址,解引用取出c+2,再解引用取出POINT的地址,

*--*++cpp+3 

首先cpp在上面已经自增过一次,再++之后cpp就指向c+1的地址,解引用取出c+1,然后c+1再–,变成c,再解引用就会取出ENTER的地址,然后+3,跳过三个元素,也就是指向E的地址,打印出ER

现在cpp是指向cp第三个元素的地址!

cpp[-2]+3,我们先来解读cpp[-2],cpp[-2]表示(cpp-2),也就是取出c+3的地址,*cpp[-2]解引用取出FIRST的地址,+3跳过三个元素,也就是指向S的地址,打印出ST

cpp[-1][-1],我们来分解一下*( *( cpp-1 )-1 ),cpp-1指向cp第二个元素的地址,解引用取出c+2, c+2-1=c+1, 解引用一下就取出NEW的地址,最后cpp[-1][-1] + 1 之后就是指向E的地址,也就是打印出EW

运行结果:

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

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

相关文章

javascript之常见设计模式

前置知识 构造函数 普通函数 在 JavaScript 中&#xff0c;很多时候&#xff0c;你需要避免使用 new 关键字。 类 单例模式 顾名思义&#xff0c;一个类确保只有一个实例&#xff0c;供全局访问 实现单例模式通常包括以下关键步骤&#xff1a; 私有构造函数&#xff1a;确保…

3.8 动态规划 背包问题

一.01背包 46. 携带研究材料&#xff08;第六期模拟笔试&#xff09; (kamacoder.com) 代码随想录 (programmercarl.com) 携带研究材料: 时间限制&#xff1a;5.000S 空间限制&#xff1a;128MB 题目描述: 小明是一位科学家&#xff0c;他需要参加一场重要的国际科学大会…

基于textdistance计算文本相似度

textdistance是Python的第三方库&#xff0c;用于计算文本之间的相似度或距离。它提供了30个算法&#xff0c;简单易用。 安装 pip install textdistance# 使用扩展库&#xff0c;提高性能 pip install "textdistance[extras]"使用 import textdistance# 计算编辑…

提醒一下!今年考研的人不要太老实了!!

今年准备计算机考研的同学&#xff0c;别太老实了&#xff01;别人说什么你就信什么 如果你的工作能力不足以支撑找到一个满意的工作&#xff0c;那我建议再沉淀两年&#xff01; 很多同学其实有点眼高手低&#xff0c;在计算机专业&#xff0c;低于1w的工作看不上&#xff0…

用一个 Python 脚本实现依次运行其他多个带 argparse 命令行参数的 .py 文件

&#x1f349; CSDN 叶庭云&#xff1a;https://yetingyun.blog.csdn.net/ 问题描述&#xff1a;在 Windows 环境中&#xff0c;您希望通过一个 Python 脚本来实现特定的自动化任务&#xff0c;该任务需要依次运行其他多个带 argparse 命令行参数的 .py 文件。您希望找到一种简…

华为设备小型园区网方案(有线+无线+防火墙)

&#xff08;一&#xff09;配置有线部分 1.配置LSW2 &#xff08;1&#xff09;创建相关vlan [LSW2]vlan batch 10 3000 &#xff08;2&#xff09;配置连接LSW1的Eth-Trunk1&#xff0c;透传VLAN 10 3000 [LSW2]int Eth-Trunk 1 [LSW2-Eth-Trunk1]port link-type trunk [LSW2…

安装nexus + 部署私有maven仓库

安装nexus 部署私有maven仓库 文章目录 安装nexus 部署私有maven仓库1.下载2.解压3.修改配置文件4.启动5.访问6.查看默认密码7.创建私库8.修改代码配置文件9.在maven 的setting.xml中配置私库的账号密码10.运行manve 【deploy】命令测试11.maven项目引用私库12. 重新加载mave…

指针进阶(4)看一下这些与指针有关的题你都会做吗?

c语言中的小小白-CSDN博客c语言中的小小白关注算法,c,c语言,贪心算法,链表,mysql,动态规划,后端,线性回归,数据结构,排序算法领域.https://blog.csdn.net/bhbcdxb123?spm1001.2014.3001.5343 给大家分享一句我很喜欢我话&#xff1a; 知不足而奋进&#xff0c;望远山而前行&am…

智慧公厕的三大特点:信息化、数字化、智慧化

智慧公厕是以物联网、互联网、大数据、云计算等先进技术为支撑&#xff0c;对公共厕所的使用、运营、管理、养护进行全方位高效应用的创新型公厕。它具有三大显著特点&#xff1a;&#xff08;ZonTree中期&#xff09;信息化、数字化和智慧化。本文以智慧公厕源头实力厂家广州中…

Vision Transformer结构解析

Vision Transformer结构解析 ViT简介ViT参数量ViT三大模块ViT图像预处理模块——PatchEmbed多层Transformer Encoder模块MLP&#xff08;FFN&#xff09;模块 基本的Transformer模块Vision Transformer类的实现Transformer知识点网络结构计算复杂度对比Transformer的参数量和计…

几种电脑提示mfc140.dll丢失的解决方法,以及如何预防mfc140.dll丢失

mfc140.dll真是一个超级关键的动态链接库文件&#xff01;一旦这个文件不翼而飞&#xff0c;可能会导致一些程序无法顺利运行&#xff0c;甚至给系统带来麻烦。但别担心&#xff01;遇到mfc140.dll文件丢失的情况&#xff0c;我们有一堆应对措施可以立马施行&#xff0c;确保问…

类和对象-C++运算符重载

#include <iostream> #include <string> using namespace std;class Person { public:Person(int age){m_Agenew int (age);}~Person(){if(m_Age!NULL){delete m_Age;m_AgeNULL;}}//重载 赋值运算符Person& operator (Person &p){//编译器提供深拷贝//m_Ag…