指针 --- 进阶

news2025/1/23 13:11:07

先看目录,看你知道多少

目录

1.字符指针

2.指针数组

3.数组指针

4.数组传参和指针传参

5.函数指针

6.函数指针数组

7.指向函数指针数组的指针

8.回调函数


什么是指针,我们在之前的《指针》章节已经接触过了,我们知道了指针的概念:

1.指针就是个变量,用来存放地址,地址唯一标识一块内存空间。

2指针的大小是固定的4/8个字节( 32位平台/64位平台)。

3.指针是有类型,指针的类型决定了指针的+-整数的步长,指针解引用操作的时候的权限。

4.指针的运算。可以比较大小,如果两个指针指向同一数组,也可以相减,得到数的绝对值是之间的元素个数。

1.字符指针

在指针类型中,我们知道有一种指针类型为字符指针 char*

一般使用情况:

int main()
{
	char ch = 'w';
	char* pc = &ch;//pc就是字符指针,指向字符的指针就是字符指针
    *pc = 'w';
	return 0;
}

还有一种使用情况如下:存放常量字符串的首字符的地址

#include<stdio.h>

int main()
{
	//下面右边是一个表达式,表达式的值是首字符的地址
	char* p = "abcdef";//是把首字符的地址放到p中  
	//*p = 'w';//错误,这里字符串是常量字符串,不能修改  
	//const char *p = "abcdef";//这样写会更加严谨,
	
	char arr[] = "abcdef"; //可以想象为字符串是arr,p是存放的arr的首元素地址
	char* parr = arr;     // p指向的是arr首元素地址,不同的是arr数组是可以修改
	*parr='w';//可以修改arr字符串的值
	printf("%s\n", arr);

	return 0;
}

代码 char* p = "abcdef"; 特别容易让我们以为是把字符串 abcdef 放到字符指针 p 里了,但是/本质是把常量字符串 abcedf 首字符的地址放到了p中,也就是a的地址放到了指针变量p中。

看一下面这道题

#include <stdio.h>

int main()
{
	char str1[] = "hello xilanhua";
	char str2[] = "hello xilanhua";

	const char* str3 = "hello xilanhua";
	const char* str4 = "hello xilanhua";

	if (str1 == str2)
	{
		printf("str1 和 str2 相等\n");
	}
	else
	{
		printf("str1 和 str2 不相等\n");
	}
	if (str3 == str4)
	{
		printf("str3 和 str4 相等\n");
	}
	else
	{
		printf("str3 和 str4 不相等\n");
	}

	return 0;
}

str1 和str2 是两个字符数组的首元素地址,在内存中开辟不同的空间,所以str1和str2不相等,

str3 , str4 存放常量字符串 "hello xilanhua" 的首字符地址,常量字符串不能被修改,所以在内存中没有必要存两份,只需要开辟一份空间,两个指针都指向的这份空间,所以两个地址相同的。

所以输出结果为

2.指针数组

可以通过类比:

整型数组,存放整形的数组

字符数组,存放字符的数组

指针数组,存放指针的数组。

int* arr1[5];//整形指针数组
char* arr2[5];//一级字符指针的数组
char** arr3[5];//二级字符指针的数组

根据上面学的字符指针,我们看一下下面代码:

#include <stdio.h>

int main()
{
	char* arr[] = { "abcdef","hehe","xilanhua" };//字符指针数组
	//               arr0	   arr1		arr2    //这里存放的是常量字符串的首字符的地址

	//*arr[1] = 'w';//错误,这里数组元素指向的也是常量字符串,不能修改

	for (int i = 0; i < 3; i++)
	{
		printf("%s\n", arr[i]);
	}

	return 0;
}

结果

 看这段代码:

#include<stdio.h>

int main()
{
	int arr1[] = { 1,2,3,4,5 };
	int arr2[] = { 2,3,4,5,6 };
	int arr3[] = { 3,4,5,6,7 };
	//数组名是首元素地址
	//arr是存放整形指针的数组    指针数组
	int* arr[] = { arr1,arr2,arr3 };
	int i = 0;
	int j = 0;
	for (i = 0; i < 3; i++)
	{
		for (j = 0; j < 5; j++)
		{
			printf("%d ", arr[i][j]);//和二维数组很像
			//printf("%d ", *(arr[i] + j));
            //*(arr[i]+j)==arr[i][j]     arr[i]==*(arr+i)
		}
		printf("\n");
	}

	return 0;
}

