C++核心编程——类和对象

本专栏记录C++学习过程包括C++基础以及数据结构和算法,其中第一部分计划时间一个月,主要跟着黑马视频教程,学习路线如下,不定时更新,欢迎关注
当前章节处于:
---------第1阶段-C++基础入门
---------第2阶段实战-通讯录管理系统,
=====>第3阶段-C++核心编程
---------第4阶段实战-基于多态的企业职工系统
---------第5阶段-C++提高编程
---------第6阶段实战-基于STL泛化编程的演讲比赛
---------第7阶段-C++实战项目机房预约管理系统

文章目录

  • 一、 封装
    • 1.1 封装基本概念
    • 1.2 访问权限
    • 1.3 成员属性设置成私有
  • 二、对象特性
    • 2.1 对象的初始化和清理
    • 2.2 构造函数的分类与调用
    • 2.3 构造函数调用规则
    • 2.4 浅拷贝与深拷贝
    • 2.5 初始化列表
    • 2.6 类对象作为类成员
    • 2.7 静态成员
  • 三. C++对象模型和this指针
    • 3.1 成员变量和成员函数分开存储
    • 3.2 this指针概念
    • 3.3 空指针访问成员函数
    • 3.4 const修饰成员函数

类和对象是C++的核心,C++面向对象的三大特性:封装、继承、多态,C++认为万事万物都是对象,对象尤其属性和行为。

一、 封装

1.1 封装基本概念

封装是C++面向对象三大特性之一,意义在于:

  • 将属性和行为作为一个整体,表现生活中的事务
  • 将属性和行为加以权限控制

语法
class 类名{ 访问权限:属性/ 行为};
代码举例说明,设计一个圆类,计算圆的周长:

#include <iostream>
using namespace std;
#define PI 3.14
class Circle {
	// 权限
public:
	//属性
	int r = 0;
	// 行为
	double getLength() {
		return r * r * PI;
	}
};

int main() {
	// 实例化对象
	Circle c1;
	c1.r = 3;
	cout << "圆的周长为:" << c1.getLength() << endl;
	system("pause");
	return 0;

}
圆的周长为:28.26
请按任意键继续. . .

创建类后需要实例化,才可以使用,相当于类是一个模具,实例化出来的对象才是产品。

#include <iostream>
using namespace std;
class Student {
	// 权限
public:
	// 属性
	string name;
	string cardnum;
	// 行为
	string getname() {
		return name;
	}
	string getcardnum() {
		return cardnum;
	}
};

int main() {
	// 实例化对象
	Student stu;
	stu.name = "张三";
	stu.cardnum = "123";
	cout << "姓名为:" << stu.getname() << "    学号为:" << stu.getcardnum() << endl;
	system("pause");
	return 0;

}
姓名为:张三    学号为:123
请按任意键继续. . .

1.2 访问权限

访问权限分为三种:

  • public 公共权限:类内可以访问,类外可以访问
  • protected 保护权限 类内可以访问,类外不可以访问
  • private 私有权限类内可以访问,类外不可以访问

protected和private区别在于继承时,儿子可以继承父亲中的protected内容,不可以继承父亲中private中内容

#include <iostream>
using namespace std;
class Person {
public:
	string name;
protected:
	string car;
private:
	string password;
public:
	void test() {
		// 类内都可以访问
		name = "张三";
		car = "自行车";
		password = "12345";
	}
};

int main() {
	Person p1;
	p1.name = "李四";// 类外可以访问
	// p1.car; // 类外不可访问
	// p1.password;// 类外不可访问
		 
	system("pause");
	return 0;

}

struct也可以定义类,但是struct默认权限是公共权限,而class默认权限是公有;

#include <iostream>
using namespace std;
class test1 {
	// 默认私有
	int a = 0;
};
struct test2 {
	// 默认公共
	int b = 0;
};

