C++ 初探:不要‘下次一定’,立即开始你的C++之旅

news2025/1/22 15:58:28

C++初识

文章目录

  • C++ hello world
  • namespace,命名空间
    • 命名空间的使用
      • 域作用限定符
      • 展开命名空间
      • 指定展开命名空间成员
    • C++的域
  • C++的输入和输出
  • 缺省参数
  • 函数重载
  • 引用(reference)
    • 引用概念
    • 引用的特性
    • 引用的使用
    • const引用
  • inline
  • nullptr

C++ hello world

#include <iostream>
//test.cpp
using namespace std;
int main()
{
	std::cout << "hello world" << std::endl;
    cout << "haha" << endl;
	return 0;
}

使用using将命名空间展开,使用cout标准输出流,和流插入运算符 << 将hello world输出到控制台。使用endl来完成C语言的换行。C++是在C语言之上发展出来的,在C++的编译器里是兼容C语言的,两者可以混合使用。

namespace,命名空间

在C/C++中,变量、函数、类大量存在,由于它们的大量存在而可能导致名称使用上的冲突。namespace关键字,是针对名称冲突而出现的。

  • namespace关键字后面跟上命名空间的名字,通过大花括号( {} )括在一起,{}中为名称空间的成员,在名称空间内可以对,变量、函数、结构体、枚举等进行重命名。
#include <iostream>
namespace hx
{
    int rand = 6;
	int Add(int left, int right)
	{
		return left + right;
	}
	struct Node
    {
      	int val;
     	struct Node* next;
	};
}
  • namespace本质上是定义了一个域,这个域跟全局域各自独立,不同域可以定义同名变量。

  • 想要使用名称空间的成员,必须在变量/函数/结构体/……前加上名称空间的名字和 ::域作用限定符。否则默认使用,其它地方出现的变量/函数/结构体/……

  • 名称空间只能在全局定义,可以嵌套使用

  • 项目工程里,多文件定义同名namespace会认为是同一个namespace,不会发生冲突

  • C++标准库都放在一个std的命名空间中

案例1:

#include <iostream>
namespace hx//命名空间的名字
{
	int range = 6;
	void swap(int x, int y)
	{
		std::cout << "void swap(int x, int y)" << std::endl;
	}
    struct Node
    {
      	int val;
     	struct Node* next;
	};
}
void swap(int& x, int& y)
{
	int temp = x;
	x = y;
	y = temp;
}
int main()
{
	int a = 5;
	int b = 10;
	int range = 1;
	printf("%d %d\n", range, hx::range);
    
	swap(a, b);
    printf("%d %d\n", a, b);
	hx::swap(a, b);
	return 0;
}

在这里插入图片描述

案例2(嵌套定义):

namespace xyz
{
	namespace xy
	{
		int range = 6;
		void swap(int x, int y)
		{
			std::cout << "void swap(int x, int y)" << std::endl;
		}
	}
	namespace yz
	{
		int range = 3;
		void swap()
		{
			std::cout << "void swap()" << std::endl;
		}
	}
}
int main()
{
	int range = 66;
	std::cout << xyz::xy::range << std::endl;
	std::cout << xyz::yz::range << std::endl;
}

在这里插入图片描述

命名空间的使用

域作用限定符

  • ::,域作用限定符。

  • ::变量名,访问全局变量。

  • 命名空间名::变量名,访问了命名空间域。

  • struct hx::Node node;,访问结构体。

#include <iostream>
using namespace std;
namespace hx
{
	int rand = 0;
	void Print()
	{
		cout << "哈哈哈" << endl;
	}
	struct Node
	{
		int val;
		struct Node* next;
	};

}
int rand = 10;
int main()
{
	int rand = 10;
	cout << ::rand << endl;
	cout << rand << endl;

	hx::Print();

	struct hx::Node node;
	
	return 0;
}

展开命名空间

使用using将名称空间全部成员展开,在项目中不推荐,名称之间的冲突很大。日常做练习,测试方便可以使用,但不能养成创建一个main函数前,第一件事将,std命名空间展开,以及忘记对std命名空间的成员前使用域作用限定符的坏习惯

使用using展开前,编译器是先查找局部,然后查找全局,有了using将命名空间展开后还会到命名空间内查找,有一种将命名空间成员,改变为全局变量的意味。

#include <iostream>
using namespace std;
int main()
{
    int a = 0;
    cin >> a;
    cout << a << endl;
	return 0;
}

使用using展开后,就不需要频繁使用 std::,方便了许多。