结果:

可以发现和二维数组很像,但是本质上是不同的,二维数组是连续存放的空间,但是这里不是,是通过指针联系起来的。

3.数组指针

也是通过类比:

整形指针,指向整形的指针变量

字符指针,指向字符的指针变量

数组指针,指向数组的指针变量

下面哪一个是数组指针?

int* p1[10];

int (*p2)[10];

解释:

int (*p)[10]

p先和*结合,说明p是一个指针变量,然后指着指向的是一个大小为10个整型的数组。所以p是一个指针,指向一个数组,叫数组指针。
这里要注意:[ ] 的优先级要高于*号的,所以必须加上()来保证p先和*结合。不加(),就是指针数组。

1. &数组名 和 数组名

看下面代码:

#include<stdio.h>
int main()
{
	int arr[10] = { 0 };
	printf("%p\n", arr);//数组名是首元素地址 两个例外 &数组名和sizeof(数组名)
	printf("%p\n", &arr[0]);//首元素地址
	
	printf("%p\n", &arr);  //三个数值上相等

	printf("%p\n", arr + 1);//int* + 1
	printf("%p\n", &arr[0] + 1);//int* + 1
	printf("%p\n", &arr + 1);//加了40 int(*)[10]+1
	int(*p)[10] = &arr;
	//int(*)[10]  这个数组指针的类型

	return 0;
}

根据上面的代码我们发现,其实&ar r和 arr,虽然值上是相同的,但是意义是不相同的。实际上:&arr表示的是数组的地址,而不是数组首元素的地址。(细细体会一下)。数组的地址+1,跳过整个数组的大小,所以 &arr+1 相对于&arr 的差值是40.

总结:

 数组名是首元素地址 有两个例外 &数组名 和 sizeof(数组名)
1.sizeof(arr)  -  sizeof内部单独放一个数组名的时候,数组名表示整个数组,计算得到的是数组的大小
2.&arr   -   这里的数组名表示的是整个数组,取出的是整个数组的地址,从地址值的角度来讲和数组的首元素地址是一样的,但是意义不一样 

2.访问数组元素的不同方式

#include<stdio.h>
int main()
{
	int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
	int i = 0;
	int sz = sizeof(arr) / sizeof(arr[0]);
	
	//下标形式访问数组
	for (i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}


	//使用指针访问数组
	int* p = arr;
	for (i = 0; i < sz; i++)
	{
		for (i = 0; i < sz; i++)
		{
			printf("%d ", *(p + i)); 
		}
	}

	//数组指针访问数组   虽然对,但是不推荐
	int(*pa)[10] = &arr; 
	//pa == &arr
	//*pa == *&arr
	//*pa == arr 数组名 不是两种特殊情况 是首元素地址
	for (i = 0; i < sz; i++)
	{
		printf("%d ", *((*pa)+i));//(*pa)[i];
	}

	return 0;
}

注意:

int(*pa)[10] = &arr; 
    //pa == &arr
    //*pa == *&arr
    //*pa == arr 数组名 不是两种特殊情况 是首元素地址

所以:数组指针 解引用 是数组首元素地址

3. 数组指针的使用

二维数组的传参要使用到指针数组。

#include<stdio.h>

//普通接收二维数组传参形式
void print1(int arr[3][5], int r, int c)
{
	int i = 0;
	int j = 0;
	for (i = 0; i < r; i++)
	{
		for (j = 0; j < c; j++)
		{
			printf("%d ", arr[i][j]);
		}
		printf("\n");
	}
}