int main() {
	test1 test_1;
	test_1.a = 10; // 不能访问
	test2 test_2;
	test_2.b = 20; // 可以访问
	system("pause");
	return 0;

}

1.3 成员属性设置成私有

优点在于:

  1. 将所有成员属性设置成私有,可以自己控制读写权限
  2. 对于写权限,可以检测数据的有效性

通过get和set方法进行访问和赋值

#include <iostream>
using namespace std;

class Student {
	// 权限
private:
	// 属性
	string name="张三"; // 可读可写
	string cardnum="123";// 只读
	int score=0; // 只写
public:
	// 可读可写
	void setname(string inputname) {
		name = inputname;
	}
	string getname() {
		return name;
	}
	// 只读
	string getcarnum() {
		return cardnum;
	}
	// 只写
	void setscore(int inputscore) {
		score = inputscore;
	}
};
int main() {
	Student stu;
	cout << stu.getname() << endl;
	stu.setname("李四");
	cout << stu.getname() << endl;
	system("pause");
	return 0;

}
张三
李四
请按任意键继续. . .

可以用get和set方法进行值的合法性判断。
练习案例1:设计立方体类

  • 设计立方体类(Cube)
  • 求出立方体的表面积和体积
  • 分别用全局函数和成员函数判断两个立方体是否相等

全局函数需要传入两个实例化对象,成员函数只要传入一个实例化对象即可

#include <iostream>
using namespace std;
class Cube {
public:
	double x=0.0f; // 边长
	double y=0.0f; // 边长
	double z=0.0f; // 边长
	void setinf(double inputx, double inputy, double inputz) {
		x = inputx;
		y = inputy;
		z = inputz;
	}
	int getS() {
		return 3 * (x * y+x*z+y*z);
	}
	int getV() {
		return x * y * z;
	}
	bool judge(Cube c2) {
		if ((c2.x == x)&&(c2.y==y)&&(c2.z==z)){
			return true;
		}
		else {
			return false;
		}
	}
};
// 定义全局函数
bool g_judge(Cube c1, Cube c2) {
	if ((c2.x == c1.x) && (c2.y == c1.y) && (c2.z == c1.z)) {
		return true;
	}
	else {
		return false;
	}
}
int main() {
	Cube c1,c2; // 创建
	c1.setinf(1, 2, 3);
	c2.setinf(2, 3, 4);
	cout <<"c1表面积:"<< c1.getS() << endl;
	cout <<"c1体积:"<< c1.getV() << endl;
	// 成员函数判断
	string flag = c1.judge(c2) ? "Yes" : "No";
	cout <<"通过成员函数判断两立方体是否相等:"<< flag << endl;
	// 全局函数判断
	string g_flag = g_judge(c1,c2) ? "Yes" : "No";
	cout << "通过全局函数判断两立方体是否相等:" << flag << endl;
	system("pause");
	return 0;

}
c1表面积:33
c1体积:6
通过成员函数判断两立方体是否相等:No
通过全局函数判断两立方体是否相等:No
请按任意键继续. . .

练习案例2:点和圆的关系
计算一个圆形类(Circle)和一个点类(Point),计算点和圆的关系

#include <iostream>
using namespace std;
class Circle {
public:
	int x = 0;
	int y = 0;
	int r;
	void set_inf(int inputx,int inputy,int inputr) {
		x = inputx;
		y = inputy;
		r = inputr;
	}
};

class Point {
public:
	int P_x = 0;
	int P_y = 0;
	void set_inf(int input_P_x, int input_P_y) {
		P_x = input_P_x;
		P_y = input_P_y;
	}
};


string judge(Circle c, Point p) {
	int distance = sqrt((c.x - p.P_x)*(c.x - p.P_x) + (c.y - p.P_y)*(c.y - p.P_y));
	if (distance == c.r) {
		return "在圆上";
	}
	else if (distance < c.r) {
		return "在圆内";
	}
	else {
		return "在圆外";
	}
}
int main() {
	Circle c1;
	Point p1;
	c1.set_inf(1, 1, 1);
	p1.set_inf(1, 0);
	string flag = judge(c1, p1);
	cout << flag << endl;
	system("pause");
	return 0;

}
在圆上
请按任意键继续. . .