指定展开命名空间成员

项目中经常使用,且不会存在冲突的名称

#include <iostream>
namespace hx
{
    int n = 10;
    int num = 20;
}
using hx::num;
int main()
{
    cout << num << endl;
    return 0;
}

C++的域

C++中域有函数局部域、全局域、命名空间域、类域,域影响的是变量/函数/类型的出处的逻辑, 有了域隔离(不同的域可以同名,同一个域不可以定义同一个东西),名字冲突的问题就解决了。

局部域和全局域除了会影响编译时的查找逻辑(先局部查找,然后全局查找),还会影响变量的生命周期。

  • 局部变量存储在栈上,生命周期时临时的,结束函数的调用或作用域结束时消亡。

  • 全局变量存储静态存储区上,生命周期是永久的,从程序开始执行诞生,到程序终止时消亡。

  • 命名空间域和类域不影响变量生命周期。

  • 命名空间域,只能定义在全局,它的成员本质上是全局变量。

  • 命名空间域,与全局进行隔离

    编译时想要使用变量/函数/结构体/……编译时向上查找过程必须包含对应的代码,先局部查找,然后全局查找。

C++的输入和输出

< iostream >,是Input Output Stream 的缩写,是标准的输入、输出流库,定义了标准的输入、输出对象。

  • std::cin,是 istream 类的对象,主要面向窄字符的标准输出,与C语言的不同,它不需要使用占位符,可以自动识别任意类型的变量。使用流提取运算符(>>)cin >> 变量名 << endl

  • std::cout,是ostream类的对象,主要面向窄字符的标准输出流,不需要使用占位符,%d、%s、%c这些东西,可以自动识别任意类型的变量、数据类型进行输出,使用流插入运算符(<<)

    • 不管是什么类型的数据,都会被转换为字符流,进行输入输出
  • std::endl(end line),是一个函数,流插入、输出时,相当于插入一个换行符 加 刷新缓冲区。

    • ‘\n’,这种换行方法,在不同的系统上可能不兼容,无法通过换行符进行换行,使用endl更加多用。
  • 使用C++输入输出更加方便,不需要像printf、scanf输入输出时那样,需要使用占位符手动指定格式,C++的输入输出可以自动的识别变量的类型。

  • C++兼容C语言,头文件< iostream >,间接包含了printf、scanf,在vs编译器上是这样,其它编译器不一定能支持。

使用cout进行输出时,输出内容不会第一时间输出,而是先将这些东西装换为字符串,然后放在cout的缓冲区,需要遇见一些刷新标志才会出缓冲区,比如换行、flush(cout的主动刷新接口)才会输出到控制台中。

  • 由于这些机制导致了cout的效率不会很高

这三句代码可以提高,C++IO流的效率

int main()
{
    ios_base::sync_with_stdio(false);//使C语言和C++不同步兼容
	cin.tie(nullptr);
 	cout.tie(nullptr);//将cin和cout相互绑定的关系,不会与其它流相互关联
	return 0;
}

缺省参数

C++中的缺省参数(默认参数),允许在对函数进行声明或定义时为函数参数提供一个缺省值。在调用该函数时,如果没有传递参数,则使用该形参对应的缺省值,否则使用传递过来的参数,缺省参数分为全缺省和半缺省

函数声明和定义分离时,缺省参数不能再函数声明和定义中同时出现,规定必须哈数声明给缺省值,例如,实现数据结构队列的.h头文件的函数声明和.c文件对函数的定义。

  • 全缺省参数,给全部形参缺省值,C++规定必须从左到右依次给实参,不能跳跃给实参
#include <iostream>
using namespace std;
int Add(int a = 1, int b = 2, int c = 3)
{
	return a + b + c;
}
int main()
{
	cout << Add() << endl;
	cout << Add(10) << endl;
	cout << Add(10, 20) << endl;
	cout << Add(10, 20, 30) << endl;
	return 0;
}

在这里插入图片描述

全缺省参数的,几种调用形式

  • Add(),没有传递参数,调用函数add是使用三个缺省值。
  • Add(10),传递了第一个参数,没有后续两个参数,形参b和c使用缺省值。
  • Add(10, 20),传递了第一个和第二个参数,没有最后一个参数,形参c使用缺省值。
  • Add(10, 20, 30),三个参数都传递了,不会使用缺省值。

  • 半缺省参数,部分形参给缺省值,C++规定半缺省值必须从右向左依次给缺省值
