C/C++:指针用法详解

C/C++:指针

指针概念

指针变量也是一个变量
指针存放的内容是一个地址,该地址指向一块内存空间
指针是一种数据类型

指针变量定义

内存最小单位:BYTE字节(比特)
对于内存,每个BYTE都有一个唯一不同的编号,这个编号就是内存的地址
一个地址编号对应的是一个BYTE的空间大小
一个地址编号在32位系统下,是一个4个字节的无符号整数;在64位系统下是一个8个字节的无符号整数

//描  述:指针的定义
#include <stdio.h>int main()
{int *p;//定义int型指针变量,名字为p,int型(可以执行int型地址)int a; //定义int型变量aa = 1;p = &a;//把a的内存地址赋值给pprintf("%p\n",p);//00B7FB24 实际上a占了4个字节(地址编号),但是只输出第一个地址编号//64位系统下a占8个字节,这也是为什么32位系统下内存用的少,64位内存用的多return 0;
}
int *p;
//p:指针变量,存放地址
//*p:指针所指内存的实际数据
*p = 10//p指向的地址存放的值是10
//指针变量只能存放地址,不能将一个int型变量直接赋值给一个指针

取地址符号&(found/address)

&可以获取一个变量在内存中的地址
register int a;//register限定符,a变成寄存器变量,存放在cpu里而不是内存中,所以是没有地址的,不能用&进行取地址操作

void指针

void代表无类型,可以指向任何类型的地址

//描  述:void指针
#include <stdio.h>int main01()
{int *p;int a;a = 1;p = &a;//把a的内存地址赋值给p*p = 10; //通过指针变量间接访问a的值,把a的值改为10printf("a = %d\n",a);printf("*p = %d\n",*p);a = 100;  //修改a的值时*p的值也会变化printf("a = %d\n",a);printf("*p = %d\n",*p);int b = 2; p = &b;  //现在指针变量p指向了b(指来指去)*p = 50;printf("b = %d\n",b);printf("*p = %d\n",*p);a = 123456;//char *p1 = &a;//p1为char型指针变量,a为int型变量//char *p1 = (char*)&a;//强制转换会导致错误的结果void *p2 = &a;//void代表无类型,可以指向任何类型的地址  return 0;
}

指针占用内存

在同一个系统下,不管指向什么类型的变量,地址编号的大小总是一样的(32位系统占4个BYTE,64位系统占8个BYTE),但是不同类型变量占用的内存是不同的,见下方的代码
就像在一个酒店里,不管是单人间,双人间还是标准间,门牌号的大小都是一样的(4位数/8位数)

//描  述:指针占用内存说明
#include <stdio.h>int main02()
{char *p;int *p1;long long *p2;printf("%lu,%lu,%lu", sizeof(p),sizeof(p1),sizeof(p2));//结果:4,4,4return 0;
}

上述的代码与下面的代码进行比较

//描  述:不同类型变量占用的内存大小#include<stdio.h>int main05()
{int a[10];printf("%p,%p,%p\n",a,&a[0],&a[1]);//输出结果:0135FCA0,0135FCA0,0135FCA4//a与&a[0]是一样的,&a[0]与&a[1]中间有0135FCA0、1、2、3,因为一个int型变量占4个内存(4个地址编号)char b[10];printf("%p,%p,%p\n",b,&b[0],&b[1]);//输出结果:0076FB24,0076FB24,0076FB25//一个char类型变量占一个内存(地址编号)long long c[10];printf("%p,%p,%p\n",c,&c[0],&c[1]);//输出结果:008FFC84,008FFC84,008FFC8Creturn 0;
}

空指针与野指针

空指针:如果一个指针变量没有明确得指向一块内存,那么就把这个指针变量指向NULL,空指针是合法的
野指针:没有初始化过值的指针(没有指定内存),野指针是非法的
程序中避免野指针,可以使用空指针

//描  述:空指针与野指针#include<stdio.h>int main03()
{/*野指针int *p;*p = 100;//没有指定内存*///空指针int *p1;p1 = NULL;return 0;
}

指针的兼容性

指针类型之间一定要匹配,指针之间的赋值比普通数据类型之间的赋值要更严格

int a;
double b = 3.45;
a = b;//正确,a = 3
int *p = &b;//错误,类型不一致