代码比较多,可以分文件编写,分成Circle.cpp,Circle.h,Point.h,Point.cpp,cpp中放函数实现方法,.h中放函数申明

circle.cpp

#include "circle.h"
void Circle::set_inf(int inputx, int inputy, int inputr) {
	x = inputx;
	y = inputy;
	r = inputr;
};

circle.h

#pragma once
#include <iostream>
using namespace std;


class Circle {
public:
	int x = 0;
	int y = 0;
	int r;
	void set_inf(int inputx, int inputy, int inputr);
};

point.cpp

#include "point.h"

void Point::set_inf(int input_P_x, int input_P_y) {
	P_x = input_P_x;
	P_y = input_P_y;
}

point.h

#pragma once
#include <iostream>
using namespace std;

class Point {
public:
	int P_x = 0;
	int P_y = 0;
	void set_inf(int input_P_x, int input_P_y);
};

main.cpp

#include <iostream>
using namespace std;
#include "circle.h"
#include "point.h"
//class Circle {
//public:
//	int x = 0;
//	int y = 0;
//	int r;
//	void set_inf(int inputx,int inputy,int inputr) {
//		x = inputx;
//		y = inputy;
//		r = inputr;
//	}
//};

//class Point {
//public:
//	int P_x = 0;
//	int P_y = 0;
//	void set_inf(int input_P_x, int input_P_y) {
//		P_x = input_P_x;
//		P_y = input_P_y;
//	}
//};


string judge(Circle c, Point p) {
	int distance = sqrt((c.x - p.P_x)*(c.x - p.P_x) + (c.y - p.P_y)*(c.y - p.P_y));
	if (distance == c.r) {
		return "在圆上";
	}
	else if (distance < c.r) {
		return "在圆内";
	}
	else {
		return "在圆外";
	}
}
int main() {
	Circle c1;
	Point p1;
	c1.set_inf(1, 1, 1);
	p1.set_inf(1, 0);
	string flag = judge(c1, p1);
	cout << flag << endl;
	system("pause");
	return 0;

}
在圆上
请按任意键继续. . .

二、对象特性

2.1 对象的初始化和清理

C++面向对象来源于生活,每个对象都会有初始化设置以及对象销毁前的清理数据设置。对象的初始化清理也是两个非常重要的安全问题。

  • 一个对象或者变脸没有初始状态,对其使用后果是未知。
  • 同样,使用完一个对象或变量,没有及时清理,也会造成一定的安全问题。

C++利用构造函数析构函数解决以上问题,两个函数会被编译器自动调用,如果不提供构造和析构,编译器会提供构造函数和析构函数的空实现。

  • 构造函数:创建对象时为对象的成员属性赋值;
  • 析构函数:对象销毁前系统自动调用,执行一些清理工作。

构造函数语法 类名(){}

  1. 构造函数,没有返回值也不写void
  2. 函数名称与类名相同
  3. 构造函数可以有参数,因此可以发生重载
  4. 程序在调用对象时候会自动调用构造,无须手动调用,而且只会调用一次

析构函数语法~类名(){}

  1. 析构函数,没有返回值也不写void
  2. 函数名称与类名相同,在名称前加上符号~
  3. 析构函数不可以有参数,因此不可以发成重载
  4. 程序在对象销毁前会自动调用析构,无须手动调用,而且只会调用一次
#include <iostream>
using namespace std;
class Person {
public: // 加上作用域
	Person() {
		cout << "构造函数的调用!" << endl;
	}
	~Person() {
		cout << "析构函数的调用!" << endl;
	}
};

void test() {
	Person p; // 在栈区创建实例化对象
}
int main() {
	test();
	system("pause");
	return 0;
}
构造函数的调用!
析构函数的调用!
请按任意键继续. . .

