C语言的回调函数(详解qsort函数)

news2025/1/12 20:06:51

在上一篇文章中我们学习了指针与数组之间的关系,学习了指针数组,数组指针变量,函数指针变量和函数指针数组。那么接下来我们要学习的是一个利用函数指针变量去解决和简化各种问题的回调函数。

一、回调函数

回调函数在编程中是一种很常见的概念,它是指将一个函数作为参数传递给另一个函数,并在特定的条件或事件发生时被调用执行的函数回调函数不是直接由函数的实现方所调用,而是在特定的条件或事件发生时,由接收它的函数间接的去调用它。它的作用就好比前段时间热映的影视剧狂飙中,高启强因为某些原因,自己没办法直接下手时,就告诉老默"我想吃鱼了",而老默作为一个接收信号的人,它接收到了高启强给他的信号,于是能够帮助高启强去间接的完成他要做的事。

光举例子肯定是不行,让我们实际操作写一段比较简单的代码,来试着用一下回调函数吧~

int Add(int x, int y)
{
	return x + y;
}
void Calc(int(*num)(int, int))
{
	int a = 0;
	int b = 0;
	printf("请输入两个计算数:>");
	scanf("%d %d", &a, &b);
	int m = num(a, b);
	printf("结果为%d\n", m);
}
int main() 
{
	Calc(Add);
	return 0;
}

在这段代码中,我们创造了一个void类型的函数Calc来接收一个"一个名为num,能够接收两个整形变量的函数"的函数指针作为参数当Calc调用Add函数时,那么此时Add函数就是一个回调函数。61f980161b9f4d5e925b68350d9b7c29.png

而此时可能有人就想发出疑问了:通常情况下我们会直接调用Add函数去计算整数之和,并且与这段代码相比,好像直接使用Add函数会更加快捷简便,那为什么还要使用回调函数呢?回调函数的作用是什么呢?在上一篇文章中,我们写过一个实现计算机的代码,最初的版本有非常多的冗余语句。d3de813c299b475aba4401113f73a462.png

而回调函数就能很好的帮我们规避掉这些冗余的句子,从而实现简化代码的作用。让我们来实际操作一下回调函数简化计算机吧~

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 Calc(int(*num)(int, int))
{
	int a = 0;
	int b = 0;
	printf("请输入两个计算数:>");
	scanf("%d %d", &a, &b);
	int m = num(a, b);
	printf("结果为%d\n", m);
}
int main() 
{
	int m;
	int n;
	int tmp = 1;
	do {
		printf("***********************\n");
		printf("**** 1.Add   2.Sub ****\n");
		printf("**** 3.Mul   4.Div ****\n");
		printf("******** 0.quit *******\n");
		printf("***********************\n");
		printf("请选择:>");
		scanf("%d", &tmp);
		switch (tmp)
		{
		case 1:Calc(Add);
			break;
		case 2:Calc(Sub);
			break;
		case 3:Calc(Mul);
			break;
		case 4:Calc(Div);
			break;
		case 0:printf("退出计算机");
			break;
		default:
			printf("输入错误,请重新输入\n");
			break;
		}
	} while (tmp);
	return 0;
}

这样就能够很明显的体现出回调函数的作用了,在一定情况下,它能够一定程度上简化我们的代码,并且解决一些解决不了的问题。

二、库函数strlen与strcmp

在讲sqort函数之前,我们需要先了解一下strlen函数和strcmp函数。它们是在一些对字符型数据进行计算时使用的,用处特别多,用法特别广泛的两个函数。

(两种函数的使用都需要加上头文件 <string.h>)

① strlen

521960f8cbb744b58e582cb5e4078d99.png

strlen的作用是:获取字符串长度,返回C字符串str的长度。

注:c字符串的长度由终止空字符确定,C字符串的长度等于字符串的开头和终止空字符之间的字符数(不包括终止空字符本身)。

比如我们定义了一个char str[100],但我们只为它初始化了11个字符,那么用int num = strlen(str)来获取字符串的长度时,就应该获取带11个字符。c0983826ee0d4f84a73d203358c009c7.png

② strcmp

212a2f6e2e504b7e86bab69ee1c110d5.png

strcmp的作用是:比较两个字符串,将C字符串 str1 与C字符组 str2进行比较。

比较方式为:该函数开始比较每个字符串的第一个字符。如果它们彼此相等,则继续进行以下配对,直到字符不同或达到终止空字符。此函数执行字符的二进制比较。