//二维数组传参用到数组指针
void print(int (*arr)[5], int r, int c)//指针形式接受  数组指针
{
	int i = 0;
	int j = 0;
	for (i = 0; i < r; i++)//arr+i是每一行  *(arr+i)是每一行的数组名 arr[i] 首元素地址就是每一行的第一个元素的地址
	{
		for (j = 0; j < c; j++)
		{
			//printf("%d ", *(*(arr + i) + j));//解引用数组指针就是数组首元素地址,二维数组是第一行地址
			printf("%d ", arr[i][j]);//*(*(arr+i)+j)==*(arr+i)[j]==arr[i][j]
		}
		printf("\n");
	}
}


int main()
{
	int arr[3][5] = { 1,2,3,4,5,2,3,4,5,6,3,4,5,6,7 };
	print(arr, 3, 5);//二维数组的数组名,也表示首元素地址
	//二维数组的首元素地址是第一行的地址,存放的是一维数组   
	//可以理解为二维数组是存放一维数组的数组
	return 0;
}

二维数组的数组名,也表示首元素地址。二维数组的首元素地址是第一行的地址,存放的是一维数组。可以理解为二维数组是存放一维数组的数组

4.数组传参和指针传参

在写代码的时候难免要把【数组】或者【指针】传递给函数,那函数的参数该如何设计呢?

一维数组传参:

形参可以是数组,也可以是指针,当形参是指针时,要注意类型。

#include<stdio.h>

void test(int arr[])//不会真的创建一个数组,不写大小
{}
void test(int arr[10])//和传入格式一样,不会真的创建
{}
void test(int* arr)//数组名,首元素地址,指针接受
{}
void test2(int* arr[20])//指针数组,和传入格式一样
{}
void test2(int** arr )//数组首元素是int* 类型,*arr说明他是指针
{}

int main()
{
	int arr[10] = { 0 };//整形数组
	int* arr2[20] = { 0 };//整形指针数组

	test(arr);

	test2(arr2);
	return 0;
}

二维数组传参:

参数可以是指针,也可以是数组,如果是数组,行可以省略,列不能省略,如果是指针,传参传过去的是第一行的指针,形参就应该是数组指针

void test(int arr[3][5])
{}
//void test(int arr[][])//错误,只能省略行,因为得知道一行有几个元素
//{}
void test(int arr[][5])
{}
//void test(int *arr)//错误,二维数组的首元素地址是第一行的地址
//{}
//void test(int* arr[5])//错误,这是一个指针数组,存放整形指针的数组
//{}
void test(int(*arr)[5])//数组指针,可以接收实参
{}
//void test(int** arr)//错误,二级指针
//{}

int main()
{
	int arr[3][5] = { 0 };
	test(arr);
	return 0;
}

一级指针传参:

#include<stdio.h>
void print(int* p, int sz)
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%d\n", *(p + i));//通过指针访问
	}
}

int main()
{
	int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
	int* p = arr;//一级指针
	int sz = sizeof(arr) / sizeof(arr[0]);//元素个数
	print(p, sz);
	return 0;
}

思考:当一个函数参数是一个指针时,函数能接收什么参数

void print(int* p);    
1.int a;
  print(&a);//变量的地址
2.int* p=&a;
  print(p);//指针变量
3.int arr[10];
  print(arr);//数组名

二级指针传参:

#include<stdio.h>

//用二级指针接收
void test(int** ptr)
{
	printf("num = %d\n", **ptr);
}

int main()
{
	int n = 10;
	int* p = &n;
	int** pp = &p;
	test(pp);
	test(&p);
	return 0;
}

思考:当函数的参数为二级指针的时候,可以接收什么参数

1.一级指针的地址

2.二级指针变量,

3.指针数组

5.函数指针

这里还是类比:

整形指针,指向整形的指针 int*

字符指针,指向字符的指针 char*

数组指针,指向数组的指针,int arr[10];  int (*p)[10] = &arr;

函数指针,指向函数的指针, 即存放函数的地址

那么问题来了,函数有地址吗

#include<stdio.h>
int Add(int x, int y)
{
	return x + y;
}
//&函数名得到就是函数的地址
int main()
{
	printf("%p\n", &Add);//可以运行
	printf("%p\n", Add);//可以运行

	return 0;
}
//函数名 和 &函数名都是函数的地址

结果: 

 可以得出结论,函数名 和 &函数名 都是函数的地址

使用函数指针调用函数:

#include<stdio.h>
int Add(int x, int y)
{
	return x + y;
}

int main()
{
	//()是pf先于*结合,说明pf是指针
	int (*pf)(int, int) = Add;//函数的地址要存起来,就得放在 函数指针变量pf 中
	//最前面的是函数返回类型,后面括号内是函数参数类型

	//通过函数指针调用函数
	int ret = (*pf)(3, 5);
	ret = Add(3, 5);//函数可以这样调用,Add是地址
	ret = pf(3, 5);//所以这样写也对
	ret = (*****pf)(3, 5);//编译器在处理时,会把*去掉,也没有问题
	printf("%d\n", ret);//8

	return 0;
}

可以阅读两段有趣的代码

1.(*(void(*)( ))0)( );

#include<stdio.h>

int main()
{
	//1.将0强制类型转换为void(*)()类型的函数指针
	//2.这就意味着0地址处放着一个函数,函数无参,返回类型是void
	//3.调用0地址处的这个函数
	(*(void(*)())0)();//前面*可以不写

	return 0;
}

2.void(* signal(int, void(*)(int) ) )(int);

int main()
{
	void(* signal(int, void(*)(int) ) )(int);
	//signal先于()结合
	//void(*)(int)
	//signal(int, void(*)(int));//函数名称和函数参数类型
	//上述代码是一个函数的声明,
	//函数的名字是signal
	//函数的参数第一个是int类型,第二个是void(*)(int)类型的函数指针
	//该函数指针指向的函数参数是int,返回类型是void
	//signal函数的返回类型也是一个函数指针
	//该函数指针指向的函数参数是int,返回类型是void
	return 0;
}

可以通过类型重命名简化为:
//typedef int(*)(int) pt_f;//写法不对
typedef void(*pf_t)(int);//将void(*)(int)类型的函数指针重命名叫pf_t 

pf_t(signal(int, pf_t));

6.函数指针数组

存放函数指针的数组

函数指针数组写法

函数指针
int (*p)(int,int);

函数指针数组只需在p后加一个[大小],让指p先于[]结合
int (*p[10])(int,int);
//数组存放元素的类型就是 int (*) (int,int)

 函数指针数组的用途:转移表

例子:(计算器)

#include<stdio.h>

int Add(int x, int y)//加法函数
{
	return x + y;
}

int Sub(int x, int y)//减法函数
{
	return x - y;
}
int Mul(int x, int y)//乘法函数
{
	return x * y;
}
int Div(int x, int y)//除法函数
{
	return x / y;
}

void menu()//菜单函数
{
	printf("************************\n");
	printf("**** 1.add    2.sub ****\n");
	printf("**** 3.mul    4.div ****\n");
	printf("****     0.exit     ****\n");
}

int main()
{
	//转移表 - 函数指针数组   函数指针类型要相同
	int(*pfArr[5])(int, int) = { NULL,Add,Sub,Mul,Div };//NULL==0
	//						下标 0  1  2    3   4
	int input = 0;
	int x = 0;
	int y = 0;
	int ret = 0;
	do
	{
		menu();
		printf("请输入:> ");
		scanf("%d", &input);
		if (input == 0)
		{
			printf("退出计算器\n");
			break;
		}
		if (input <= 4 && input >= 1)
		{
			printf("请输入两个操作数:>");
			scanf("%d %d", &x, &y);
			printf("%d\n",pfArr[input](x, y));//调用相应的计算函数
		}
		else
		{
			printf("输入非法,请重新输入\n");
		}
		
	} while (input);
	return 0;
}

7.指向函数指针数组的指针

指向函数指针数组的指针是一个指针,指针指向一个数组,数组的元素都是函数指针

 示例:

#include<stdio.h>

int Add(int x, int y)
{
	return x + y;
}
int Sub(int x, int y)
{
	return x - y;
}

int main()
{
    //函数指针
	int(*pf)(int, int) = Add;
	//函数指针数组
	int(*pfArr[4])(int, int) = { Add,Sub };

	//ppfArr是一个指向一个函数指针数组的指针变量
	int(*(*ppfArr)[4])(int,int) = &pfArr;

	return 0;
}