#include <iostream>
using namespace std;
int Add(int a, int b, int c = 3)
{
	return a + b + c;
}
int main()
{
	cout << Add(10, 20, 30) << endl;
	return 0;
}


例如,在对栈的数据结构进行初始化时,可以给一个缺省值,我可以不传递参数让其默认开辟4个空间。

//初始化
void StackInit(Stack* ps, int n = 4)
{
	assert(ps);
	ps->arr = (StackDatdType*)malloc(sizeof(StackDatdType) * n);
	ps->capacity = ps->top = 0;
}

函数重载

C++中允许在同一个作用域里定义多个同名函数,只要它们的参数列表不同。这种特性使得同一个函数名可以执行不同的操作,C语言不支持这种操作。

  • 参数列表不同
  • 返回类型可以相同,也可以不同,只凭借返回类型无法实现函数重载
  • 作用域必须在同一个
#include <iostream>
int Add(int left, int right)
{
	return left + right;
}
double Add(double left, double right)
{
	return left + right;
}

int main()
{
	Add(1, 2);
	Add(1.2, 5.2);
	return 0;
}

这里实际上调用了两个函数


这里同时使用了函数重载和缺省参数,语法上没问题,当直接调用 test(),编译器直接傻眼了不知道到的用那个函数,有歧义。

void test()
{
    cout << "void test()" << endl;
}
void test(int n = 4)
{
    cout << "void test(int n = 4)" << endl;
}
int main()
{
    test();
    return 0;
}

在这里插入图片描述

引用(reference)

引用概念

很牛的东西。

引用就是取别名,引用不是新定义一个变量,而是给以存在变量取一个别名,编译器不会为引用变量开辟内存空间,它和引用的变量共同使用同一块内存空间。比如土豆、洋芋、马铃薯,不同的称呼,指向的都是同一个东西。

C++中为了避免引⼊太多的运算符,会复⽤C语⾔的⼀些符号,⽐如前⾯的<<和>>,这⾥引⽤也和取 地址使⽤了同⼀个符号&。
在这里插入图片描述


类型& 引用别名 = 引用对象

int main()
{
	int a = 6;
    int& b = a;
    int& c = a;
    int& d = b;//也可以对别名b取别名,d也是指向变量a
    
    d++;//对d加加,a、b、c都会加1,它们指向同一块内存。
}
int main()
{
    int a = 6;
    int& b = a;
    int& c = a;
    int& d = b;

	cout << &a << endl;
    cout << &b << endl;
    cout << &c << endl;
    cout << &d << endl;

    d++;
    cout << a << endl;
}

在这里插入图片描述

引用的特性

  • 引用在定义时必须初始化
    在这里插入图片描述

  • 一个变量可以有多个引用(上面提过例子)

  • 引用一旦引用一个实体,就不能引用其它实体,这里使用ra对变量a引用之后,还使用b进行引用,编译代码时没有报错,但运行后会报错,提示ra多次初始化

int main()
{
	int a = 1;
	int b = 10;
	int& ra = a;

	int& ra = b;
   
    //这里并非是是引用,而是一个赋值,将10赋给ra
    ra = b;
	return 0;
}

在这里插入图片描述

引用的使用

  • 引用在实践中主要时用于传参和引用做返回值中,减少拷贝提高效率和改变引用对象时同时改变被用于对象

    例如,我传递100w个大小的整形数组,来解决某问题,传递数组时需要临时开辟100w个大小的整形空间,非常浪费,这里对数组使用引用就可以进行优化,从而提高效率。指针也能达到这种效果。

    对函数使用引用本质上是在传地址

  • 引用传参跟指针传参相对比较复杂,指针也也有类似场景。

    int& StackTop(Stack& ps)
    {
    	assert(ps.top > 0);
    	return ps.arr[ps.top - 1];
    }
    int main()
    {
    	Stack sk;
    	StackInit(&sk);
    	StackPush(&sk, 1);
    	StackPush(&sk, 2);
    
    	StackTop(sk)++;
    
    	return 0;
    }
    

    这里将栈顶元素放回后++,编译器会报错。调用函数返回后,它的返回值存放在一块临时对象中它具有常性无法被修改。

    在这里插入图片描述

    解决办法:将返回值进行引用,这里返回的就不会将返回值拷贝在临时对象中,而是给这个返回值取了一个别名返回的是栈顶元素的别名,对别名加加的结果是栈顶元素大小加1。

    在这里插入图片描述
    在这里插入图片描述

    int& StackTop(Stack& ps)
    {
    	assert(ps.top > 0);
    	return ps.arr[ps.top - 1];
    }
    int main()
    {
    	Stack sk;
    	StackInit(&sk);
    	StackPush(&sk, 1);
    	StackPush(&sk, 2);
    
    	StackTop(sk)++;
    
    	return 0;
    }
    