大白话:不要把指针想的太神秘,指针变量只是个变量而已,它里面放的就是一些地址编号,这些地址编号就是无符号的整数,在32位系统下是4个字节,在64位系统下是8个字节,但是这些整数不能直接赋值,来源于另外一个变量的取地址操作

指向常量的指针与指针常量

指向常量的指针
int a = 1;
const int *p = &a;//p可以指向一个int类型的地址,但不可以用*p的方式修改这个内存的值
*p = 10;//会报错:不能给常量赋值
printf("a = %d\n",a); 

对于常量来说,值不能改变

const int b = 0;
b = 10;//报错,b为常量,值不能改变

但是可以创建一个指针指向它,修改

const int b = 0;
int *p = &b;
*p = 0;//不会报错,但是会warning,实际上这是不合理的,c语言的漏洞 

c语言中的const是有问题的,可以通过指针变量间接修改const常量的值(c++中是无法修改的),所以在c语言中用#define常量的时候更多

指针常量
int *const p2 = &a;//p2只能指向a的地址
p2 = &b;//错误,p2是一个常量指针,只能指向固定的一个变量a的地址
//但是可以通过*p2来读写这个变量的值
*p2 = 10;

注意区分常量指针与指针常量

指针与数组的关系

p=a;数组的名字就代表数组第一个元素的地址,等同于 p = &a[0]
p1 = &a[5];//把a5的地址给指针p1
*p1 = 1000;//改变a[5]的值
p1[2] = 666;//p1指向a[5],则p1[2]顺延指向a[7],改变的是a[7]的值,把数组想象成一个队列
p[3] = 100;//当指针变量指向一个数组的时候,c语言语法规定指针变量名可以当数组名用,区别在哪里?
//区别:
printf("%lu,%lu\n", sizeof(a),sizeof(p));//对数组来讲返回数组的大小40,对指针来讲返回指针的大小4
//描  述:指针与数组的关系#include<stdio.h>int main()
{int a[10] = {1,2,3,4,5,6,7,8,9,10};int *p;p = a;//可以这样指向,数组的名字就代表数组第一个元素的地址,等同于 p = &a[0]int *p1;p1 = &a[5];//把a5的地址给指针p1*p1 = 1000;p1[2] = 666;//p1指向a[5],则p1[2]顺延指向a[7],改变的是a[7]的值,把数组想象成一个队列p[3] = 100;//当指针变量指向一个数组的时候,c语言语法规定指针变量名可以当数组名用,区别在哪里?//区别:printf("%lu,%lu\n", sizeof(a),sizeof(p));//对数组来讲返回数组的大小40,对指针来讲返回指针的大小4int i;for(i=0;i<10;i++){printf("a[%d]=%d\n",i,a[i]);printf("a[%d]=%d\n",i,p[i]);//像数组一样使用指针}return 0;
}
/*
运算结果:
40,4
a[0]=1
a[0]=1
a[1]=2
a[1]=2
a[2]=3
a[2]=3
a[3]=100
a[3]=100
a[4]=5
a[4]=5
a[5]=1000
a[5]=1000
a[6]=7
a[6]=7
a[7]=666
a[7]=666
a[8]=9
a[8]=9
a[9]=10
a[9]=10
*/

指针运算

指针变量可以计算,int* 类型加一,变化4个整数(增加4个字节),char* 类型加一,变化1个整数

//描  述:指针运算-加一#include<stdio.h>int main07()
{int a = 0;int *p = &a;printf("%p,%p,%p\n",p,p+1,p+2);//结果:0118FCA4,0118FCA8,0118FCAC(加1操作加4个字节)char c = 0;char *p1 = &c;printf("%p,%p,%p\n",p1,p1+1,p1+2);//结果:0075F937,0075F938,0075F939return 0;
}
//描  述:指针运算2#include<stdio.h>int main()
{int a[10] = { 0 };int *p1 = a;p1 += 5;*p1 = 1;p1 -= 2;*p1 = 3;//p1 *= 2;//混淆int i;for(i=0;i<10;i++){printf("a[%d]=%d; ",i,a[i]);//结果:a[0]=0; a[1]=0; a[2]=0; a[3]=3; a[4]=0; a[5]=1; a[6]=0; a[7]=0; a[8]=0; a[9]=0;}return 0;
}

增加/减少指针值:p++;p–
求差值:pa-pb,通常用于同一个数组内求两个元素之间的距离
比较:pa == pb,通常用来比较两个指针是否指向同一个位置

通过指针使用数组元素