若str1小于str2,则返回负整数,即小于0的数。

若str1和str2相等,则返回0。

若str1大于str2,则返回正整数,即大于0的数。

我们还是来通过代码来进一步的了解一下它的作用效果:

#include <stdio.h>
#include <string.h>
int main() 
{
	char str1[] = "abcdef";
	char str2[] = "abcdefg";
	if (strcmp(str1, str2) > 0)
	{
		printf("str1比str2长\n");
	}
	else if (strcmp(str1, str2) < 0)
	{
		printf("str2比str1长\n");
	}
	else
	{
		printf("str1和str2一样长\n");
	}
	return 0;
}

7a5ac3b9c2e94602bc097522399b4bbe.png

这样就能够实现,通过strcmp来比较两个字符串的大小了。

三、qsort函数

通过上面我们对回调函数的解释,大家应该对回调函数的应用有了一定的了解。而说到回调函数的应用,这里我们就不得不提到qsort函数的应用了。

①qsort的参数及原理

qsort函数的作用就是对数据进行排序,比如现在有一组整型数组arr,里面有十个大小不同随意顺序的十个整形变量,那我们就能够通过qsort对它进行大小的排序。在学习qsort之前我们学习过冒泡排序,冒泡排序同样也能够对整型数组中的元素进行大小的排序:9c229f20f2bd441e81ac421af009c84a.png

这样就能够使数组中的元素大小排列了,但是仔细观察我们并不难发现,其实冒泡排序的缺点非常明显:

① 冒泡排序只适用于整型变量的排序。

② 时间复杂度O(n^2)太高,效率慢。

而这次我们要学习的qsort函数就能够完美的避免这两种问题,不仅能适用多种类型变量的排序,并且效率相较于冒泡排序也要大大提高。

qsort函数的格式:

void qsort (void* base, size_t num, size_t size,int (*compar)(const void*,const void*));

我们来分别了解一下qsort这四个参数:

①void *base:指向需要排序的数组的首元素,该对象转换为void*。

②size_t num:表示数组中元素的个数。

③size_t size:数组中每个元素的大小(以字节为单位)。size_t是无符号整型类型。

④int (*compar)(const void*,const void*):指向比较两个元素的函数。qsort调用这个函数来比较两个元素的大小。函数compar需要接收两个参数(都转换成const void*)并对两个参数进行比较,通过返回值来表示两个元素的大小关系。

Ⅰ— 如果返回值小于 0,表示 a 应该排在 b 的前面。

Ⅱ — 如果返回值等于 0,表示 a 和 b 相等,顺序不变。

Ⅲ — 如果返回值大于 0,表示 a 应该排在 b 的后面。

②qsort对整型数组排列

那么既然已经了解了qsort函数的使用方法,那让我们来利用qsort函数试试实现冒泡排序吧~

int compar(const void* a, const void* b)
{
	return (*(int*)a - *(int*)b);
}
int main() 
{
	int arr[] = { 7,5,3,9,8,4,6,2,1,0, };
	int sz = sizeof(arr) / sizeof(arr[0]);
	qsort(arr, sz, sizeof(arr[0]), compar);
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

因为数组名代表的就是数组首元素的地址,所以第一个参数直接传arr就行,并且在定义比较函数compar时,返回值的计算需要我们将传入的const void*强制类型转换成int*并解引用,从而就能够计算出数组中两个元素的大小关系了

a697d74bef3d414281f1c7bd8b224d2f.png按照这种写法输出的冒泡排序是升序,那么我们能不能把它变成降序呢?其实很简单,像冒泡排序一样修改大于和小于号一样,我们只需要将return (*(int*)a - *(int*)b)修改成return (*(int*)b - *(int*)a)就好了。b09e582334914b74b3a815b31daa4a90.png

那我们能不能再尝试一下用qsort来为字符型数组排序呢?让我们尝试一下吧~!!

③qsort对字符串排列

在使用qsort对字符型数组排序前,我们需要思考一个问题:整型变量能通过大于号,小于号来进行大小的表示,可是字符型不可以呀,那应该怎么表示两个元素的大小关系呢?嘿嘿~这就体现了我上面讲到的strcmp函数的作用了,strcmp可以用于比较字符之间的大小,并且大于返回>0的数,小于返回<0的数,等于返回0,和qsort中比较函数的返回值也是一样的形式,而获取字符串长度可以使用上面讲到的strlen函数,因为它不会把\0也算进长度。那思路有了,让我们尝试一番吧

int compar(const void* a, const void* b)
{
	return strcmp((char*)a, (char*)b);
}
int main()
{
	char arr[] = "ihfknac";
	int sz = strlen(arr);
	qsort(arr, sz, sizeof(arr[0]), compar);
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%c ", arr[i]);
	}
	return 0;
}