无法使用引用进行返回的场景

//指针
int* test()
{
	int ret = 10;
	
	//……
	
	return &ret;//返回的是局部变量,函数调用完成后被销毁,返回地址的是一个野指针。
}
//引用
int& test()
{
	int ret = 10;
	
	//……
	
	return ret;//引用同理,引用一块被销毁的空间
}
int main()
{
    test()++;
    return 0;
}

在这里插入图片描述

  • 引用和指针在实践中相辅相成,功能有重叠性,但是各有特点,互相不可替代最大的点,C++引用定义后不能改变指向
#include <iostream>
using namespace std;

void Swap(int& x, int& y)
{
	int temp = x;
	x = y;
	y = x;
}
int main()
{
	int a = 10;
	int b = 50;
	Swap(a, b);
	cout << a << endl << b;
	return 0;
}

const引用

  • 可以用于const对象,但是必须使用const引用。const引用也可以引用普通对象,因为对象的访问权限在引用过程中可以缩小,但不能放大。指针也存在权限放大缩小。

    int main()
    {
    	const int a = 0;
    	//编译器报错  error C2440: “初始化”: 无法从“const int”转换为“int &”
        //int& ra = a;//权限放大
    	const int& ra = a;//平等关系
    
    	int b = 0;
    	int& rc = b;
    	const int& rd = b;//权限缩小
    
    	//error C3892: “rd”: 不能给常量赋值
    	//rd++;
    	rc++;
    	return 0;
    }
    
  • 类似于对 int& rb = b * 2; float& d = 3.14; int& rd = d;。进行引用。对 b * 2的结果会存放在一个临时变量中,将b * 2的结果计算出来后,编译器放在一个临时对象中,存储中间值。int& rd = d;对一个浮点数进行引用,先对变量d进行类型转换,这个类型转换的结果也会被存放在临时对象中。C++规定临时对象具有常性(常性,形如被const修饰),这里触发了权限放大,必须使用常引用才可行。

  • 所谓临时对象就是编译器需要⼀个空间暂存表达式的求值结果时临时创建的⼀个未命名的对象, C++中把这个未命名对象叫做临时对象。

  • const修饰的引用作为函数参数,可以引用临时对象、普通对象、const修饰的对象。

    void fun(const int& ra);
    

inline

使用inline修饰的函数叫做内联函数,编译时C++编译器会在调用内联函数的地方展开内联函数,这种调用内联函数的方式就不需要要创建函数栈帧,提高了效率。

  • inline只是一种建议,是否展开还得看编译器,加了inline的函数编译器可以选在在调用的地方不展开,不同编译器处理inline的方法和场景不同,C++并没有对这些进行限制标准。内联函数一般用于代码短小,频繁带哦用的函数。
  • 内联函数易错点含有循环语句、含有递归语句、含代码量大,满足这三点其一,都不能称之为内联函数,加上inline后会被编译器忽略编译器的防御手段
    • 若有1w次调用一个100行的内联函数的地方,内联函数展开后,代码量变为 10000 * 100 = 100w行。
      • 函数被编译后是一堆指令需要存储起来执行,内联展开可能会导致可执行程序变大,内存炸开,指令膨胀

  • C++设计内联函数的设计目的是为了替代C语言的宏函数,C语言里的宏函数容易出错,需要频繁使用括号,不方便调试。
  • inline函数不支持声明和定义分离到两个文件,因为编译器一旦将一个函数作为内联函数处理,就会在调用位置展开,即该函数是没有地址的,也不能在其他源文件中调用,所以一般都是直接在源文件中定义内联函数的。
//#define ADD(a, b) a + b;
//#define ADD(a, b) (a + b)
//#define ADD(a, b) ((a) + (b))
// 为什么不能加分号? 例如if语句调用加分号的ADD函数
// 为什么要加外⾯的括号,为什么要加⾥⾯的括号?

C语言debug版本默认不展开inline。debug版本想要展开需要设置这两点。

  • 第一步找到vs编译器的 项目,点击
  • 点击项目内容的最后一个选项。

找到C/C++选项里的常规,将调试信息格式,设置为 程序数据库(/Zi)
在这里插入图片描述

找到C/C++选项里的优化,将内联函数拓展设置为 只适用 _inline(/Ob1)