//描  述:指针与数组#include<stdio.h>int main00()
{int a[10] = {1,2,3,4,5,6,7,8,9};int *p = a;p[3] = 100;//等同于下面一句,一般习惯这样写,简洁//*[p+3] = 100;int i;for(i=0;i<10;i++){printf("a[%d] = %d\n",i,a[i]);}return 0;
}int main()
{//c语言中所有数据类型都可以理解为一个char的数组int a = 0x12345678;//0x开头,16进制,int是4个字节,可以视作4个char(12,34,56,78)的数组char *p = (char *)&a;//为了防止出现warning,使用强制转换printf("%x,%x\n", *p,p[1]);//结果:78,56,倒着放,小端对齐的概念*p = 0;//相当于p[0],a[0]printf("%x,%x\n", *p,p[1]);p[2] = 0;printf("%x\n",a);//结果:12005678printf("--------------------\n");char b[20] = {0};int *p1 = (int *)&b;//防止warning强转p1[3] = 0x12345678;//int占4个字符,3*4=12,从第13个位置也就是b[12]开始放(倒着放,小端对齐概念)int i;for(i=0;i<20;i++){printf("b[%d] = %x\n",i,b[i]);}
/*输出
--------------------
b[0] = 0
b[1] = 0
b[2] = 0
b[3] = 0
b[4] = 0
b[5] = 0
b[6] = 0
b[7] = 0
b[8] = 0
b[9] = 0
b[10] = 0
b[11] = 0
b[12] = 78
b[13] = 56
b[14] = 34
b[15] = 12
b[16] = 0
b[17] = 0
b[18] = 0
b[19] = 0
*/return 0;
}

c语言中所有数据类型都可以理解为一个char的数组

练习:把ip地址转化为整数

输入ip地址
char a[100]=“192.168.2.5”
把这个ip转化为unsigned int类型的整数

//描  述:把ip地址转化为整数#include<stdio.h>int main003()
{char a[] = "192.168.2.5";unsigned int ip = 0;unsigned char *p = (unsigned char *)&ip;int a1,a2,a3,a4;sscanf(a,"%d.%d.%d.%d",&a1,&a2,&a3,&a4);printf("%d,%d,%d,%d\n",a1,a2,a3,a4);p[0] = a4;p[1] = a3;p[2] = a2;p[3] = a1;printf("%u\n",ip);//运行结果:192,168,2,5    3232236037return 0;
}

练习:利用指针进行多维数组排序

//描  述:指针的灵活性——利用指针对多维数组进行排序#include<stdio.h>int main()
{char a[2][5] = {{3,4,1,2,9},{3,44,98,0,5}};//把二维数组"拉直"当成一维数组char *p =(char *)a;int i,j;for(i=0;i<10;i++){for(j=0;j<10-i;j++){if(p[j]<p[j-1]){char tmp = p[j];p[j] = p[j-1];p[j-1] = tmp;}}}for(i=0;i<2;i++){for(j=0;j<5;j++){printf("%d\n",a[i][j]);}}return 0;
}

指针数组

指针数组的定义:

char *a[10];//定义指针数组a,每个成员是char*类型的,一共10个成员
int *b[10];//定义指针数组b,每个成员是int*类型的,一共10个成员
printf("%lu,%lu\n",sizeof(a),sizeof(b));//结果:40,40

给数组成员赋值:

char i = 0;
//a = &i;//a和b为数组名,数组名不能作为左值
//b = &i;
a[0] = &i;//合法
printf("%lu,%lu\n",sizeof(a[0]),sizeof(*a[0]));//输出结果4,1
//描  述:指针数组#include<stdio.h>int main0000()
{char *a[10];//定义指针数组a,每个成员是char*类型的,一共10个成员int *b[10];//定义指针数组b,每个成员是int*类型的,一共10个成员printf("%lu,%lu\n",sizeof(a),sizeof(b));//结果:40,40char i = 0;//a = &i;//a和b为数组名,数组名不能作为左值//b = &i;a[0] = &i;//合法printf("%lu,%lu\n",sizeof(a[0]),sizeof(*a[0]));//输出结果4,1return 0;
}int main0001()
{int *b[10] = { NULL };int a,b1,c;b[0] = &a;b[1] = &b1;b[2] = &c;*b[0] = 10;printf("%d\n",a);return 0;
}

二级指针——指向指针的指针