c230f599253f4f2ebc0ae5b875258c9b.png

这样就成功的把字符串也按顺序排列好啦~

④qsort对结构体排列

那么我们再来尝试一次,用字符串按顺序来对结构体进行排列~对结构体排列需要明确我们要排列的元素类型,比如我们现在创造一个结构体:

struct stu
{
	char name[20];
	int age;
};

如果模棱两可的,不知道想要比较哪一个数据,那当我们创造比较函数的时候就会陷入返回值的强制类型转换不知道该怎么写。所以我们要明确自己想要排列的数据才行。比如我们想排列结构体stu中的name,那我们就需要先定义一个结构体数组,然后在比较函数中使用strcmp比较大小(名字也是字符型),让我们动手试试吧:

struct stu
{
	char name[20];
	int age;
};
int compar(const void* a, const void* b)
{
	return strcmp(((struct stu*)a)->name, ((struct stu*)b)->name);
//也可以写作return strcmp((*((struct stu*)a)).name, (*((struct stu*)b)).name);
}
int main()
{
	struct stu s[3] = { {"xiaowang",23} ,{"xiaoli",18}, {"xiaozhang",29} };
	int sz = sizeof(s) / sizeof(s[0]);
	qsort(s, sz, sizeof(s[0]), compar);
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%s ", s[i].name);
	}
	return 0;
}

7d54362e9b6449d9bf52bdf20038cb84.png

前面的xiao都相同,从第五个字符开始w,l,z都不同,于是通过对w,l,z的大小比较来进行排列。由于l<w<z,最后结果则是xiaoli,xiaowang,xiaozhang。那我们再试着对age排序吧~

struct stu
{
	char name[20];
	int age;
};
int compar(const void* a, const void* b)
{
	return (((struct stu*)a)->age) - (((struct stu*)b)->age);
}
int main()
{
	struct stu s[3] = { {"xiaowang",23} ,{"xiaoli",18}, {"xiaozhang",29} };
	int sz = sizeof(s) / sizeof(s[0]);
	qsort(s, sz, sizeof(s[0]), compar);
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%d ", s[i].age);
	}
	return 0;
}

ed4a31288f9b452ca27cc2436da71d46.png

怎么样,是不是感觉以前很难实现的问题现在使用qsort就能非常轻松简便的解决啦~

四、构造并模拟实现qsort函数

其实我们学到刚刚也只是学会了如何使用qsort,如果想真的弄懂qsort的运作原理,不妨我们尝试一下,以冒泡排序的函数为基底,对冒泡排序函数一步步进行改造,将它变成一个qsort函数呢?

①BubbleSort的函数定义

对于自行构造出一个函数并使它实现qsort的功能,首先我们需要写出函数接收的参数,而参数并不需要什么多余思考,我们只需要借鉴qsort函数的参数就好了。

void BubbleSort (void* base, size_t num, size_t size,int (*compar)(const void*,const void*));

但是我们需要思考一下:为什么传进BubbleSort函数中的base要定义成void*类型呢?

这是因为想要实现为不同类型的数据进行排序,就要能够做到对各种类型的数据进行接收,如果我们在这里只是单一的写成int*,那么就无法对char*类型进行接收,如果写成char*,又无法对结构体类型进行接收,而void * 为"无类型指针",可以指向任意类型的数据,就是说可以用任意类型的指针对 void 指针赋值。

而后面函数指针中传参的const void*类型也是同样的道理。

②BubbleSort中实现元素互换

在BubbleSort()函数中,我们并无法确定接收到的会是什么类型的数据,因此我们无法像之前的整形冒泡排序一样,直接以一个整形变量tmp间接的转换元素。而我们如果直接使用某种类型来接收,就会导致我们无法接收其他的元素。而此时我们可以选择不去管它传进来的到底是什么类型(因为就算纠结,也还是不知道传进的是什么),就算我们不直接对值进行交换,别忘了我们还学过指针呢~而指针通常以字节为单位,虽然我们不知道传进来的是什么类型的参数,但是我们可以知道一个类型变量占多少个字节,而将两个元素的所有的字节都相互替换,也就能够做到互换两个元素啦~