8.回调函数

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的用于对该事件或条件进行响应。

这里演示一个用到回调函数的库函数,qsort 函数,包含头文件 stdlib.h 。

我们通过c++官方网站旧版上可以查到 qsort 函数的用法qsort用法

 先看返回值类型和参数列表

 它可以排序任何类型的数据,但是需要自己写一个函数用来可以确定两个元素的大小,也就是参数列表中的compar函数。

void qsort( void* base,//待排序数组的第一个元素
			size_t num, //待排序的元素个数
			size_t size, //每个元素的大小
			int(*cmp)(const void*, const void*));//函数指针,指向一个函数,这个函数可以比较2个元素的大小

qsort 函数底层使用的是快速排序,也是一种排序算法,我们这里用冒泡排序实现一下。

因为这个函数可以排序任何类型,所以我们最开始用  void * 指针来接收传来的指针,然后再通过强制类型转换 (char*) 再加上 size 来确定操作空间的大小。

我们先来实现以下cmp 函数

这是cmp函数的要求,返回值有大于0,小于0,和等于0,三种情况。 大于0指向p1,p1大,小于0指向p2,p2大,相等返回值是0。

int cmp_int(const void* p1, const void* p2)
{
	return *(int*)p1 - *(int*)p2;
    //这是我们自己写的函数,我们知道是int类型,所以强制类型转换为(int*)
}

交换函数:

因为我们不知道要交换什么类型元素,所以是使用 (char*) 强制类型转换,每次交换一个字节,循环size次的方法

void Swap(char* p1, char* p2, size_t size)
{
	for (size_t i = 0; i < size; i++)
	{
		char t = *(p1 + i);
		*(p1 + i) = *(p2 + i);
		*(p2 + i) = t;
	}
}

排序内部:

void bubble_sort(void* base, size_t num, size_t size, int(*cmp)(const void*, const void*))
{
	size_t i = 0;
	size_t j = 0;
	for (i = 0; i < num - 1; i++)//一趟冒泡排序
	{
		for (j = 0; j < num - 1 - i; j++)
		{
			//两个相邻元素比较
			//arr[j]与arr[j+1]  调用我们自己写的比较函数,回调函数
			if (cmp((char*)base + j * size, (char*)base + (j + 1) * size) > 0)
			{
				//交换
				Swap((char*)base + j * size, (char*)base + (j + 1) * size,size);
			}
		}
	}

全部代码:

#include<stdio.h>
#include<stdlib.h>

int cmp_int(const void* p1, const void* p2)
{
	return *(int*)p1 - *(int*)p2;
	//这是我们自己写的函数,我们知道是int类型,所以强制类型转换为(int*)
}

void Swap(char* p1, char* p2, size_t size)
{
	for (size_t i = 0; i < size; i++)
	{
		char t = *(p1 + i);
		*(p1 + i) = *(p2 + i);
		*(p2 + i) = t;
	}
}

void bubble_sort(void* base, size_t num, size_t size, int(*cmp)(const void*, const void*))
{
	size_t i = 0;
	size_t j = 0;
	for (i = 0; i < num - 1; i++)//一趟冒泡排序
	{
		for (j = 0; j < num - 1 - i; j++)
		{
			//两个相邻元素比较
			//arr[j]与arr[j+1]
			if (cmp((char*)base + j * size, (char*)base + (j + 1) * size) > 0)
			{
				//交换
				Swap((char*)base + j * size, (char*)base + (j + 1) * size, size);
			}
		}
	}
}

int main()
{
	int arr[10] = { 1,2,8,9,10,5,3,6,4,7 };

	qsort(arr, 10, sizeof(arr[0]), cmp_int);
	
	for(int i=0;i<10;i++)
	{
		printf("%d ", arr[i]);
	}
	printf("\n");

	int arr1[10] = { 1,2,8,9,10,5,3,6,4,7 };

	bubble_sort(arr1, 10, sizeof(arr[0]), cmp_int);

	for (int i = 0; i < 10; i++)
	{
		printf("%d ", arr[i]);
	}

	return 0;
}

本篇结束

 

 

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

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

相关文章

软件测试——功能测试,使用Java,IDEA,Selenium进行web自动化测试

视频地址&#xff1a;03-web元素定位ID_哔哩哔哩_bilibili p1.下载jdk,maven,idea p2.配置java-selenium环境正式开始&#xff1a; &#xff08;1&#xff09;创建代码&#xff1a; &#xff08;2&#xff09;第一次运行会报错&#xff1a;要下载东西 &#xff08;3&…

如何设计一个过压保护电路

有时候在电源输入处我们希望当电源的输入电压超过允许的最大值后电源与后级电路就自动断开&#xff0c;防止输入电压过高而损坏后级电路。 具有这种功能的电路叫做过压保护电路&#xff0c;英文简称叫OVP。 现在大家看到的就是一个典型的过压保护电压电路&#xff0c;主要包含…

(一)springboot实战——为什么是springboot?

前言 为什么是springboot&#xff1f;江湖夜雨&#xff0c;传说依旧&#xff0c;不懂springboot一技之长&#xff0c;如何混迹java圈&#xff0c;本节内容我们介绍一下spring的一些基本特性。尤其是springboot3的基本特性&#xff0c;使得我们更好的理解springboot3。 正文 …

【vue3】07-vue组件之间的通信-父子互传-事件总线

文章目录 Vue的组件嵌套Vue组件化-组件间通信父子组件之间通信非父子组件间的通信 Vue的组件嵌套 前面我们是将所有的逻辑放到一个App.vue中: 在之前的案例中&#xff0c;我们只是创建了一个组件App;如果我们一个应用程序将所有的逻辑都放在一个组件中&#xff0c;那么这个组…

linux下一次复制cp多个文件(含scp命令)

linux cp 参数说明 -a&#xff1a;此选项通常在复制目录时使用&#xff0c;它保留链接、文件属性&#xff0c;并复制目录下的所有内容。其作用等于dpR参数组合。 -d&#xff1a;复制时保留链接。这里所说的链接相当于Windows系统中的快捷方式。 -f&#xff1a;覆盖已经存在的目…

实训笔记6.5

实训笔记 6.5一、座右铭二、上周回顾Java基本语法1、Java的安装和环境变量的配置2、Java的标识符、关键字和保留字3、Java中的变量和常量、字面量4、数据类型5、运算符6、流程控制7、数组8、JVM内存图9、面向对象的两大概念-类和对象 三、类的组成3.1 属性的声明和使用3.1.1 属…

扩散模型之DDPM

扩散模型之DDPM 文章前置生成模型总结 Design of DDPM正向加噪过程反向去噪过程 文章前置 最原文链接&#xff08;英文&#xff09;&#xff1a;What are Diffusion Models? 原文链接&#xff1a;Diffusion扩散模型大白话讲解&#xff0c;看完还不懂&#xff1f;不可能 原文链…

第10章 对象和类

第10章 对象和类 10.1 过程性编程和面相对象编程10.2 抽象和类10.2.1 类型是什么10.2.2 C中的类10.2.3 实现类成员函数10.2.4 使用类 10.1 过程性编程和面相对象编程 采用OOP方法时&#xff0c;首先从用户的角度考虑对象——描述对象所需的数据以及描述用户与数据交互所需的操…

【蓝桥刷题】备战国赛——异或三角

蓝桥杯2021国赛真题——异或三角 &#x1f680; 每日一题&#xff0c;冲刺国赛 &#x1f680; 题目导航&#xff1a; 异或三角 &#x1f387;思路&#xff1a;数位 d p dp dp d f s dfs dfs 思维 &#x1f531;思路分析&#xff1a; ✅数理基础&#xff1a; 按位异或&am…

(2017,AdaIN)用自适应实例归一化进行实时的任意风格迁移

Arbitrary Style Transfer in Real-time with Adaptive Instance Normalization 公众号&#xff1a;EDPJ 目录 ​​​​​​​0. 摘要 1. 简介 2. 相关工作 3. 背景 3.1 批量归一化&#xff08;Batch Normalization&#xff0c;BN&#xff09; 3.2 实例归一化&#xff…

入驻京东直播间、成功借壳上市,交个朋友和时间“交朋友”

交个朋友一直走在“交朋友”的路上。 5月31日晚&#xff0c;交个朋友正式入驻京东直播间&#xff0c;首播销售额超1.5亿元&#xff0c;直播热度榜达人榜第一名&#xff0c;累计访问人次超1700万。据了解&#xff0c;京东618期间&#xff0c;除了日常直播外&#xff0c;交个朋友…

java并发编程:可见性、原子性、有序性三大特性详解

文章目录 可见性导致可见性的原因线程交叉执行重排序结合线程交叉执行共享变量更新后没有及时更新 如何解决可见性问题 原子性出现原子性问题的原因如何解决原子性问题 有序性导致有序性的原因如何解决有序性问题 总结 可见性 内存可见性&#xff0c;指的是线程之间的可见性&am…

IPv6NAT-PT实验:IPv4和IPv6地址转换的配置和验证

IPv6NAT-PT实验&#xff1a;IPv4和IPv6地址转换的配置和验证 【实验目的】 熟悉IPv6NAT-PT的概念。 掌握静态IPv6NAT-PT的配置 掌握动态IPv6NAT-PT的配置。 验证配置。 【实验拓扑】 设备参数如下表所示。 设备 接口 IP地址 子网掩码 默认网关 R1 S0/0 192.168.12…

2023HW护网红队必备工具总结

一、信息收集 1、AppInfoScanner 一款适用于以HVV行动/红队/渗透测试团队为场景的移动端(Android、iOS、WEB、H5、静态网站)信息收集扫描工具&#xff0c;可以帮助渗透测试工程师、红队成员快速收集到移动端或者静态WEB站点中关键的资产信息并提供基本的信息输出,如&#xff…

Java设计模式—模板方法模式

前言&#xff1a;模板方法模式是模板模式的一个具体实现&#xff0c;它定义了一个抽象类&#xff0c;其中包含一个模板方法和若干个基本方法。其中模板方法定义了算法骨架&#xff0c;而基本方法则由子类来实现。因此&#xff0c;模板方法在定义算法的结构方面提供了支持&#…

springMvc 解决 multipart/form-data 方式提交请求 不能获取非文件类型参数的问题和指定springmvc获取静态资源路径

问题&#xff1a; RequestMapping(value "/test",method RequestMethod.POST)ResponseBodypublic String test(String name,String id,MultipartFile file){System.out.println(name);System.out.println(id);System.out.println(file.getOriginalFilename());ret…

onnx模型的修改与调试demo

主要参考&#xff1a; 模型部署入门教程&#xff08;五&#xff09;&#xff1a;ONNX 模型的修改与调试 第五章&#xff1a;ONNX 模型的修改与调试 使用netron 可视化模型 读写onnx 构造onnx 创建一个描述线性函数 output axb 的onnx模型。 需要两个节点&#xff0c;第一个…

造轮子--自己封装一个 start

背景 都说不要造轮子&#xff0c;但是青柠认为&#xff0c;有的时候&#xff0c;造个轮子&#xff0c;更有助于清楚详细的业务逻辑&#xff0c;所以咱也自己写一个轮子&#xff0c;搞个系统开发。大体思路就是先搭建基础框架&#xff0c;然后细写业务逻辑&#xff0c;会涵盖主…

图解HTTP书籍学习2

确保Web安全的HTTPS HTTP的缺点 ●通信使用明文&#xff08;不加密&#xff09;&#xff0c;内容可能会被窃听 ●不验证通信方的身份&#xff0c;因此有可能遭遇伪装 ●无法证明报文的完整性&#xff0c;所以有可能已遭篡改 加密处理防止被窃听 通信的加密 一种方式就是…

【React】类组件,JSX语法,JSX原理,传递参数,条件渲染,列表渲染

❤️ Author&#xff1a; 老九 ☕️ 个人博客&#xff1a;老九的CSDN博客 &#x1f64f; 个人名言&#xff1a;不可控之事 乐观面对 &#x1f60d; 系列专栏&#xff1a; 文章目录 React使用组件&#xff08;类组件&#xff09;JSX语法书写规范JSX插入的内容JSX属性绑定JSX类绑…