int a = 0; 地址是0x123456
int *p = &a; p为0x123456,*p为0,指针p的存放地址是0x100
int **pp = &p; pp为0x100(p的地址), *pp为0x123456,**pp为0
//描  述:二级指针#include<stdio.h>//二级指针
int main0002()
{int a = 0;int *p = &a;int **pp;//二级指针,二级指针pp存放指针p的地址,pp = &p;//int **pp = &p;//前两句的结合体return 0;
}//二级指针与数组
int main0003()
{int a[10];int *p = a;//p是指向数组a的指针p[0] = 0;p[1] = 2;int *b[10];//定义指针数组bint **p1 = b;//定义指向指针数组b的二级指针p1(不能用一级指针指向b)p1[0] = NULL;printf("%lu\n",sizeof(p1[0]));//运行结果:4return 0;
}

对于一个指针变量b,想指向它需要定义一个二级指针a

多级指针

能用低级指针的尽量不要用高级指针,一级指针二级指针最常用,三级指针很少

//描  述:二级指针#include<stdio.h>//二级指针
int main0002()
{int a = 0;int *p = &a;int **pp;//二级指针,二级指针pp存放指针p的地址,pp = &p;//int **pp = &p;//前两句的结合体return 0;
}//二级指针与数组
int main0003()
{int a[10];int *p = a;//p是指向数组a的指针p[0] = 0;p[1] = 2;int *b[10];//定义指针数组bint **p1 = b;//定义指向指针数组b的二级指针p1(不能用一级指针指向b)p1[0] = NULL;printf("%lu\n",sizeof(p1[0]));//运行结果:4return 0;
}
int a = 0;
int *p = &a;
int **pp = &p;//pp代表p的地址,*pp代表a的地址,**pp代表a的值
int ***ppp = &pp;//ppp代表pp的地址,*ppp代表p的地址,**ppp代表a的地址,***ppp代表a的值

函数的参数作为指针变量

c语言想通过函数内部修改实参的值,只能给函数实参传递实参的地址来间接修改实参的值

//描  述:函数的参数作为指针变量#include<stdio.h>void swap(int *a,int *b)
{int tmp = *a;*a = *b;*b = tmp;
}int main0006()
{int a = 1;int b = 2;swap(&a,&b);//c语言想通过函数内部修改实参的值,只能给函数实参传递实参的地址来间接修改实参的值printf("a=%d,b=%d\n",a,b);return 0;
}

思考代码为什么输出为4和40?

//描  述:当数组名作为函数形参时,c语言将数组名解析为指针#include<stdio.h>//void test(int a[10])
void test(int *a)//最常用最简单
//void test(int a[])
{printf("%lu\n",sizeof(a));//输出结果:4a[5] = 100;
}
int main()
{int a[10] = {1,2,3,4,5,6,7,8,9,10};printf("%lu\n",sizeof(a));//输出结果:40printf("----------------\n");test(a);int i;for(i=0;i<10;i++){printf("a[%d]=%d\n",i,a[i]);/*结果:a[0]=1a[1]=2a[2]=3a[3]=4a[4]=5a[5]=100a[6]=7a[7]=8a[8]=9a[9]=10*/}return 0;
}
以下三条语句是一样的:
void test(int a[10])//当数组名作为函数形参时,c语言将数组名解析为指针  
void test(int *a)//最常用最简单
void test(int a[])

函数指针参数使用const保护:

void test(const int *a)//为了不让函数内部修改数组成员的值

在c语言中,可以转个弯(强转)在函数内部修改成员的值(限制菜鸟),但是在c++中是改不了的

函数的返回类型可以是指针类型

int *test()
{return NULL;
}

memset与memcpy与memmove

memset:将指定区域的内存置空(参数1:指定要置空内存的首地址;参数2:0;参数3:这块内存的大小,单位,字节)

memset(a,0,sizeof(a));
//描  述:memset:将指定区域的内存置空(参数1:指定要置空内存的首地址;参数2:0;参数3:这块内存的大小,单位,字节)#include<stdio.h>
#include<string.h>
int main0008()
{int a[10] = {1,2,3,4,5,6,7,8,9,10};//a[10] = { 0 };//错误赋值方法,这种初始化的方法只能在创建a数组时才能用//如果要把数组a置空,需要遍历修改/*int i;for(i= 0;i<10;i++){a[i] = 0;}*///采用memset可以简化上述的代码,需要加string.h的头文件memset(a, 0, sizeof(a));//memset(a, 0 ,10);//错误,只会把前10个字节清空/*int *p = a;memset(a, 0 ,sizeof(p));错误,指针变量的大小是8或4,所以只会清空前两个或者前一个字节*/int i;for (i=0;i<10;i++){printf("a[%d] = %d",i,a[i]);}return 0;
}