2.2 构造函数的分类与调用

分类方式:

  1. 按参数分:有参构造和无参构造
  2. 按类型分:普通构造和拷贝构造

调用方式:

  1. 括号法
  2. 显示法
  3. 隐式转化法
#include <iostream>
using namespace std;

class Person {
	// 构造函数
	// 分类方式:有参构造、无参构造   普通构造和拷贝构造
	// 无参构造
public:
	Person() {
		cout << "调用的为无参构造!" << endl;
	}
	// 有参构造
	Person(int a) {
		age = a;
		cout << "调用的为有参构造!" << endl;
	}
	// 以上都为普通构造
	// 拷贝构造
	Person(const Person &p) {
		age = p.age;
		cout << "调用的为拷贝构造!" << endl;
	}

	int age;
};


int main() {
	// 调用方式
	// 1. 括号法
	// 无参调用
	Person p1;
	// 有参调用
	Person p2(10);
	// 拷贝调用
	Person p3(p2);

	// 2.显示法
	Person p4 = Person(10);
	// Person(p4); 不可用拷贝构造 初始化匿名函数
	Person p5 = Person(p4); // Person(p4)为匿名对象,当前行执行结束后,会立即被析构掉

	// 隐式转化法
	Person p6 = 10; //相当于Person p6 = Person(10)
	system("pause");
	return 0;
}
调用的为无参构造!
调用的为有参构造!
调用的为拷贝构造!
调用的为有参构造!
调用的为拷贝构造!
调用的为有参构造!
请按任意键继续. . .

拷贝构造函数使用时机

  1. 使用一个已经创建完毕的对象来初始化一个新对象
  2. 值传递的方式给函数参数传值
  3. 值方式返回局部对象
#include <iostream>
using namespace std;
class Person {
public:
	Person() {
		cout << "无参构造" << endl;
	}
	Person(int a) {
		age = a;
		cout << "有参构造" << endl;
	}
	Person(const Person &p) {
		age = p.age;
		cout << "拷贝构造" << endl;
	}
	int age;
};

void test() {
	Person p1(10);
	Person p2(p1); // 用一个已经创建好的对象来初始化一个新对象
	cout <<"p1的年龄为:"<< p1.age << endl;
	cout <<"p2的年龄为:"<< p2.age << endl;

}
void test2(Person p) {
}

Person test3() {
	static Person p;
	return p;
}
//void todo() {
//	test3();
//}
int main() {
	//Person p(10);
	Person p = test3();
	system("pause");
	return 0;
}
无参构造
拷贝构造
请按任意键继续. . .

值传递的方式给函数参数传值和值方式返回局部对象都相当于将原来的值拷贝了一份,因此相当于把原来的拷贝了一份.

2.3 构造函数调用规则

  • 如果用户定义有参构造函数,C++不再提供默认无参构造,但是会提供默认拷贝构造
  • 如果用户定义拷贝构造函数,C++不会再提供其他构造函数

可以这么理解: 拷贝构造>有参构造>无参构造,如果用户提供一个构造,编译器会自动补齐比他更高级的构造,不会提供比他更低级的构造。比如如果用户定义有参构造函数,C++不再提供默认无参构造,但是会提供默认拷贝构造

#include <iostream>
using namespace std;

class Person {
public:
	Person(int a) {
		m_age = a;
	}
	int m_age;
};

int main() {
	//Person p;// 报错
	Person p(10); // 不报错
	Person p2(p); // 不报错
	system("pause");
	return 0;

}

2.4 浅拷贝与深拷贝

  • 浅拷贝:简单的赋值拷贝操作
  • 深拷贝:在堆区重新申请空间,进行拷贝操作
    在这里插入图片描述