void swap(char* p1, char* p2, int sz)//用char是因为char只有一个字节
{                                    //这样才能做到每次只跳过一个字节
	int i = 0;                       //从而做到交换每一个字节
	for (i = 0; i < sz; i++)
	{
		char tmp = *p1;
		*p1 = *p2;
		*p2 = tmp;
		p1++;
		p2++;
	}
}

这样就能实现利用指针,一个字节一个字节的交换,实现元素的交换~

③BubbleSort的函数结构体

对于BubbleSort函数的结构,我们就仿照冒泡排序的结构就可以,但我们仍然不清楚传进来的参数是什么类型,所以没办法直接进行对比,还是用字节来一个字节一个字节的进行比较,所以传进compar的两个参数也强制转换成char*类型,对之后的操作也都方便。

void BubbleSort(void* base, size_t num, size_t size, int (*compar)(const void* a, const void* b))
{
	int i = 0;
	int j = 0;
	for (i = 0; i < num - 1; i++)
	{
		for (j = 0; j < num - 1 - i; j++)
		{
			if (compar((char*)base + j * size, (char*)base + (j + 1) * size) > 0)
			{
				swap((char*)base + j * size, (char*)base + (j + 1) * size,size);
			}
		}
	}
}

这样我们就成功的使用BubbleSort函数构造并模拟实现qsort函数的功能啦~!费尽周折终于算是把qsort模拟出来了,那么让我们尝试一下,用我们自己构造的BubbleSort给整型数组排序一下试试看吧~