memcpy:在两块内存之间拷贝数据(参数1:目标地址;参数2:原地址;参数3:拷贝多少内容,字节)

memcpy(a,b,sizeof(b));
//描  述:memcpy在两块内存之间拷贝数据(参数1:目标地址;参数2:原地址;参数3:拷贝多少内容,字节)#include<stdio.h>
#include<string.h>int main0009()
{short a[10] = {1,2,3,4,5,6,7,8,9,10};//short一个整数占2个字节int b[10] = { 0 };//int型一个整数占4个字节memcpy(b, a, sizeof(a));int i;for(i = 0;i<10;i++){printf("%08x\n",b[i]);//%x以十六进制数格式输出整数,08补齐前面的0,为了看起来更容易理解一些}/*输出结果:由于a和b类型不一致,所以拷贝的时候会出现下列情况00020001000400030006000500080007000a00090000000000000000000000000000000000000000*/return 0;
}

memmove:移动内存(参数与memcpy一致)

memmove(a,b,sizeof(b));

注意:使用memcpy的时候一定要确保内存没有重叠区域

memcpy(&a[3],&a[0],20);//出现内存重叠区域

指针小结

int i;      定义整型变量
int *p;     定义指向int型变量的指针变量
int a[10];  定义int数组
int *a[10]; 定义指针数组,数组中的每个元素指向一个int型变量的地址
int func(); 定义函数,返回值类型为int
int *func();定义函数,返回值类型为int *型
int **p;    定义指向int型指针的指针,二级指针

字符指针与字符串

练习:利用指针对字符串进行倒置

//描  述:利用指针对字符串进行倒置
//思  路:设两个指针,p和p1,p从头开始,p1从后开始,第一个字符与最后一个字符交换,第二个与倒数第二个,直到p>p1;
#include<stdio.h>
#include<string.h>int main0011()
{char a[100] = "hello world";char *p = a;int len = strlen(a);char *p1 = p;//等同于*p1 = a;p1 += len -1;while(p<p1){char tmp = *p;//值交换*p = *p1;*p1 = tmp;p1--;//地址加1减1p++;}printf("%s\n",a);//%s字符串return 0;
}

c语言中,大部分的字符操作就是指针操作。

char s[] = "hello,world"
char *p = s;
p[0] = 'a';

数组作为函数的参数

如果一个数组作为函数的参数,那么数组的成员数量在函数内部是不可见的,一般在传递一个数组的时候,同时提供另一个参数,表明这个数组有几个成员
但是如果传递的是一个字符串,那么并不需要传递一个参数说明字符串的长度(因为字符串总是以0结尾,可以在内部判断0从而判断字符串的长度)
数组作为函数的参数

//描  述:数组作为函数的参数#include<stdio.h>//void test(char a[10])等价
//void test(char a[])
void print_array(int n,int *a)//提供另一个参数n表明数组的成员数量
{int i;for(i = 0;i<n;i++){printf("%d\n",a[i]);}}
int main()
{int a[] = {1,2,3,4,5,6,7,8,9};print_array(sizeof(a)/sizeof(a[0]),a);//n=sizeof(a)/sizeof(a[0])return 0;
}

指针数组作为main函数的形参

int main(int argc, char *argv[]);  

main函数是系统调用的,main函数的参数功能是得到命令行的参数
argc:argv数组的成员数量
argv:数组的每个成员都是char类型
argc:命令行参数的数量
argv:命令行参数的字符串数组
指针数组作为main函数的形参

//描  述:指针数组作为main函数的形参#include<stdio.h>//int main(int argv, char *args[])
int main(int argv, char **args)//args是一个指针数组char*,argc代表数组的成员数量
{int i;for(i=0;i<argv;i++){printf("%s\n",args[i]);}return 0 ;
}

练习:通过main函数实现四则加法运算,例如,命令行输入a 5 + 6 输出5+6的结果11