#include <iostream>
using namespace std;
class Person {
public:
	Person() {
		cout << "无参构造" << endl;
	}
	Person(int a,int n) {
		cout << "有参构造" << endl;
		age = a;
		// num = &n;报错
		num = new int(n);
	}
	Person(const Person& p) {
		cout << "拷贝函数" << endl;
		age = p.age;
		//num = p.num; // 报错
		num = new int (*p.num);
	}

	// 析构函数
	~Person() {
		cout << "调用析构函数" << endl;
		// 将指针开辟区域释放
		if (num != NULL) {
			delete num;
			num = NULL;
		}
	}
	int age;
	int* num;  // 指针在堆区开辟
};
void test() {
	int num = 123;
	// 有参构造
	Person p1(10, num);
	// 拷贝构造
	Person p2(p1); // 报错
}
int main() {
	test();
	system("pause");
	return 0;

}
有参构造
拷贝函数
调用析构函数
调用析构函数
请按任意键继续. . .

主要原因就是浅拷贝的地址被释放之后,二次释放。深拷贝会重新开辟一个地址,这样就不会再同一个地址释放多次。

2.5 初始化列表

语法:构造函数():属性1(值1),属性2(值2)…{}

#include <iostream>
using namespace std;


class Person {
public:
	Person() {

	}
	Person(string n,int a,string ad):name(n),age(a),address(ad){

	}
	string name;
	int age;
	string address;
};

int main() {
	Person p1("张三", 13, "安徽");
	cout << "姓名:" <<p1.name<< "   年龄:" << p1.age<<"   住址:" << p1.address<<endl;
	//Person p2("李四", 12); //报错,少参数
	//cout << "姓名:" << p2.name << "   年龄:" << p2.age << "   住址:" << p2.address << endl;
	system("pause");
	return 0;

}
姓名:张三   年龄:13   住址:安徽
请按任意键继续. . .

2.6 类对象作为类成员

#include <iostream>
using namespace std;

class Student {
public:
	string name;
	int score;
	Student(string n,int s):name(n),score(s){
		cout << "Student有参构造" << endl;
	}
	~Student() {
		cout << "Student析构函数" << endl;
	}

};

class Teacher {
public:
	string name;
	Student stu;
	Teacher(string t_n,string s_n,int s):name(t_n),stu(s_n,s){

	}
};


void test() {
	Teacher t1("李老师", "张三", 12);
	cout << "老师姓名:" << t1.name << "  学生姓名:" << t1.stu.name << "   学生分数:" << t1.stu.score << endl;
}
int main() {
	test();
	system("pause");
	return 0;

}
Student有参构造
老师姓名:李老师  学生姓名:张三   学生分数:12
Student析构函数
请按任意键继续. . .

2.7 静态成员

在成员变量和成员函数前加上关键字static,成为静态成员,分为:

  • 静态成员变量

    • 所有对象共享同一份数据
    • 在编译阶段分配内存
    • 类内声明,类外初始化
  • 静态成员函数

    • 所有对象共享同一个函数
    • 静态成员函数只能访问静态成员变量
#include <iostream>
using namespace std;
class Person {
public:
	string name;
	static int score; // 静态成员变量,需要类内定义,类外初始化
	void func() {
		name = "张三";
		score = 10;
		cout << "调用函数func!" << endl;
	}
	static void func2() {
		//name = "李四"; // 报错,静态成员函数只能调用静态成员变量
		score = 200;
		cout << "调用函数func2!" << endl;
	}
};
int Person::score=100;

int main() {
	Person p1;
	cout <<"p1分数为:"<< p1.score << endl;
	Person p2;
	p2.score = 60;
	cout << "p2分数为" << p2.score << endl;
	cout << "p1分数为" << p1.score << endl;
	// 直接使用类名调用静态成员变量
	cout << "Person::score:" << Person::score << endl;  // 注意不能使用. 而是使用::


	// 静态成员函数
	p2.func();
	p2.func2();
	system("pause");
	return 0;

}
p1分数为:100
p2分数为60
p1分数为60
Person::score:60
调用函数func!
调用函数func2!
请按任意键继续. . .