在这里插入图片描述

nullptr

NULL是一个宏,在C语言头文件里(stdd.f)可以观察。

#ifndef NULL
 	#ifdef __cplusplus
 		#define NULL 0
	#else
		#define NULL ((void*) 0)
	#endif
#endif
  • C++中的NULL被定义为字面常量0,C语言里被定义为为类型指针的常量。

在这种情况下,传递NULL想调用int*版本的f函数,但调用的结果时int版本的函数,为了弥补这样的不足,C++引用了关键字nullptr

void f(int x)
{
    cout << "void f(int x)" << endl;
}
void f(int* x)
{
   cout << "void f(int* x)" << endl;
}
int main()
{
    f(1);
    f(NULL);
    return 0;
}

在这里插入图片描述

  • nullptr在C++11中引用的特殊关键字,nullptr是一种特殊类型的字面常量,它可以转换为其它类型的指针类型。使用nullptr定义空指针可以避免类型转换的问题,因为nullptr只能被隐式地转换为指针类型,而不能被转换为整数类型。

    • 使用nullptr就可以避免上述情况出现的问题

    在这里插入图片描述

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

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

相关文章

学习C语言遇到的问题

前言 这是博主在班训班培训期间学习C基础过程中遇到的一些问题&#xff0c;我把遇到的问题以及这些问题的解答链接统一放在这篇文章了。这只是博主遇到的一些问题&#xff0c;可能不会适用于全部人&#xff0c;希望大家把这篇文章当做查漏补缺的内容吧。 问题一:把字符串赋给…

三十种未授权访问漏洞复现 合集( 三)

未授权访问漏洞介绍 未授权访问可以理解为需要安全配置或权限认证的地址、授权页面存在缺陷&#xff0c;导致其他用户可以直接访问&#xff0c;从而引发重要权限可被操作、数据库、网站目录等敏感信息泄露。---->目录遍历 目前主要存在未授权访问漏洞的有:NFS服务&a…

CnosDB 元数据集群 – 分布式时序数据库的大脑

CnosDB 是一个分布式时序数据库系统&#xff0c;其中元数据集群是核心组件之一&#xff0c;负责管理整个集群的元数据信息。 1. 概述 CnosDB 是一个分布式时序数据库系统&#xff0c;其中元数据集群是核心组件之一&#xff0c;负责管理整个集群的元数据信息。元数据包括数据库…

用Ollama 和 Open WebUI本地部署Llama 3.1 8B

说明&#xff1a; 本人运行环境windows11 N卡6G显存。部署Llama3.1 8B 简介 Ollama是一个开源的大型语言模型服务工具&#xff0c;它允许用户在自己的硬件环境中轻松部署和使用大规模预训练模型。Ollama 的主要功能是在Docker容器内部署和管理大型语言模型&#xff08;LLM&…

[CR]厚云填补_GridFormer

GridFormer: Residual Dense Transformer with Grid Structure for Image Restoration in Adverse Weather Conditions Abstract 恶劣天气条件下的图像恢复是计算机视觉中的一个难点。在本文中&#xff0c;我们提出了一种新的基于变压器的框架GridFormer&#xff0c;它可以作为…

【Android】ContentProvider基本概念

ContentProvider Android权限机制详解 <manifest xmlns:android"http://schemas.android.com/apk/res/android"package"com.example.broadcasttest"> <uses-permission android:name"android.permission.RECEIVE_BOOT_COMPLETED" />…

8.2 grafana上导入模板看图并讲解告警

本节重点介绍 : grafana 上导入mysqld-dashboardglobal status 相关源码解读重要指标讲解 连接数内存TPS、QPS 将采集任务添加到prometheus中 - job_name: mysqld_exporterhonor_timestamps: truescrape_interval: 8sscrape_timeout: 8smetrics_path: /metricsscheme: httpf…

七天打造一套量化交易系统:Day7-实盘交易接入方式与注意事项

七天打造一套量化交易系统&#xff1a;Day7-实盘交易接入方式与注意事项 前情回顾证券交易接口XTP 接口头文件列表XTP 接口 demo 示例 期货交易接口CTP-API开发系列专栏 数字货币交易接口实盘接入注意事项 量化交易系统的核心要素包括选择投资标的、资金的分配、何时入场、何时…

Midjourney咒语之装修设计

装修设计 living room with a chinese shanshui painting frame on a wall with a 2 inch frame, colors: blue, white, focus on the picture, 35mm lens, realistic, design, commercial, plants, furniture, centered painting --s 750 --ar 16:9 80 square meter minimalis…