//描  述:利用main实现四则加法运算,例如,命令行输入a 5 + 6 输出5+6的结果11#include<stdio.h>
#include<stdlib.h>//atoi的头文件int main(int argc, char **args)
{if (argc<4)//如果参数不足直接提示并退出{printf("参数不足\n");return 0;}//注意:main函数的参数都是字符串,没有char,中间的+是“+”而不是‘+’int a = atoi(args[1]);//把第一个参数5转化为int型int b = atoi(args[3]);//把第三个参数6转化为int型char *s = args[2];//+是字符串,因此不能写成下面一句//char c = args[2];char c = s[0];//+是“+”的第一个字符,可得到第二个参数+字符//以上两句建议合并为下面一句//char c = args[2][0];switch (c){case '+':printf("%d\n",a+b);case '-':printf("%d\n",a-b);case '*':printf("%d\n",a*b);case '/':printf("%d\n",a/b);default:printf("error\n");}return 0;
}
//注意:*代表的是通配符,运算时结果是error,要想使用*运算符,加转移符号\
//如:a 1 \* 5

总结

在这里插入图片描述

  最近刚考完嵌入式操作系统,顺路将指针用法整理了一下,如有错误,请大家在评论区或私信我就可以,谢谢!

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

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

相关文章

赶紧转行大模型,预计风口就今年一年,明年市场就饱和了!不是开玩笑

恕我直言&#xff0c;就这几天&#xff0c;各大厂都在裁员&#xff0c;什么开发测试运维都裁&#xff0c;只有大模型是急招人。 你说你不知道大模型是什么&#xff1f;那可太对了&#xff0c;你不知道说明别人也不知道&#xff0c;就是要趁只有业内部分人知道的时候入局&#…

String常用方法详解

auth&#xff1a;别晃我的可乐 date&#xff1a;2024年06月16日 比较大小 equals(Object obj): 用于比较字符串内容是否相等。compareTo(String anotherString): 按字典顺序比较两个字符串。 String str1 "hello"; String str2 "world";boolean isEqual …

配置Linux DNS服务器作为自己的windows 的 DNS服务器和 配置遇到的问题

安装DNS 库 和 DNS工具&#xff1a; # bind 是用于创建 dns服务的&#xff0c; bind-utils是用于测试DNS服务的工具 yum -y install bind bind-utils配置主配置文件&#xff1a; # 下载好后就已经有DNS服务&#xff0c;但是需要你自己去配置DNS服务信息# 配置主配置文件 [rootl…

SylixOS下UDP组播测试程序