三. C++对象模型和this指针

3.1 成员变量和成员函数分开存储

C++中类内成员变量和成员函数分开存储,只有非静态成员变量才属于类的对象上

#include <iostream>
using namespace std;
class Person {
public:
	int a;  // 是类的对象
	static int b;// 不是类的对象
	void func() {

	}
	static void func2(){
	}
};
int Person::b = 0;
int main() {
	Person p;
	cout << "size of p=" << sizeof(p) << endl;
	system("pause");
	return 0;

}
size of p=4
请按任意键继续. . .

3.2 this指针概念

c++提供特殊的对象指针——this指针,this指针指向被调用的成员函数所属对象。this指针本质是一个指针常量,指向对象不能发生变化,指向对象的值可以发生变化。

  • this指针时隐含每一个非静态成员函数内的一种指针
  • this指针不需要定义,直接使用即可

this指针的用途:

  • 当形参和成员变量同名时,可用this指针来区分
  • 在类的非静态成员函数中返回对象本身,可使用return *this
#include <iostream>
using namespace std;
class Person {
public:
	int age;
	Person(int age) {
		//age = age; // 编译器认为这两个age是一样的
		//1. 防止形参和成员变量同名
		this->age = age; // 将形参age赋值给成员变量age
	}
	//2. 返回对象本身
	// 如果不是引用类型的话,每一次都会创建一个新对象,而不是在原来的对象上进行累加
	Person& Personadd(Person p) {
		this->age += p.age;
		return *this;
	}
};



int main() {
	Person p(10);
	Person p1(20);
	cout << "p年龄:" << p.age << endl;
	cout << "p1年龄:" << p.age << endl;
	p1.Personadd(p).Personadd(p); // 加两次
	cout << "年龄:" << p1.age << endl;

	system("pause");
	return 0;

}
p年龄:10
p1年龄:10
年龄:40
请按任意键继续. . .

3.3 空指针访问成员函数

空指针可以调用成员函数,但是要注意有没有用到this指针。如果用到了this指针,需要加以判断保证代码的健壮性

#include <iostream>
using namespace std;

class Person {
public:
	int age;
	void func() {
		cout << "func的调用" << endl;
	}
	void func2() {
		// 添加判断 增强代码的健壮性
		if (this == NULL) {
			return;
		}
		cout << "年龄为" << age << endl;  // age相当于this.age 如果是空指针的话会报错
	}
};

int main() {
	// 创建一个空指针
	Person* p = NULL;
	p->func();
	p->func2(); // 用到this指针会报错
	system("pause");
	return 0;

}
func的调用
请按任意键继续. . .

3.4 const修饰成员函数

常函数

  • 成员函数后加const后我们称这个函数为常函数
  • 常函数不可以修改成员属性
  • 成员属性声明时加关键字mutable后,在常函数中仍然可以修改

常对象

  • 声明对象前加const称该对象为常对象
  • 常对象只能调用常函数
#include <iostream>
#include <iostream>
using namespace std;
class Person{
public:
	string name; //成员变量
	mutable int age; // mutable修饰使得常函数也能进行修改
	// 无参构造函数
	Person() {
	}
	// 定义一个成员函数
	void test() {
		cout << "调用test函数" << endl;
	}
	// 定义一个常函数
	void func() const{
		//name = "李四"; // 报错,不能修改正常的成员属性
		age = 12; // 不报错,可以修改mutable修饰的成员变量
	}

};

int main() {
	// 创建一个常对象
	const Person p;
	//p.test();// 不可以调用正常的成员函数
	p.func(); // 可以调用常函数
	cout << "年龄为:" << p.age << endl;
	system("pause");
	return 0;

}
年龄为:12
请按任意键继续. . .

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Android DataStore:安全存储和轻松管理数据

