指针进阶(下)指针实操

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

运行结果:

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

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

相关文章

USB2.0设备检测过程信号分析

1.简介 USB设备接入的Hub端口负责检测USB2.0设备是否存在和确定USB2.0设备的速度。检测设备是否存在和确定设备速度涉及一系列的信号交互&#xff0c;下面将分析该过程。 2.硬件 USB低速设备和全速/高速设备的连接器在硬件结构上有所不同&#xff0c;而主机或者Hub接收端连接…

redis中的zset的原理

一、zset有序集合的原理 如果有序集合元素个数少于128个且元素值小于64字节&#xff0c;使用压缩列表&#xff08;新版本已经废弃压缩列表改用listpack数据结构了&#xff09; 如果不满足上述条件&#xff0c;采用跳表作为redis的底层数据结构 二、压缩列表 1.由连续内存块组…

一张照片一键换脸:无需数据集和训练 | 开源日报 No.186

s0md3v/roop Stars: 23.6k License: AGPL-3.0 roop 是一个一键换脸的项目。 该项目可以通过一张目标人物的照片&#xff0c;实现对视频中人脸进行替换&#xff0c;无需数据集和训练。其主要功能、关键特性和核心优势包括&#xff1a; 可以在计算机上运行&#xff0c;并支持 C…

mysql 8.0 日志文件无权限问题处理

无论如何修改权限总是报这个日志文件权限问题。 解决方法 输入指令&#xff1a; setenforce 0 systemctl restart mysgld

csgo搬砖核心步骤,月入1000-10000你也可以的!

近年网络游戏产业的爆炸式增长&#xff0c;虚拟物品的交易需求也越来越大&#xff0c;为了满足虚拟物品的交易需求&#xff0c;网络游戏交易平台开始兴起和发展。网游交易平台的交易项目包括帐号交易、游戏币交易、装备交易这几种主要交易项目&#xff0c;其交易模式可分为C2C模…

01、python_爬虫的相关概念

一、什么是爬虫&#xff1f; 爬虫是网络爬虫的简称&#xff0c;指的是一种自动化程序&#xff0c;用于在互联网上抓取信息。爬虫的核心工作包括爬取网页、解析数据和存储数据。 通俗来说就是&#xff1a;通过一个程序&#xff0c;根据url(http://taobao.com)进行爬取网页&…

模拟实现strlen函数

一、逐个计数法 #include<assert.h> #include<stdio.h>size_t my_strlen(const char* p) {int count 0;assert(p);//断言while (*p ! \0){p;count;}return count; }int main() {char str[] "hello world";size_t len my_strlen(str);printf("%d…

【重制版】WSDM 2024 2023时空时序论文总结

&#x1f31f;【紧跟前沿】“时空探索之旅”与你一起探索时空奥秘&#xff01;&#x1f680; 欢迎大家关注时空探索之旅 WSDM 2024于2024年3月4日-3月8日在墨西哥梅里达&#xff08;Mrida, Mxico&#xff09;正在举行。目前官网已经放出了所有被录用论文的表单&#xff08;链接…

向量的内积、长度、正交性

目录 向量的内积 向量的长度&#xff08;模&#xff09; 标准正交基 标准正交化 正交矩阵 向量的内积 向量的长度&#xff08;模&#xff09; 标准正交基 标准正交化 正交矩阵

网工内推 | 网络工程师,IE认证优先,最高15K,有项目绩效奖金

01 重庆并联网络科技有限公司 招聘岗位&#xff1a;网络工程师 职责描述&#xff1a; 1、负责集成项目的相关实施工作&#xff08;设备上架安装、网络设备配置、服务器相关系统配置安装、相关软件环境搭建及配置等&#xff09; 2、负责项目现场技术维护与技术支持&#xff1b;…

html--心花怒放

代码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><title>Canvas 绘制一个❤</title><link rel"shortcut icon" href"../../assets/images/icon/favicon.ico" type"ima…

Unity插件之天气系统UniStorm

首先呢&#xff0c;它是一款强大的动态昼夜天气系统&#xff0c;能够以较快的帧速率创建AAA级动态生成的天气、照明和天空&#xff0c;并且具有300多个可定制的组件&#xff0c;允许用户创建任何可以想象的环境。 第一步&#xff1a;他需要两个物体Camera摄像机、Player播放器…

力扣刷题Days13-101对称二叉树(js)

目录 1,题目 2&#xff0c;代码 2.1递归思想 2.2队列--迭代思想 3&#xff0c;学习与总结 1,题目 给你一个二叉树的根节点 root &#xff0c; 检查它是否轴对称。 2&#xff0c;代码 2.1递归思想 return dfs(left.left, right.right) && dfs(left.right, right.l…

计算机组成原理之机器:存储器之高速缓冲存储器

计算机组成原理之机器&#xff1a;存储器之高速缓冲存储器 笔记来源&#xff1a;哈尔滨工业大学计算机组成原理&#xff08;哈工大刘宏伟&#xff09; Chapter3&#xff1a;存储器之高速缓冲存储器 3.1 概述 3.1.1 为什么用cache&#xff1f; 角度一&#xff1a;I/O设备向…

el-form-item内的el-select如何自适应宽度

最近在使用element-ui做后台管理的时候&#xff0c;有个需求是在弹窗组件里面&#xff0c;添加一个el-select下拉框选项&#xff0c;但是给el-select设置的宽度无法自适应&#xff0c;原因很简单&#xff0c;我们不需要设置固定宽度&#xff0c;设置百分比就行了&#xff0c;让…

你适合学Python吗?学了Python可以做什么工作?

每天叫醒你的不是闹钟&#xff0c;而是梦想 目前在很多行业中都在越来越多的应用Python&#xff0c;这也是很多行业学习Python的原因&#xff0c;Python主要的应用领域有哪些呢&#xff1f;今天我们就来详细看一下。 谁适合学Python&#xff1f; 我们首先来看一看谁在学Pytho…

【AI+应用】模仿爆款视频二次创作短视频操作步骤

本来不想水这篇的&#xff0c; 剪辑软件估计很多人用的比我还6。 今天自己遇到1个需求&#xff0c;我看到一篇公众号文章的视频觉得有意思&#xff0c;但视频有点长&#xff0c;我没带耳机看视频的习惯&#xff0c;就想着能不能下载下来&#xff0c; 提取视频的音频转为文字&am…

腾讯:《智能科技 跨界相变——2024数字科技前沿应用趋势》

1月23日&#xff0c;腾讯发布了题为《智能科技 跨界相变——2024数字科技前沿应用趋势》的报告&#xff0c;报告从计算重塑、智能升维、沉浸交互、未来连接四个方面&#xff0c;对100多项未来技术和重点方向给出了趋势性判断。并表示我们正驶向一个由连接衍生交互、由计算催生智…

JavaWeb笔记 --- 二、Maven

二、Maven Maven概述 所有的IDE创建的Maven项目都可以使用 Maven简介 Maven模型 Maven常用命令 Maven生命周期 Maven坐标 依赖管理 dpendencies&#xff1a;依赖 依赖范围

2016年认证杯SPSSPRO杯数学建模C题(第一阶段)如何有效的抑制校园霸凌事件的发生解题全过程文档及程序

2016年认证杯SPSSPRO杯数学建模 C题 如何有效的抑制校园霸凌事件的发生 原题再现&#xff1a; 近年来&#xff0c;我国发生的多起校园霸凌事件在媒体的报道下引发了许多国人的关注。霸凌事件对学生身体和精神上的影响是极为严重而长远的&#xff0c;因此对于这些情况我们应该…