SylixOS下UDP组播测试 测试效果截图如下: udp组播发送测试程序。 /********************************************************************************************************* ** ** 中国软件开源组织 ** ** …

vue的安装配置并创建项目

npm 工具的安装 安装node.js之后&#xff0c;npm工具会自动安装到系统环境中 网址:https://nodejs.org/en vue的安装 安装vue并创建项目 npm create vuelatest 进入项目之后&#xff0c;然后启动npm run dev 解决方法&#xff1a; npm install -g create-vite 再次启动 通…

13.泛型、trait和生命周期(下)

目录 6. 生命周期与引用有效性6.1 生命周期引入6.2 得到长度最大的String值6.3 生命周期标注语法1&#xff09;说明2&#xff09;普通标注示例3&#xff09;函数参数中的生命周期标注 6.4 深入理解生命周期6.5 结构体定义中的生命周期标注6.6 生命周期省略 6.7 方法定义中的生命…

应急响应 | 基本技能 | 01-系统排查

系统排查 目录 系统基本信息 Windows系统Linux系统 用户信息 Windows系统 1、命令行方式2、图形界面方法3、注册表方法4、wmic方法 Linux系统 查看所有用户信息分析超级权限账户查看可登录的用户查看用户错误的登录信息查看所有用户最后的登录信息查看用户最近登录信息查看当…

机器学习周报第46周

目录 摘要Abstract一、文献阅读1.1 摘要1.2 研究背景1.3 论文方法1.4 模块分析1.5 网络规格1.6 高效的端到端对象检测1.7 mobile former模块代码 目录 摘要Abstract一、文献阅读1.1 摘要1.2 研究背景1.3 论文方法1.4 模块分析1.5 网络规格1.6 高效的端到端对象检测1.7 mobile f…

C++ 58 之 计算器案例

虚函数,vitual function C动态多态性是通过虚函数来实现的&#xff0c;虚函数允许子类&#xff08;派生类&#xff09;重新定义父类&#xff08;基类&#xff09;成员函数&#xff0c;而子类&#xff08;派生类&#xff09;重新定义父类&#xff08;基类&#xff09;虚函数的做…

day38-39| 509. 斐波那契数 70. 爬楼梯 746. 使用最小花费爬楼梯 62.不同路径 343. 整数拆分 96.不同的二叉搜索树

文章目录 前言动态规划理论基础509. 斐波那契数思路方法一 完整动态规划方法二 dp简化版方法三 使用递归 70. 爬楼梯思路方法一 动态规划方法一2 教程里面的简化方法方法二 拓展 746. 使用最小花费爬楼梯思路方法一方法二 拓展 62.不同路径思路 动态规划方法一方法二 递归 63. …

设计模式学习(二)工厂模式——工厂方法模式

设计模式学习&#xff08;二&#xff09;工厂模式——工厂方法模式 前言工厂方法模式简介示例优点缺点使用场景 前言 前一篇文章介绍了简单工厂模式&#xff0c;提到了简单工厂模式的缺点&#xff08;违反开闭原则&#xff0c;扩展困难&#xff09;&#xff0c;本文要介绍的工…

Hadoop 2.0:主流开源云架构(四)

目录 五、Hadoop 2.0访问接口&#xff08;一&#xff09;访问接口综述&#xff08;二&#xff09;浏览器接口&#xff08;三&#xff09;命令行接口 六、Hadoop 2.0编程接口&#xff08;一&#xff09;HDFS编程&#xff08;二&#xff09;Yarn编程 五、Hadoop 2.0访问接口 &am…

java打印helloworld

源代码 public class Function1 {public static void main(String[] args) {System.out.println("hello world");}} 打印结果

为什么说Python 是胶水语言?

​ "Python 是胶水语言"这一说法是指它很擅长将不同的程序或代码库连接在一起&#xff0c;能够让来自不同编程语言或框架的组件无缝协作。Python 具有丰富的库和简单的语法&#xff0c;使得它可以轻松调用其他语言编写的程序或使用不同技术栈的模块。 ​ 以下是几个…

C学习自学笔记-会陆续完善对应章节编程经典例子

C学习笔记 0>C语言概述 为什么学习C语言 1&#xff09;C的起源和发展------了解即可 B语言、C语言、C语言的产生地&#xff1a;都出自 美国贝尔实验室 2&#xff09;C的特点 优点&#xff1a;代码量小、速度快、功能强大 缺点&#xff1a;危险性高、开发周期长、可移植性…

MATLAB直方图中bin中心与bin边界之间的转换

要将 bin 中心转换为 bin 边界&#xff0c;请计算 centers 中各连续值之间的中点。 d diff(centers)/2; edges [centers(1)-d(1), centers(1:end-1)d, centers(end)d(end)];要将 bin 边界转换为bin 中心 bincenters binedges(1:end-1)diff(binedges)/2;

每日一练——用队列实现栈

225. 用队列实现栈 - 力扣&#xff08;LeetCode&#xff09; Queue.h #pragma once #include<stdlib.h> #include<assert.h> #include<stdbool.h>typedef int QDataType;typedef struct QNode {QDataType data;struct QNode* next; } QNode;typedef struct …

【办公类-04-03】华为助手导出照片视频分类(根据图片、视频的文件名日期分类导出)

背景需求&#xff1a; 用华为手机助手导出的照片视频&#xff0c;只能将jpg照片&#xff08;exifread读取图片的exif拍摄日期&#xff0c;Png、JPEG、mp4都无法识别到exif信息&#xff09; 【办公类-04-02】华为助手导出照片&#xff08;jpg&#xff09;读取拍摄时间分类导出…

LeetCode | 709.转换成小写字母

这道题可以用api也可以自己实现&#xff0c;都不难&#xff0c;大小字母之前相差了32&#xff0c;检查到大写字母时加上32即可 class Solution(object):def toLowerCase(self, s):""":type s: str:rtype: str"""return s.lower()class Solution…

试论地产需求政策的有效性边界

分析师通过对传统框架因子的分析和美日地产的回顾&#xff0c;指出收入政策将成为核心&#xff0c;测算认为地方收储面积约0.5-1.1亿平、收储资金0.8-1.9万亿元&#xff0c;70城二手房价降幅收窄至[-4.5%&#xff0c;-1.6%]。 事件&#xff1a;2024年5月17日&#xff0c;央行印…