关于作者&#xff1a;CSDN内容合伙人、技术专家&#xff0c; 从零开始做日活千万级APP。 专注于分享各领域原创系列文章 &#xff0c;擅长java后端、移动开发、人工智能等&#xff0c;希望大家多多支持。 目录 一、导读二、概览三、使用 3.1 Preferences DataStore 添加依赖数…

Python面向对象之继承

【 一 】什么是继承&#xff08;Inheritance&#xff09; 继承允许创建一个新类&#xff08;称为子类或派生类&#xff09;&#xff0c;从已存在的类&#xff08;称为父类或基类&#xff09;继承属性和方法。子类可以继承父类的特性&#xff0c;并可以通过添加新的属性和方法来…

JVM基础(5)——JVM垃圾回收算法

作者简介&#xff1a;大家好&#xff0c;我是smart哥&#xff0c;前中兴通讯、美团架构师&#xff0c;现某互联网公司CTO 联系qq&#xff1a;184480602&#xff0c;加我进群&#xff0c;大家一起学习&#xff0c;一起进步&#xff0c;一起对抗互联网寒冬 学习必须往深处挖&…

HarmonyOS应用开发学习笔记 UI布局学习 创建轮播(Swiper) artTS 轮播组件 简单使用

官方文档 Swiper组件提供滑动轮播显示的能力。Swiper本身是一个容器组件&#xff0c;当设置了多个子组件后&#xff0c;可以对这些子组件进行轮播显示。通常&#xff0c;在一些应用首页显示推荐的内容时&#xff0c;需要用到轮播显示的能力。 1、简单用法 loop 控制是否循环 …

TS 36.331 V12.0.0-过程(4)-测量

​本文的内容主要涉及TS 36.331&#xff0c;版本是C00&#xff0c;也就是V12.0.0。

SVG 绘制基本的几何图形

SVG 绘制基本的几何图形 简介SVG 是什么SVG 的优点一个简单的例子 SVG 基本图形矩形 rect圆形 circle椭圆 ellipse线条 line多边形 polygon折线 polyline路径 path 简介 SVG 是什么 SVG 全称为 Scalable Vector Graphics&#xff0c;即可缩放矢量图形。SVG 使用 XML 格式定义…

【OpenCV学习笔记08】- 图像基本操作

关于 OpenCV 官方文档的 GUI功能告一段落&#xff0c;接下来开始核心操作的学习。学习笔记中会记录官方给出的例子&#xff0c;也会给出自己根据官方的例子完成的更改代码&#xff0c;同样彩蛋的实现也会结合多个知识点一起实现一些小功能&#xff0c;来帮助我们对学会的知识点…

leetcode:LCR 159. 库存管理 III(python3解法)

难度&#xff1a;简单 仓库管理员以数组 stock 形式记录商品库存表&#xff0c;其中 stock[i] 表示对应商品库存余量。请返回库存余量最少的 cnt 个商品余量&#xff0c;返回 顺序不限。 示例 1&#xff1a; 输入&#xff1a;stock [2,5,7,4], cnt 1 输出&#xff1a;[2]示例…

通过wireshark抓取的流量还原文件(以zip为例)

wireshark打开流量包&#xff0c;通过zip关键字查找 追踪流可查看详细信息 选中media Type右键&#xff0c; 点击导出分组字节流选项 将生成的文件进行命名&#xff0c;需要时什么格式就以什么格式后缀

uniapp 字母索引列表插件(组件版) Ba-SortList

简介&#xff08;下载地址&#xff09; Ba-SortList 是一款字母索引列表组件版插件&#xff0c;可自定义样式&#xff0c;支持首字母字母检索、首字检索、搜索等等&#xff1b;支持点击事件。 支持首字母字母检索支持首字检索支持搜索支持点击事件支持长按事件支持在uniapp界…

代币中的decimal精度代表了什么