浅谈简单的搜索算法(c++)

目录 DFS思路实现应用场景DFS 的优缺点优点缺点 例题讲解N皇后问题[题目描述]输入输出样例输入样例输出 思路AC代码排列数字[题目描述]输入格式输出格式数据范围输入样例&#xff1a;输出样例&#xff1a;思路AC代码 树的重心[题目描述]输入格式输出格式数据范围输入样例输出样…

百度网盘不下载怎么直接打印文件?

在数字化时代&#xff0c;百度网盘作为我们存储和分享文件的重要工具&#xff0c;承载了大量的文档、图片和资料。然而&#xff0c;当需要打印这些文件时&#xff0c;很多用户会面临一个共同的问题&#xff1a;不想下载到本地再打印&#xff0c;既占用空间又浪费时间。那么&…

自闭症儿童无法上学?专业康复机构是希望的灯塔

面对自闭症儿童因特殊需求而无法融入普通学校的困境&#xff0c;每一位家长的心中都充满了焦虑与无助。然而&#xff0c;在这个充满挑战的时刻&#xff0c;选择一条科学、系统的康复之路&#xff0c;成为了引领孩子走向未来的关键。星启帆&#xff0c;作为国内规模较大全寄宿制…

0_(机器学习)逻辑回归介绍

模型简介 逻辑回归&#xff08;logistic回归&#xff09;即对数几率回归&#xff0c;它虽然被称作“回归”&#xff0c;但却是一种用于二分类的分类方法。逻辑回归是通过分析一个样本被分为各个类的概率比较后得出该样本最有可能属于的类的一种分类方法。 逻辑回归公式推导 训…

《Python数据结构精要:选择与应用》

本文将深入探讨Python中的几种常见数据结构&#xff0c;并通过实际案例来展示它们的应用场景和优缺点。通过本文的学习&#xff0c;读者可以更好地理解何时使用哪种数据结构以达到最优的程序性能。 正文内容&#xff1a; 引言 介绍数据结构的重要性及其在Python中的实现。简…

8.4 字符串中等 443 String Compression 467 Unique Substrings in Wraparound String

443 String Compression 注意&#xff1a;这里是按照顺序压缩&#xff0c;不忽略顺序就不能用字母表计数再还原了。 如果char num 1 只需要压入char本身 num > 1 时还需要压入char的个数 按字符压入 class Solution { public:vector<char> Push(vector<char>&a…

吴恩达机器学习COURSE1 WEEK3

COURSE1 WEEK3 逻辑回归 逻辑回归主要用于分类任务 只有两种输出结果的分类任务叫做二元分类&#xff0c;例如预测垃圾邮件&#xff0c;只能回答是或否 实际上&#xff0c;在逻辑回归中&#xff0c;我们要做的任务就类似于在数据集中画出一个这样的曲线&#xff0c;用来作为…

数据拯救利器:必备免费数据恢复软件清单

说起办公室里的那些小插曲&#xff0c;有时候真是让人哭笑不得。这不&#xff0c;前几天我就遇到了个大麻烦——硬盘分区一不小心给搞砸了&#xff0c;眼看着那些重要的文件、报告还有客户资料就要跟我“说拜拜”&#xff0c;心里那个急啊&#xff0c;简直就像热锅上的蚂蚁&…

CSP2019第二题: 公交换乘

CSP 2019 公交换乘 题目来源&#xff1a;牛客网 题目&#xff1a;* 示例1 输入 6 0 10 3 1 5 46 0 12 50 1 3 96 0 5 110 1 6 135输出 36题意&#xff1a; 根据输入&#xff0c;计算地铁花费不能用到优惠券的公交车的花费 知识点&#xff1a; 结构体 思路&#xff1…

Spring(Day2)

一、静态代理 静态代理的主要特点是代理类和被代理类通常具有相同的接口&#xff0c;这样客户端代码可以透明地使用代理类代替被代理类。 首先我们建立一个接口Shopping&#xff0c;在里面定义一个shopping方法。然后创建两个类EasyA和Proxy类来继承Shopping类&#xff0c;并重…

前端 react 实现图片上传前压缩 缩率图

目录 一、安装 二、编写工具类 三、获取压缩后的File对象 一、安装 npm install compressorjs 或 yarn add compressorjs 官方文档&#xff1a;compressorjs - npm (npmjs.com) 二、编写工具类 /*** author Dragon Wu* since 2024/8/4 12:23* 图片压缩工具*/ import Com…