C语言_指针_进阶
引言:在前面的c语言_指针初阶上,我们了解了简单的指针类型以及使用,下面我们将进入更深层次的指针学习,对指针的理解会有一个极大的提升。从此以后,指针将不再是难点,而是学习底层语言的一把利器。
本章重点
- 字符指针
- 数组指针
- 指针数组
- 数组传参和指针传参
- 函数指针
- 函数指针数组
- 指向函数指针数组的指针
- 回调函数
- 指针和数组面试题的解析
指针的主题,我们在初级阶段的《指针》章节已经接触过了,我们知道了指针的概念:
指针就是个变量,用来存放地址,地址唯一标识一块内存空间。
指针的大小是固定的4/8个字节(32位平台/64位平台)。
指针是有类型,指针的类型决定了指针的±整数的步长,指针解引用操作的时候的权限。
指针的运算。
这个章节,我们继续探讨指针的高级主题。
1. 字符指针
在指针的类型中我们知道有一种指针类型为字符指针char*
;
一般使用:
int main()
{char ch = 'w';char *pc = &ch;*pc = 'w';return 0;
}
还有一种使用方式如下:
int main()
{char* pstr = "hello bit.";//这里是把一个字符串放到pstr指针变量里了吗?printf("%s\n", pstr);return 0;
}
代码char* pstr = "hello bit."
;特别容易让同学以为是把字符串hello bit
放到字符指针了,但是本质是把字符串pstr 里hello bit
,首字符的地址放到了pstr中。
类似于int arr[3] = {1, 2, 3}; int* parr = arr。我们知道数组名就是首元素地址,那么将数组名当作地址存入parr的指针变量中。
我们可以看到结果确实是把字符串的首元素地址存储在了parr指针变量中。
一道面试题:
输出结果:
为什么会产生出这种结果?
通俗易懂来说,str1以及str2它们各自开辟了一块空间,然后都拿了相同的字符串进行存储,那么即便值相同,地址却是不相同的。
str3和str4都指向了一块常量,我们知道常量是不可以被修改的,所以也没必要产生出两个指针指向同一个常量,由此可得,str3 == str4。
所以这里的区别就是:一个是存储值,一个是指向常量。
当我们往下面学习指针数组和数组指针的时候,就会产生一种没来由的困惑,读起来都有点绕口。其实辨别起来很简单:只需要去看最后两个字是什么。
指针数组:数组,装有指针元素的数组
数组指针:指针,指向数组的指针
2. 指针数组
在C语言_指针_初阶章节,我们也学了指针数组,指针数组是一个存放指针的数组。
这里我们再复习一下,下面指针数组是什么意思?
int* arr1[10]; //整形指针的数组
char *arr2[4]; //一级字符指针的数组
char **arr3[5];//二级字符指针的数组
通过指针数组模拟实现二维数组
3. 数组指针
3.1 数组指针的定义
数组指针是指针?还是数组?
答案是:指针。
我们已经熟悉了:整型指针:int* ptr
;能够指向整形数据的指针。浮点型指针:float* ptr
;能够指向浮点型数据的指针。
那数组指针也就很明确了:指向数组的指针。
下面代码哪个是数组指针?
int *p1[10];
//p1先跟[]结合,说明他是数组,[]里面是10,说明装有10个整型指针的元素。
int (*p2)[10];//数组指针//p1, p2分别是什么?
画图详解
解释:
首先,p和先做结合,也就是( * p),说明p是指针变量,然后外面还剩下int [10],这个我们之前见过,也就是装有10个整型元素的数组。那么好像就能合起来解释了,也就是指向
装有10个整型元素的数组
的指针。
总结:p是指针,指向一个数组,也就是数组指针。
这里需要注意的是:[]的优先级是要高于 * 号的,所以必须加上()来保证p先和结合。
3.2 &数组名 VS 数组名
对于下面的数组:
int arr[10];
arr
和 &arr
分别是啥?
我们知道arr是数组名,数组名表示数组首元素的地址。
那&arr数组名到底是啥?
我们看一段代码:
#include <stdio.h>
int main()
{int arr[10] = {0};printf("%p\n", arr);printf("%p\n", &arr);return 0;
}
输出结果
可见数组名和&数组名打印的地址是一样的。
难道两个是一样的吗?
我们再看一段代码:
#include <stdio.h>
int main()
{int arr[10] = { 0 };printf("arr = %p\n", arr);printf("&arr= %p\n", &arr);printf("arr+1 = %p\n", arr+1);printf("&arr+1= %p\n", &arr+1);return 0;
}
输出结果
根据上面的代码我们发现,其实&arr和arr,虽然值是一样的,但是意义应该是不一样的。
实际上:&arr 表示的是数组的地址,而不是数组首元素的地址。(细细体会一下)
数组的地址+1,跳过整个数组的大小,所以&arr+1 相对于 &arr 的差值是40。
从上面可知,数组名就是首元素的地址,那么sizeof(数组名),难道取的是指针变量的字节嘛?
由此可得,
sizeof(数组名)
并不是取得首元素地址的字节;这里的数组名代表了整个数组,也就是取得是整个数组的地址。
我们可以看到虽然表达出来的结果完全一样,但内在的意义确实完全不相同。
总结:
数组名大部分情况下都是首元素地址,但有两个例外,如下:
1.sizeof(数组名),这里面的数组名代表的是整个数组,所以取出来的字节也就是整个数组的字节。
2.&数组名,这里的数组名代表的也是整个数组,那么取出来的地址也就是整个数组的地址。
3.3 数组指针的使用
那数组指针是怎么使用的呢?
既然数组指针指向的是数组,那数组指针中存放的应该是数组的地址。
下面我们通过数组指针完成对数组内的元素进行一个遍历。
画图详解
上面我们讲述了一维数组取地址变成数组指针的使用方式,下面我们来讲述二维数组取地址变成数组指针的使用方式,并且会画图详细的描述整个流程。
画图详解 -> 分批次详细解读针对二维数组取地址变成数组指对内部的值进行访问的一个过程
4. 数组参数、指针参数
4.1 一维数组传参
#include <stdio.h>
void test(int arr[])//ok?
//实参传递了数组名,形参就可以用数组类型进行接受
{}
void test(int arr[10])//ok?
//完全没问题
{}
void test(int *arr)//ok?
//指向int类型元素的地址,也就是指向首元素的地址。
{}
void test2(int *arr[20])//ok?
//完全没问题
{}
void test2(int **arr)//ok?
//(*arr)是一个指针,指向了一个int*类型的元素。
{}int main()
{int arr[10] = {0};int *arr2[20] = {0};test(arr);test2(arr2);
}
4.2 二维数组传参
void test(int arr[3][5])//ok?
//完全没问题
{}
void test(int arr[][])//ok?
//不能省略一维数组中存储元素的个数。
{}
void test(int arr[][5])//ok?
//正确,可以省略二维数组的一维元素个数。
{}
//总结:二维数组传参,函数形参的设计只能省略第一个[]的数字。
//因为对一个二维数组,可以不知道有多少行,但是必须知道一行多少元素。
//这样才方便运算。
void test(int *arr)//ok?
//错误,形参的意思是指向整型元素的地址,跟题意完全不同。
{}
void test(int* arr[5])//ok?
//错误,形参是指针数组的意思,跟题意完全不同。
{}
void test(int (*arr)[5])//ok?
//正确,(*arr)是指针,指向了一个装有五个整型元素的数组。
{}
void test(int **arr)//ok?
//错误,首元素地址代表的是一维数组的整个数组地址。
{}int main()
{int arr[3][5] = {0};test(arr);
}
一级指针传参
#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};int *p = arr;int sz = sizeof(arr)/sizeof(arr[0]);//一级指针p,传给函数print(p, sz);return 0;
}
当一个函数的参数部分为一级指针的时候,函数能接收什么参数?
void test(char* ptr)
{}int main()
{char arr[10] = "abcdef";char ch = 'a';char* ptr = arr;test(arr);test(&ch);test(ptr);return 0;
}
二级指针传参
#include <stdio.h>
void test(int** ptr)
{printf("num = %d\n", **ptr);
}int main()
{int n = 10;int*p = &n;int **pp = &p;int* arr[5];test(pp);test(&p);test(arr);return 0;
}
5. 函数指针
首先,我们看最后两位是指针,那么他其实就是指向函数的指针。
输出的是两个地址,这两个地址是test
函数的地址。 那我们的函数的地址要想保存起来,怎么保存?
下面我们看代码:
void test(char* pc, int arr[10])
{}int main()
{void (*ptr)(char*, int [10]) = &test;//&test跟test都能表达地址,可以省略&。return 0;
}
首先*和ptr先结合,说明它是指针,指向的是一个函数,里面有char*的类型和int [10] 类型,这个函数的类型是void类型。
总结:指向了一个void类型函数里面参数有char*类型以及int[10]类型
的指针。
那么,我们既然知道了函数指针怎么创建,那下面我们就继续讨论,怎么使用函数指针。
看起来我们好像知道了该怎么使用函数指针进行传参了,上面我们已经了解了
pf
就是函数地址,但是,这里的(*pf)并不是对pf做一个解引用的访问,在函数指针中,函数名 == &函数名
,也就是说这里的*号仅仅起到装饰的作用,并不能跟解引用访问串联起来。
聪明的朋友已经看出来了,add和&add都是代表着函数地址的意思,那么pf好像等价于add,那有没有可能pf(参数1,参数2)
也可以进行函数调用呢?
看起来跟我们的想法是正确的。
阅读两段有趣的代码:
(*(void (*)())0)();
画图详解
void (*signal(int , void(*)(int)))(int);
画图详解
在做一个类似的题目再次进行巩固所学的知识
6. 函数指针数组
数组是一个存放相同类型数据的存储空间,那我们已经学习了指针数组, 比如:
int *arr[10];
//数组的每个元素是int*
那要把函数的地址存到一个数组中,那这个数组就叫函数指针数组,那函数指针的数组如何定义呢?
int (*parr1[10]])();
parr1先和[]结合,说明它是一个数组,数组里面装有10个int(*)()函数类型元素。
那么我们从这里就明白了,函数指针数组的写法,那么我们开拓一下思路,是不是也就知道了数组指针数组的写法?
int arr_1[5] = { 1, 2, 3, 4, 5 };
int arr_2[5] = { 1, 2, 3, 4, 5 };
int arr_3[5] = { 1, 2, 3, 4, 5 };
int (*parr[3])[5] = {&arr_1, &arr_2, &arr_3};
parr先和[3]结合,说明parr是一个数组,里面装有三个
指向装有5个int类型元素的数组的指针
听着可能有些绕口,我们把这段话再次拆开来细细分析。
我们就先看里面的内容,简单来说是不是就是指向数组的指针?然后数组里面有三个元素都是这样的类型。
函数指针数组的用途:转移表
例子:(计算器)
//加法
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");printf("****************************\n");
}int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{menu();printf("请选择:");scanf("%d", &input);switch (input){case 1:printf("请输入两个数字:");scanf("%d %d", &x, &y);ret = add(x, y);printf("ret = %d\n", ret);break;case 2:printf("请输入两个数字:");scanf("%d %d", &x, &y);ret = sub(x, y);printf("ret = %d\n", ret);break;case 3:printf("请输入两个数字:");scanf("%d %d", &x, &y);ret = mul(x, y);printf("ret = %d\n", ret);break;case 4:printf("请输入两个数字:");scanf("%d %d", &x, &y);ret = div(x, y);printf("ret = %d\n", ret);break;case 0:printf("退出计算器!!!\n");break;default:printf("输入错误,请重新选择!!!\n");break;}} while (input);return 0;
}
虽然实现了计算器项目,但是可以看到代码中出现了大量重复的代码,显得十分冗余。
所以,衍生出了下面的这个方式,使用函数指针数组实现计算器:
//加法
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");printf("****************************\n");
}int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;//函数指针数组int (*PfArr[5])(int, int) = {NULL, add, sub, mul, div};do{menu();printf("请选择:");scanf("%d", &input);if (input >= 1 && input <= 4){printf("请输入两个数字:");scanf("%d %d", &x, &y);//通过函数指针数组调用函数 - 转移表//通过函数指针数组找到对应下标然后跳到该函数位置得到结果再次返回来。int ret = (*PfArr[input])(x, y);printf("ret = %d\n", ret);}else if (input == 0){printf("退出计算器!!!\n");}else{printf("输入有误,请重新检查!!!\n");}} while (input);return 0;
}
可以明显的感受到代码进行了简化。
7. 指向函数指针数组的指针
其实很好理解,指针指向了一个数组,数组里面的元素都是函数指针。
void test(const char* str)
{printf("%s\n", str);
}int main()
{//函数指针void(*pf)(const char*) = &test;//函数指针数组void(*pf[5])(const char*);//指向函数指针数组的指针void(*(*pf)[5])(const char*);//再次拓展//指向函数指针数组的指针的数组void(*(*pf[5])[5])(const char*);return 0;
}
8. 回调函数
回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。
画图详解:
那我们知道了回调函数的概念,通过回调函数来继续实现计算器项目
//加法
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");printf("****************************\n");
}//在特定条件下通过传递过来的地址作为媒介去调用函数
void Cacl(int(*pf)(int, int))
{int x = 0;int y = 0;printf("请输入两个数字:");scanf("%d %d", &x, &y);int ret = (*pf)(x, y);printf("ret = %d\n", ret);
}int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{menu();printf("请选择:");scanf("%d", &input);switch (input){case 1:Cacl(add);break;case 2:Cacl(sub);break;case 3:Cacl(mul);break;case 4:Cacl(div);break;case 0:printf("退出计算器!!!\n");break;default:printf("输入有误,请重新输入!!!\n");break;}} while (input);return 0;
}
通过回调函数的方式完成计算器项目,代码也是得到了很大的简化。
下面在讲qsort之前,我们先了解一下冒泡排序的思路。
其实冒泡排序的核心思路不难,就是通过相邻两个数字进行对比,不断的将一个最大值或者最小值挪到最右边,以上仅仅是一趟,我们要通过多躺实现最终的效果。
我们将冒泡排序再次复习一遍。
//冒泡排序
void Bubble_sort(int* arr, int sz)
{int i = 0;//趟数for (i = 0; i < sz - 1; i++){int j = 0;//一趟比较的次数for (j = 0; j < sz - 1 - i; j++){//相邻两个数字进行比对if (arr[j] < arr[j + 1]){//三个空杯子原理进行交换int tmp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = tmp;}}}
}//输出排序结果
void Printf(int* arr, int sz)
{int i = 0;for (i = 0; i < sz; i++){printf("%d ", arr[i]);}return 0;
}int main()
{int arr[10] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };int sz = sizeof(arr) / sizeof(arr[0]);Bubble_sort(arr, sz);Printf(arr, sz);return 0;
}
输出结果:
首先演示一下qsort函数的使用:
#include <stdlib.h>
#include <string.h>//void*指针 - 无具体类型指针
// void*指针可以接受任意类型的指针
//void*类型指针不能直接进行解引用访问
//也不能直接用来进行指针运算//整型数组通过整型元素比较
int cmp_int(const void* p1, const void* p2)
{return *(int*)p2 - *(int*)p1;
}//输出结果
void Printf(int* arr, int sz)
{int i = 0;for (i = 0; i < sz; i++){printf("%d ", arr[i]);}
}//测试qsort排序整型
void test1()
{int arr[10] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(int), cmp_int);Printf(arr, sz);
}//测试qsort排序结构体
typedef struct Stu Stu;
struct Stu
{char name[20];int age;
};//结构体通过年龄比较
int cmp_stu_by_age(const void* p1, const void* p2)
{return (((Stu*)p1)->age - ((Stu*)p2)->age);
}//结构体通过名字比较
int cmp_stu_by_name(const void* p1, const void* p2)
{//名字不能直接相减,需要通过专门的字符比较函数 - strcmpreturn strcmp(((Stu*)p1)->name, ((Stu*)p1)->name);
}void test2()
{Stu arr[3] = { {"XingC", 22}, {"qmx_07", 20}, {"Yuu", 21}};int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(arr[0]), cmp_stu_by_age);printf("%d\n", arr[2].age);
}void test3()
{Stu arr[3] = { {"XingC", 22}, {"qmx_07", 20}, {"Yuu", 21} };int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(arr[0]), cmp_stu_by_name);int i = 0;for (i = 0; i < 3; i++){printf("name = %s\tage = %d\n", arr[i].name, arr[i].age);}
}int main()
{test1();test2();test3();return 0;
}
使用回调函数,模拟实现qsort(采用冒泡的方式)。
问题一:传统冒泡排序我们会用整型数组类型接收,那么我们想模拟实现快排,那就不能固定使用整型数组类型
问题二:传统冒泡排序比较使用大小符号比较的,那么结构体如何使用大小符号比较呢?
问题三:传统冒泡排序符合条件进行交换是通过三个杯子交换实现的,但是不同的数据交换略有差异。
//交换
//因为我们不知道我们要交换的类型是什么,所以以最小字节来进行一一交换最合适不过
//我们还是需要知道这个类型的字节多少,所以传入了一个sz
void Swap(char* buf1, char* buf2, int sz)
{int i = 0;//我们知道字节大小的情况下,循环交换两数的每个字节for (i = 0; i < sz; i++){char tmp = *(buf1 + i);*(buf1 + i) = *(buf2 + i);*(buf2 + i) = tmp;}
}void Bubble_sort(const void* base, int num, int sz, int (*cmp)(const void*, const void*))
{int i = 0;for (i = 0; i < num - 1; i++){int j = 0;for (j = 0; j < num - 1 - i; j++){//通过我们自己写的比较函数来判断是否符合条件if ((*cmp)((char*)base + j * sz, (char*)base + (j + 1) * sz) > 0){//交换Swap((char*)base + j * sz, (char*)base + (j + 1) * sz, sz);}}}
}//定义结构体
typedef struct Stu Stu;
struct Stu
{char name[20];int age;
};//结构体年龄成员比较方式
int cmp_stu_by_age(const void* p1, const void* p2)
{return (((Stu*)p1)->age - ((Stu*)p2)->age);
}//结构体名字成员比较方式
int cmp_stu_by_name(const void* p1, const void* p2)
{return strcmp(((Stu*)p1)->name, ((Stu*)p2)->name);
}//测试Bubble_sort 排序结构体数据
void test_2()
{Stu arr[3] = { {"XingC", 22}, {"qmx_07", 20}, {"Yuu", 21} };int sz = sizeof(arr) / sizeof(arr[0]);Bubble_sort(arr, sz, sizeof(arr[0]), cmp_stu_by_name);int i = 0;for (i = 0; i < sz; i++){printf("name = %s\tage = %d\n", arr[i].name, arr[i].age);}
}//整型比较方式
int cmp_int(const void* p1, const void* p2)
{return *(int*)p1 - *(int*)p2;
}//测试Bubble_sort 排序整型数据
void test_1()
{int arr[10] = {5, 7, 2, 3, 9, 1, 6, 8, 4, 10};int sz = sizeof(arr) / sizeof(arr[0]);//cmp_int是函数名;实参传入函数名,形参接收函数的类型Bubble_sort(arr, sz, sizeof(arr[0]), cmp_int);int i = 0;for (i = 0; i < sz; i++){printf("%d ", arr[i]);}
}int main()
{/*test_1();*/test_2();return 0;
}
9. 指针和数组笔试题解析
画图详解
对于指针来说,类型只是一个环节,最重要的是你要知道指针指向哪里。
总结:
- sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小。
- &数组名,这里的数组名表示整个数组,取出的是整个数组的地址。
- 除此之外所有的数组名都表示首元素的地址。
10. 指针笔试题
输出结果:
总结:
-
当我们能学到这里的时候,想必已经对指针了解的比较深入了,纵观整个指针的学习过程中,其实一直绕不开的就是指针类型,最重要的也就是指针类型,我们一直围绕的也就是指针类型。
-
其实指针就是一个指向或者是一个地址,当不需要进行任何操作的时候,就可以用
void*
替代,但涉及到任何运算以及解引用操作权限都需要指针类型。 -
当我们能彻底搞清楚我们指针的类型的时候,那么也就对解引用操作以及运算移动操作都将非常清晰,让我们在底层的访问中无往不利。
本章完~
相关文章:

C语言_指针_进阶
引言:在前面的c语言_指针初阶上,我们了解了简单的指针类型以及使用,下面我们将进入更深层次的指针学习,对指针的理解会有一个极大的提升。从此以后,指针将不再是难点,而是学习底层语言的一把利器。 本章重点…...

chat_gpt回答:python使用writearray写tiff速度太慢,有什么快速的方法吗
如果你在使用 Python 的 tifffile 库(或类似库)写入 TIFF 文件时速度太慢,以下是几个加速写入的优化方法和替代方案: 1. 优化文件压缩设置 TIFF 支持压缩格式,但压缩过程可能非常耗时。如果你不需要压缩,…...

【时时三省】(C语言基础)函数介绍strcat
山不在高,有仙则名。水不在深,有龙则灵。 ----CSDN 时时三省 strcat 字符串追加 示例: 比如我要把world加到hello后面去 就可以用这个 还有一种方法是这样 这两个代码的意思是一样的 只是写法不一样 写的时候要注意这些 •源字符串必须…...

ESP32C3 开发板在Linux环境下,进行JTAG 调试演示-启明云端乐鑫代理商
JTAG 调试教程 本教程主要演示 esp32c3 开发板在 linux 环境下,通过 vscode 的 esp-idf 插件使用 jtag 调试工具。 esp32c3 不但内置了USB-JTAG,还内置了USB-SERIAL,仅需要一根USB线即可实现下载和调试仿真。 下面演示调试仿真的过程。 创…...

《计算机视觉》—— 基于PyCharm中的dlib库实现人脸关键点定位
文章目录 1. 安装必要的库2. 下载dlib的人脸检测器和关键点预测器模型3. 编写代码 人脸关键点定位是指通过计算机视觉技术,识别和定位人脸图像中的关键点,如眼睛、鼻子、嘴巴等特定位置。这些关键点的准确定位对于人脸识别、表情分析、姿态估计等应用具有…...

c++习题34-说谎
目录 一,题目 二,思路 三,代码 一,题目 描述 ljc以自己的人格担保他最后一个回答一定是正确的,但并不保证其它的回答是对的。 每个数为ljc对上一个的回答,若为0表示说上句话是错的,若为…...

如何使用Android Profiler进行性能分析?
Android Profiler是Android Studio中一个功能强大的性能分析工具,它可以帮助开发者实时监控应用的CPU、内存、网络、电量和图形渲染等性能指标,从而发现并解决性能瓶颈。以下是如何使用Android Profiler进行性能分析的详细步骤和技巧。 一、准备工作 安…...

整理—MySQL
目录 NOSQL和SQL的区别 数据库三大范式 MySQL 怎么连表查询 MySQL如何避免重复插入数据? CHAR 和 VARCHAR有什么区别? Text数据类型可以无限大吗? 说一下外键约束 MySQL的关键字in和exist mysql中的一些基本函数 SQL查询语句的执行顺…...

临时配置linux Bridge网桥
Linux Bridge(网桥)是用纯软件实现的虚拟交换机,有着和物理交换机相同的功能,例如二层交换,MAC地址学习等。因此我们可以把tun/tap,veth pair等设备绑定到网桥上,就像是把设备连接到物理交换机上…...

【Canvas与化学】铁元素图标
【成图】 【代码】 <!DOCTYPE html> <html lang"utf-8"> <meta http-equiv"Content-Type" content"text/html; charsetutf-8"/> <head><title>铁元素图标Draft1</title><style type"text/css"…...

list转map常用方法
利用Collectors.toMap收集指定属性 public Map<Long, String> getIdNameMap(List<Account> accounts) {return accounts.stream().collect(Collectors.toMap(Account::getId, Account::getUsername)); } 收集对象实体本身 - 在开发过程中我们也需要有时候对自己…...

C++容器适配器的模拟实现-stack、queue、priority_queue
### 容器适配器是将容器转换到其他容器自身不方便使用的地方,但是就容器适配器其本身还是包装的容器,所以这个类模板中各个接口的实现都是调用的容器的接口,因为容器适配器可能适配多个容器,所以这个类模板的模板参数中有一个参数…...

fastapi的docs页面是空白解决
环境:openEuler、python 3.11.6、fastapi 0.115.2 背景:居家办公,默认搭建的fastapi的docs接口为空白 时间:20241016 说明:网上很多教程的缺点是复杂(但是能够了解的更清楚),使用…...

浙大数据结构:11-散列4 Hashing - Hard Version
这道题主要在于思路,感觉像个模拟题,用到了线性探测的算法 机翻 1、条件准备 visit数组看这个位置有没有放进来数,num存非负整数,s存未到放入时机的数。 answer存答案。n为总共数量。 #include <iostream> #include<…...

pm2 守护http-server
PM2(Process Manager 2)是一个用于Node.js应用程序的进程管理器。以下是使用PM2守护HTTP服务器的步骤: 1. 安装PM2 如果你还没有安装PM2,可以使用以下命令安装: npm install pm2 -g 2. 启动HTTP服务器 你需要一个HTT…...

国外电商系统开发-运维系统应用管理
还记得您常用的 service httpd start 、service sshd stop这样的命令吗?这些都是在停止启动服务,为了让研发人员,或者是快速操作服务,这里给大家制定了简单的应用管理。在这里,您可以把上面的命令加入进来,…...

剖析线程池实现原理
前置推荐阅读:java并发之线程池使用-CSDN博客 自定义实现一个带监控的线程池 首先我们继承ThreadPoolExecutor,实现构造函数以及重写beforeExecute和afterExecute两个函数,具体调用我们会在代码实现层面进行详细的分析。 import java.util.…...

【中危】Oracle TNS Listener SID 可以被猜测
一、漏洞详情 Oracle 打补丁后,复测出一处中危漏洞:Oracle TNS Listener SID 可以被猜测。 可以通过暴力猜测的方法探测出Oracle TNS Listener SID,探测出的SID可以用于进一步探测Oracle 数据库的口令。 建议解决办法: 1. 不应该使…...

三维测量与建模笔记 - 简介
计算机视觉相关主题 主要有两个最主要的层面,几何和语义。几何层面描述了客观事实,比如物体的距离、大小、形状、位置等。语义层面则是从人类抽象出的概念出发,描述了物体是什么、行为是什么、为什么,比如自动驾驶场景中识别出信号…...

Glide 简易教程
文章目录 1 引入依赖2 图片形状2.1 圆形 CircleCrop2.2 旋转 Rotate2.3 圆角 RoundedCorners2.4 自定义圆角 GranularRoundedCorners 1 引入依赖 implementation("com.github.bumptech.glide:glide:4.16.0")2 图片形状 2.1 圆形 CircleCrop Glide.with(this).load…...

flutter 使用三方/自家字体
将字体放入assets/fonts下 在pubspec.yaml文件中flutter下添加如下代码: flutter:fonts:- family: MyCustomFontfonts:- asset: assets/fonts/MyCustomFont.ttf 在flutter Text widget中使用字体 import package:flutter/material.dart;void main() > runApp(…...

2024台州赛CTFwp
备注: 解题过程中,关键步骤不可省略,不可含糊其辞、一笔带过。解题过程中如是自己编写的脚本,不可省略,不可截图(代码字体可以调小;而如果代码太长,则贴关键代码函数)。…...

词根plac-和place、please
英文有一个词根和单词place(v.放,放置 n.位置,地方;位,职位)长得很像,这个词根就是plac-,它有两个语义:高兴,愉悦;平静,抚平。 其实,place这个单…...

ubuntu下route命令详解
buntu下route命令详解 1、显示路由表 route -n2、临时路由设置,重启网卡失效#添加一条路由(发往192.168.62这个网段的全部要经过网关192.168.1.1)route add -net 192.168.62.0 netmask 255.255.255.0 gw 192.168.1.1#删除一条路由 删除的时候不用写网关route del …...

13.java面向对象:面向对象的三大特征
java面向对象:面向对象的三大特征 面向对象的三大特征1.封装get和set规范属性的合法化 2.继承类继承子类调用父类方法super的用法通过super调用父类public的属性super注意点super对比this 方法重写静态方法中奇怪的现象非静态方法 3.多态多态存在的条件多态中成员访…...

【VUE】Vue中的内置组件
Vue2中的内置组件: <component>:动态组件,可以根据传递的 is 属性值渲染不同的组件。<transition>:过渡动画组件,可以在元素插入、更新或移除时添加动画效果。<transition-group>:过渡动…...

若依框架篇-若依框架搭建具体过程、后端源代码分析、功能详解(权限控制、数据字典、定时任务、代码生成、表单构建、接口测试)
🔥博客主页: 【小扳_-CSDN博客】 ❤感谢大家点赞👍收藏⭐评论✍ 文章目录 1.0 若依框架概述 1.1 若依构建 1.2 后端项目搭建 1.3 前端项目搭建 2.0 利用若依框架生成前后端代码案例 3.0 功能详解 3.1 功能详解 - 权限控制 3.1.1 使用权限控制…...

恢复已删除文件的 10 种安卓数据恢复工具
由于我们现在在智能手机上存储了大量重要文件,因此了解数据恢复工具变得很重要。您永远不会知道什么时候需要使用 安卓 数据恢复工具。 由于不乏 Windows 数据恢复工具,因此从崩溃的计算机中恢复文件很容易。但是,当涉及到从 安卓恢复数据时…...

Internet Download Manager2025快速下载,新功能解锁!
🌟下载界的“速度与激情”:Internet Download Manager超燃体验!🔥 嘿,各位小伙伴们!👋今天我要来给你们安利一个让我上网冲浪效率翻倍的神奇软件——Internet Download Manager(简称…...

传感器应用注意事项
一、通断型传感器 多数活动部件可直接作为导电材料的传感器为通断型传感器,在受力的条件下,其两个引脚的通断状态会发生改变。 常见通断型传感器 单通道按键多通道按键拨码开关接线帽磁力开关轻触开关… 通断型传感器无需供电,其控制环路…...