精度的意义在于允许发送小数的代币。举例&#xff0c;一个CAT代币合约的精度为6。那么 你拥有1个CAT就意味着合约中的balance 1 * 10^6 , 转账 0.1CAT出去的话&#xff0c;就需要输入 0.1*10^6 10^5。 也就时在涉及代币时&#xff0c;查询到的余额、转账的代币数量 都和 代币…

扫描错题用什么软件?分享4个好用的工具!

在学习的道路上&#xff0c;我们难免会遇到错题。有些时候&#xff0c;手抄的笔记或是纸质错题集不易携带、查找不方便&#xff0c;还容易丢失。为了解决这些问题&#xff0c;现在有许多软件可以帮助我们快速、准确地扫描错题&#xff0c;并进行整理和纠正。本文将为你介绍几款…

满足ITOM需求的网络监控工具

IT 运营管理&#xff08;ITOM&#xff09;可以定义为监督 IT 基础架构的各种物理和虚拟组件的过程;确保其性能、运行状况和可用性;并使它们能够与基础架构的其他组件无缝协作。IT 运营管理&#xff08;ITOM&#xff09;在大型 IT 管理模型中也发挥着积极作用&#xff0c;包括 I…

使用GraphQL实现简单的增删改查

使用GraphQL实现简单的增删改查 GraphQL官网&#xff1a;https://graphql.cn/ Altair Graphql 调试工具&#xff1a;https://saltair.sirmuel.design/#download 或者添加扩展使用网页版&#xff1a;https://chrome.google.com/webstore/detail/altair-graphql-client/flnheeel…

在线图表编辑工具Draw.io本地部署并结合内网穿透实现远程协作办公

前言 提到流程图&#xff0c;大家第一时间可能会想到Visio&#xff0c;不可否认&#xff0c;VIsio确实是功能强大&#xff0c;但是软件为收费&#xff0c;并且因为其功能强大&#xff0c;导致安装需要很多的系统内存&#xff0c;并且是不可跨平台使用。所以&#xff0c;今天给…

怎么处理网站的一些安全风险

随时互联网的持续发展&#xff0c;数字化转型步伐不断加快&#xff0c;社会各行业都走进了信息化、数字化。但与此同时&#xff0c;网络发展带来了许多风险&#xff0c;各行业面临着日益复杂的数据安全和网络安全威胁。其中&#xff0c;网站的安全风险持续增长&#xff0c;是各…

R语言生物群落(生态)数据统计分析与绘图教程

详情点击链接&#xff1a;R语言生物群落&#xff08;生态&#xff09;数据统计分析与绘图教程 前沿 R 语言作的开源、自由、免费等特点使其广泛应用于生物群落数据统计分析。生物群落数据多样而复杂&#xff0c;涉及众多统计分析方法。 一&#xff1a;R和Rstudio及入门和作…

零基础学习数学建模——(一)什么是数学建模

本篇博客将详细介绍什么是数学建模。 文章目录 个人简介什么是数学建模&#xff08;一&#xff09;引例&#xff1a;高中数学里的简单线性规划问题数学建模的定义及用途数学建模的定义数学建模的用途 正确认识数学建模 个人简介 ​ 本人在本科阶段获得过国赛省一、mathorcup数…

【算法】基础算法001之双指针

&#x1f440;樊梓慕&#xff1a;个人主页 &#x1f3a5;个人专栏&#xff1a;《C语言》《数据结构》《蓝桥杯试题》《LeetCode刷题笔记》《实训项目》《C》《Linux》《算法》 &#x1f31d;每一个不曾起舞的日子&#xff0c;都是对生命的辜负 目录 前言 1.数组分块&#xf…

UTONMOS:探索元宇宙,开启未来游戏新篇章

在元宇宙的世界里&#xff0c;游戏不再只是消遣&#xff0c;而是一个全新的互动世界&#xff0c;等待你来探索&#xff01; 逼真的虚拟现实技术&#xff0c;让你沉浸在充满想象力的游戏世界中&#xff0c;体验前所未有的刺激和乐趣。 与来自全球的玩家互动交流&#xff0c;结…