int compar(const void* a, const void* b)
{
	return *(int*)a - *(int*)b;
}
void swap(char* p1, char* p2, int sz)
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		char tmp = *p1;
		*p1 = *p2;
		*p2 = tmp;
		p1++;
		p2++;
	}
}
void BubbleSort(void* base, size_t num, size_t size, int (*compar)(const void* a, const void* b))
{
	int i = 0;
	int j = 0;
	for (i = 0; i < num - 1; i++)
	{
		for (j = 0; j < num - 1 - i; j++)
		{
			if (compar((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[] = { 7,8,4,5,1,6,2,9,3,0 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	BubbleSort(arr, sz, sizeof(arr[0]), compar);
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

61643996610044d8b748ee094c9be573.png

结果是没有问题的呢,那么我们再尝试一下给字符串儿排序。

int compar(const void* a, const void* b)
{
	return strcmp((char*)a, (char*)b);
}
void swap(char* p1, char* p2, int sz)
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		char tmp = *p1;
		*p1 = *p2;
		*p2 = tmp;
		p1++;
		p2++;
	}
}
void BubbleSort(void* base, size_t num, size_t size, int (*compar)(const void* a, const void* b))
{
	int i = 0;
	int j = 0;
	for (i = 0; i < num - 1; i++)
	{
		for (j = 0; j < num - 1 - i; j++)
		{
			if (compar((char*)base + j * size, (char*)base + (j + 1) * size) > 0)
			{
				swap((char*)base + j * size, (char*)base + (j + 1) * size,size);
			}
		}
	}
}
int main()
{
	char arr[] = "hdngijedg";
	int sz = sizeof(arr) / sizeof(arr[0]);
	BubbleSort(arr, sz, sizeof(arr[0]), compar);
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%c ", arr[i]);
	}
	return 0;
}

99b31a93eed34598bfd7862589823b21.png依然是没有问题的,那么我们再尝试一次性把结构体的name和age都排序输出吧!

struct stu
{
	char name[20];
	int age;
};
int compar1(const void* a, const void* b)
{
	return ((struct stu*)a)->age - ((struct stu*)b)->age;
}
int compar2(const void* a, const void* b)
{
	return strcmp(((struct stu*)a)->name,((struct stu*)b)->name);
}
void swap(char* p1, char* p2, int sz)
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		char tmp = *p1;
		*p1 = *p2;
		*p2 = tmp;
		p1++;
		p2++;
	}
}
void BubbleSort(void* base, size_t num, size_t size, int (*compar)(const void* a, const void* b))
{
	int i = 0;
	int j = 0;
	for (i = 0; i < num - 1; i++)
	{
		for (j = 0; j < num - 1 - i; j++)
		{
			if (compar((char*)base + j * size, (char*)base + (j + 1) * size) > 0)
			{
				swap((char*)base + j * size, (char*)base + (j + 1) * size,size);
			}
		}
	}
}
int main()
{
	struct stu s[3] = { {"xiaowang",19},{"xiaozhang",28},{"xiaoli",12}};
	int sz = sizeof(s) / sizeof(s[0]);
	BubbleSort(s, sz, sizeof(s[0]), compar1);
	BubbleSort(s, sz, sizeof(s[0]), compar2);
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%s ", s[i].name);
	}
	printf("\n");
	for (i = 0; i < sz; i++)
	{
		printf("%d ", s[i].age);
	}
	return 0;
}

9bb7e4f88a724cc1846a9d4b41524003.png

也是运行成功啦~这样看来我们的构造并模拟实现qsort函数就大功告成啦,那今天关于回调函数的知识分享就到这里啦,我们下次再见哦~

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

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

相关文章

pip install -r requirements.txt下载某个.whl文件较慢的问题

访问PYPI官网。搜索该文件名字 如果不是自己要的版本&#xff0c;点击左边的Release history。 选择版本后再次在Download files页面下载.whl文件 下载完之后切换至自己的项目解释器虚拟环境。 python -m pip install 文件名.whl 如果你改名python.exe为python3.exe&#xff…

【消息队列】RabbitMQ 面试篇

&#x1f397;️ 主页&#xff1a;小夜时雨 &#x1f397;️专栏&#xff1a;消息队列 &#x1f397;️如何活着&#xff0c;是我找寻的方向 目录 1. RabbitMQ 有哪些工作模式&#xff1f;2. 如何保证消息不丢失&#xff08;可靠性&#xff09;3. 什么是死信队列&#xff0c;如…

基于K210智能人脸识别+车牌识别系统(完整工程资料源码)

运行效果&#xff1a; 基于K210的智能人脸与车牌识别系统工程 目录&#xff1a; 运行效果&#xff1a; 目录&#xff1a; 前言&#xff1a; 一、国内外研究现状与发展趋势 二、相关技术基础 2.1 人脸识别技术 2.2 车牌识别技术 三、智能小区门禁系统设计 3.1 系统设计方案 3.2 …

【网络编程】网络原理(一)

系列文章目录 1、 初识网络 2、网络编程的基础使用&#xff08;一&#xff09; 文章目录 系列文章目录前言一、端口号的使用二、UDP报文学习1.报文格式2.MD5算法 总结 前言 在前文中&#xff0c;主要对UDP和TCP协议有了简单的了解&#xff0c;而这两种协议是负责传输层的内容…

组件设计原则

state数据结构设计 用数据描述所有内容数据要结构化&#xff0c;易于程序操作&#xff08;遍历、查找&#xff09;数据要可扩展&#xff0c;以便增加新的功能 组件设计组件通讯 从功能上拆分层次尽量让组件原子化容器组件&#xff08;只管理数据&#xff09;& UI组件&am…

【MySQL】库操作,数据类型

目录 MySQL简介SQL语句分类库操作语句展示数据库创建数据库使用数据库删除数据库 数据类型整型浮点型字符串日期类型 MySQL简介 数据库有关系型数据库和非关系型数据库。 关系型数据库&#xff1a;是指采用了关系模型来组织数据的数据库。 简单来说&#xff0c;关系模型指的就…

科普文:JUC系列之ConcurrentLinkedQueue非阻塞队列用法

概叙 **Queue接口与List、Set同一级别&#xff0c;都是继承了Collection接口**。队列是一种数据结构&#xff0e;它有两个基本操作&#xff1a;在队列尾部加人一个元素&#xff0c;和从队列头部移除一个元素&#xff0c;队列以一种先进先出的方式管理数据。 队列分为两种&#…

pikachu漏洞平台~文件包含漏洞

在PHP程序中使用文件包含的对象可以被前端的用户控制且没有经过过滤或严格的定义&#xff0c;用户可以将其他的文件作为参数带入到PHP代码中解释执行&#xff0c;从而造成敏感信息泄露/程序文件读取/GetShell等危害的漏洞。 0x01文件包含漏洞 连接成功&#xff0c;完毕&#x…

聚焦IOC容器刷新环节postProcessBeanFactory(BeanFactory后置处理)专项

目录 一、IOC容器的刷新环节快速回顾 二、postProcessBeanFactory源码展示分析 &#xff08;一&#xff09;模版方法postProcessBeanFactory &#xff08;二&#xff09;AnnotationConfigServletWebServerApplicationContext 调用父类的 postProcessBeanFactory 包扫描 …

oracle(19c)用户管理

简介 本文介绍 Oracle 中的用户管理&#xff0c;包含以下内容&#xff1a; 概念介绍 系统用户 解锁 hr 用户 创建用户 用户相关案例 使用 Profile 管理用户口令 Oracle 的认证方式 重置管理员(sys)密码 1. 概念介绍 使用前可以自行安装oracle数据库 oracle19c安装&a…

【系统架构设计师】二十四、安全架构设计理论与实践④

目录 六、数据库系统的安全设计 6.1 数据库的完整性设计 6.1.1 数据库完整性设计原则 6.1.2 数据库完整性的作用 6.1.3 数据库完整性设计示例 七、系统架构的脆弱性分析 7.1 软件脆弱性的特点和分类 7.2 软件脆弱性的生命周期 7.2.1 脆弱性的引入阶段 7.2.2 产生破坏…

pythonflaskMYSQL自驾游搜索系统32127-计算机毕业设计项目选题推荐(附源码)

目 录 摘要 1 绪论 1.1研究背景 1.2爬虫技术 1.3flask框架介绍 2 1.4论文结构与章节安排 3 2 自驾游搜索系统分析 4 2.1 可行性分析 4 2.2 系统流程分析 4 2.2.1数据增加流程 5 2.3.2数据修改流程 5 2.3.3数据删除流程 5 2.3 系统功能分析 5 2.3.1 功能性分析 6 2.3.2 非功…

C++初阶学习——探索STL奥秘——模拟实现string类

1、string类的构造 上面的代码从表面看没什么问题&#xff0c;但是运行后会发现程序有多处bug 但是如上图一样&#xff0c;这样改进依然有bug 因为我们编写无参构造函数的时候&#xff0c;肯定要让_str默认为nullptr&#xff0c;但是这样的话&#xff0c;在main函数中创建对象…

使用npm全局安装typescript

查看npm安装 npm -v 安装typescript npm i -g typescript 查看安装 tsc 这就是标致着安装完成。

uBlock Origin很快将无法在Chrome上使用 开发者发布情况说明

Chrome v127 版开始扩展程序页面将自动显示即将不再支持的扩展程序&#xff0c;包括知名的广告拦截扩展程序 uBlock Origin 也在谷歌的警告列表中。昨天 uBO 团队发布新的支持文档对目前的情况进行说明&#xff0c;简单来说就是 Chrome 将不再支持基于 Manifest v2 开发的扩展程…

【设计模式入门】设计模式全解析:23种经典模式介绍与评级指南(设计师必备)

文章目录 设计模式简介引言七项基本原则创建型模式单例模式&#xff08;Singleton&#xff09;工厂方法模式&#xff08;Factory Method&#xff09;抽象工厂模式&#xff08;Abstract Factory&#xff09;建造者模式&#xff08;Builder&#xff09;原型模式&#xff08;Proto…

文件夹图标变白色无法打开:高效数据恢复指南

在日常使用电脑的过程中&#xff0c;我们可能会遇到一种令人困扰的情况——文件夹图标突然变成白色且无法正常打开。这一现象不仅影响了文件管理的便捷性&#xff0c;还可能意味着重要数据的丢失风险。本文将深入探讨这一现象的原因&#xff0c;并提供一种专业且高效的数据恢复…

一二三应用开发平台应用开发示例(11)——收藏夹功能高代码改造及总结

背景 前面使用低代码配置&#xff0c;把实体配置、库表和模式化的代码生成出来了&#xff0c;实际上是用平台帮忙开发人员把“体力活”给干了。接下来&#xff0c;就需要在此基础上进行个性化逻辑的开发。 文档收藏 完善实体模型 文档收藏夹我们原先配置了三个关键属性&…

【Mind+】掌控板入门教程04 迷你动画片

还记得小时候每天放学必看的动画片吗&#xff1f;还记得那些年陪伴我一起长大的卡通人物吗&#xff1f;勇救爷爷的葫芦娃&#xff0c;我们的朋友小哪吒&#xff0c;相信这些经典的动画形象已经成为了一代人童年的美好回忆。今天就让我们用掌控板来制作一部迷你动画片吧。 项目示…

stm32入门学习10-I2C和陀螺仪模块

&#xff08;一&#xff09;I2C通信 &#xff08;1&#xff09;通信方式 I2C是一种同步半双工的通信方式&#xff0c;同步指的是通信双方时钟为一个时钟&#xff0c;半双工指的是在同一时间只能进行接收数据或发送数据&#xff0c;其有一条时钟线&#xff08;SCL&#xff09;…