C++之“继承”
继续开始关于C++相关的内容。C++作为面向对象的语言,有三大特性:封装,继承,多态。
这篇文章我们开始学习:继承。
一、继承的概念和定义
1. 继承的概念
什么是继承呢?
字面意思理解来看:继承就是获得某个人的所有遗产。在这里我们可以理解为:一个类继承了父类的成员变量以及成员函数。
接下来是准确的介绍:继承机制是面向对象程序设计使代码可以复用的一个重要手段,它允许我们在原来的类上进行扩展,增加新的属性(成员变量)和方法(成员方法),从而生成一个新的类,称之为派生类。继承呈现了面向对象程序设计的层次结构,体现了由简单到复杂的认知过程。以前我们接触的是函数层次的复用,而继承就是类设计层次的复用。【父类/基类:这个类里面放的是共有的属性和方法。派生类/子类:可以看作是在父类上的扩展】
以前我们都是函数层面的复用(在一个函数里调用另一个函数),现在是类设计层面的复用。接下来用例子进一步说明什么是继承:
红色圈住的部分是两个类一模一样的部分,它们的成员函数和成员变量都有很多相似,看起来比较冗余,对于两个类我们可以采取继承的方法:将这两个类里面相同的(重复的)成员提取出来,放到父类,再将不同的成员(各自的成员)分别放在各自的类中不动,紧接着让派生类继承父类。
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<iostream>
class common //共有成员
{
public:void identity(){std::cout << "void identity()" << std::endl;}
protected:std::string _name;std::string _address;std::string _tel;
private:int _age;
};class student :public common //学生的那个类
{
public:void study(){}
protected:int _stuid;
};class teacher :public common //老师的类
{
public:void teaching(){}
protected:int _title;};
int main()
{//实例化对象student s;teacher t;s.identity(); t.identity();return 0;
}
公共的成员都放到common类中,student和teacher都继承common,就可以复用这些成员,不需要重复定义,省去了很多麻烦。
2. 继承的定义
定义格式
class student :public common
//class 派生类/子类:继承方式 父类/基类
继承方式
从图中可以看出:共有三种继承方式,这三个关键字,既可以在类里取修饰成员变量和函数,还可以在此处继承。它既是继承方式,也是访问限定符。之前了解到的是:使用public(公有的),类里类外都可以访问,private(私有的),只能在类里面使用,类外面不可以。在这里继续了解protected:使派生类可以访问到基类的成员变量/成员方法
分析:
1> 基类的private成员在派生类中无论以什么样的方式去继承,它们在派生类中其实都是不可见的。这里的不可见指的是基类的私有成员它还是被继承到了派生类的对象中,但是在语法上限制了派生类对象(用private限制的基类的成员,它们是存在的,只是我们不可见),不管是在派生类里面还是在派生类外面都不可以去访问基类中用private访问限定符限定的成员的,只可以在基类中自己使用。
第一种
第二种:
2>如果想让基类成员在类外面不能被访问,但是可以在派生类中被访问,可以将这个成员的方位限定符设为protected。(从这里就可以看出来protected(保护限定符)其实就是为了继承才设计出来的。)
3> 基类中的私有成员在子类(派生类)或者类外面是绝对无法被访问的,那基类的其他成员在派生类中的访问方式是什么呢?是:Min(成员在基类的访问限定符,继承方式)(public >protected> private)
举例:在上图中,基类中的成员变量_age是被protected访问限定符限制的,派生类在这里是以public继承方式去继承基类的,public和protected中protected权限最小,所以_age就相当于是被访问限定符限制protected限定在派生类中
4>使用关键词class时默认的继承方式是private,使用struct关键字是默认的继承方式是public,不过最好还是显示的写出继承方式,以增加代码的可读性。
5>我们在实际运用中一般都是用public的继承方式去继承的,几乎很少看见使用protected / private这两种继承方式去继承的,因为protected / private继承下来的成员都只能在派生类里面去使用,实际中扩展维护性也不强。
3.继承类模板
在这里我们尝试通过继承一个vector类模板来实现栈这个结构
将stack作为vector的派生类来生成的,我们在模拟实现stack的成员方法时,是直接通过复用基类的成员方法来进行实现的。
注意点:
1. 继承的时候,不止要写类名vector,还要写模板参数<T>.
namespace hou
{template<class T> //这个是模板参数class stack :public vector<T>//类名是vector,模板参数是T{};
}
2. 基类是类模板时,需要指定一下类域
namespace hou
{template<class T> //这个是模板参数class stack :public vector<T>//类名是vector,模板参数是T{public:void push(const T& x){vector<T>::push_back(x); //不能直接写push_back(x);// 基类是类模板时,需要指定⼀下类域, 否则编译报错:error C3861: “push_back”: 找不到标识符 // 因为stack<int>实例化时,也实例化vector<int>了(但需要注意的是,这里只是实例化了stack和vector这两个模板中的默认构造函数,其余成员函数均未进行实例化操作)//模版是按需实例化,push_back等成员函数未实例化,所以找不到 }};
}
我们每次复用基类的成员方法时,我们都需要使用作用域运算符来指定类域,虽然按道理来说,我们实例化出来派生类了,基类应该也就实例化出来了,事实确实如此,但是基类并不是一把实例化出来的,他的有些成员还没实例化出来,这时候我们就需要自己手动指示类域进行实例化了(如果我们不指明的话,编译器就会报未声明的报错),这就是我们之前所说的按需实例化。
#include<stdio.h>
#include<iostream>
#include<vector>
using namespace std;namespace hou
{template<class T> //这个是模板参数class stack :public vector<T>{public:void push(const T& x){vector<T>::push_back(x); //不能直接写push_back(x);}void pop(){vector<T>::pop_back();}const T& top(){return vector<T>::back();}bool empty(){return vector<T>::empty();}};
}int main()
{hou::stack<int> st;st.push(1);st.push(2);while (!st.empty()){std::cout << st.top() << " ";st.pop();}return 0;
}
二、基类和派生类之间的转换
public继承的派生类对象可以赋值给基类的对象 / 基类的指针 / 基类的引用。这里有个形象的说法叫做切片或者是切割。寓意把派生类中基类的那一部分切割出来,基类指针或引用指向的是派生类中切出来的基类那部分。
再回顾一下之前的类型转换:
double d=1.1;
const int& i = d;
这里不可以int& i = d;因为d的整型部分先赋给临时变量(临时变量有常性,意思是用const修饰了,所以将临时变量给给i,i必须是const修饰的int类型的对象,代码第二行)
1. 派生类(子类)对象可以赋值给(基类)父类的对象,调用父类中的拷贝构造函数即可实现。
student sobj; //student是子类
person p = sobj; //person是父类
2. 派生类(子类)对象可以赋值给基类(父类)指针,就相当于是把子类中父类的那部分切割出来,pp指针就指向切割出来的那部分的地址
person* pp = &sobj;
3.派生类(子类)对象可以赋值给基类(父类)引用。(把子类中父类的那一部分切割出来,p它所引用的就是从子类中切割出来的那一部分)
在图片中,红线框里并没有写const,但是没有报错,这说明中间没有产生临时变量。这里是编译器的特殊处理:赋值兼容转换。
(1)父类对象不能赋值给子类对象,会编译报错(少赋值给多?x)
三、继承中的作用域
在C++中,基类与派生类是两个不同的作用域(派生类是由基类继承下来的,但它们两个的作用域不是同一个,是独立的两个作用域)
隐藏规则:
1. 如果派生类和基类中的成员同名,那派生类就会屏蔽基类的同名成员,这叫做隐藏。但如果想在派生类中使用(和派生类同名的成员),也是可以的,直接指明父类的类域即可(基类::基类成员)
2. 对于成员函数,只要两者的函数名相同,那么两者就会构造隐藏关系【意思是:如果两个函数名相同,但是参数不同(void fun()和void fun(int i)),这两个在继承的情况下是隐藏关系。】
还需要回顾一下:什么是重载(需要满足的一个条件是:这两个函数在同一个作用域)【函数重载是指在一个作用域内,允许多个同名函数的存在,只要它们的形式参数(包括参数的数量、类型或顺序)有所不同即可。这种机制使得同一个函数名称能够执行不同的操作。】
四、派生类的默认成员函数
默认的意思就是指我们不写,编译器会变我们自动生成一个,那么在派生类中,这几个成员函数是如何生成的呢?(派生类本质也是一种类,只不过它与基类有着千丝万缕的关系)
派生类的构造函数
普通的类:在默认生成的时候:成员变量分为:内置类型(如果我们不写,它不确定,有可能随机值,也有可能初始化为0;如果有缺省值,就会用缺省值)和自定义类型(会调用这个自定义类型的默认构造,如果自定义类型没有默认构造,需要显式去写它的构造,在初始化列表定义初始化自定义类型)
1. 对派生类进行构造时,要记得,派生类有自己独有的成员,还有在父类里边儿共有的成员,(我们可以将子类中继承下来的父类成员,当作一个整体对象)默认构造:派生类独有的(内置和自定义)+父类成员(调用自己的默认构造)(如果父类没有默认构造,就需要自己显示地去写)
(派生类的构造函数)必须(调用基类的构造函数初始化基类那一部分的成员),如果基类没有默认的构造函数,则必须在派生类构造函数的初始化列表阶段显示调用
之前普通的类的情况:初始化列表中,按照声明的顺序进行初始化的,在这里是先初始化父类的。
派生类对象 初始化时,先调用基类构造,再调用派生类构造函数。(和析构相反)
拷贝构造函数
1. 如果不写拷贝构造,编译器默认生成的是什么样子的?
:子类成员(内置类型[默认拷贝构造的话是值拷贝]+自定义类型[默认构造的话会调用这个类型的拷贝构造])+父类成员(必须调用父类的拷贝构造)
默认拷贝构造完成的挺好,不需要我们插手。
那什么时候,派生类的拷贝构造需要自己写--->假设一个成员需要深拷贝
int* ptr = new int[10];
(指针是内置类型)内置类型的默认拷贝构造(浅拷贝)会造成两个指针指向同一个空间,析构的时候会崩溃。当需要深拷贝的时候,就需要自己写拷贝构造了。
接下来看,如何显式地写拷贝构造:
//父类中的拷贝构造 Person(const Person& p): _name(p._name){cout << "Person(const Person& p)" << endl;} //子类中的拷贝构造 Student(const Student& s): Person(s), _num(s._num),_address(s._address){cout << "Student(const Student& s)" << endl;}
派生类对象的拷贝构造函数,必须调用基类的拷贝构造来完成拷贝初始化。
赋值重载
如果没有需要深拷贝的,就不用自己写了
int main()
{Student s1( 18,"202306","hou");Student s3(99, "39393", "jing");s1 = s3;return 0;
}
//父类Person& operator=(const Person& p){cout << "Person operator=(const Person& p)" << endl;if (this != &p) //如果是自己给自己赋值的话,就直接返回自己就可以了,有效地节省了时间,大大地提高了运行效率_name = p._name;return *this;} //派生类Student& operator = (const Student& s){cout << "Student& operator= (const Student& s)" << endl;if (this != &s) {// 构成隐藏,所以需要显⽰调⽤Person::operator =(s); //调用父类的operator=()函数去给父类中的那一部分成员变量进行赋值操作_num = s._num;_address = s._address;}return *this;}
派生类的operator=()函数隐藏了基类的operator=()函数,所以需要我们去显示调用基类的operator=()函数,就必须要去指定基类的作用域,只有这样才能访问基类中的那个operator=()赋值构造函数
析构函数
析构时,必须确保先析构子,再析构父
派生类的析构函数会在被调用完成后,自动去调用基类的析构函数去清理基类成员【所以,不需要显示调用父类析构】。因为只有这样才能保证派生类对象(先清理派生类成员),(再去清理基类成员的顺序。
~student()
{_num = 0;_address=nullptr;person::~person();//我们在去调用父类的析构函数去清理父类中的成员变量时,我们一定要在这里指定作用域,因为析构函数在最后它都会被处理成desructor()这个函数,进而会构成隐藏。
}//我们在显示调用父类的析构函数时,一定要注意要等到将子类中的成员变量全部清理完了以后,再去调用父类的析构函数去清理父类中的那一部分成员变量,通过我们前面所学过的知识可知,先构造的后析构,由于person是被继承过来的,就相当于是先有person,再有student,因此后析构person(也就是person)。
因为多态中一些场景导致析构函数需要重写,而重写的条件之一就是函数名相同(这个写一篇博客会讲到,这里我们只需要明白析构函数会被进行特殊处理即可),那么编译器在这里会对析构函数进行一个特殊处理的操作,将析构函数名处理成destructor(),所以基类的析构函数在不加virtual的情况下,派生类析构函数和基类析构函数构成隐藏关系。
实现一个不能被继承的类
方法一:将构造函数私有化(C++98)
为什么将构造函数私有化就不能被继承了呢?
派生类的构造必须调用基类的构造函数,但是基类的构造函数被私有化private,派生类看不见就自然不能去调用了,进而就无法实例化出对象了。
方法二:final关键字
C++11中增加了一个final关键字,final修改了基类后,派生类就不可以继承基类了。
继承与友元
友元关系不能被继承
回顾:通过使用friend关键字,可以让(一个类或函数)访问(另一个类)中的私有成员(私有成员变量和私有成员函数)和保护成员。Display是父类的友元,那么在Display这个函数里,可以访问父类里的私有和保护成员。
友元关系不能被继承,简单理解:父类的朋友,并不是,子类的朋友。所以,在Display中,并不能访问子类的私有和保护成员。
继承与静态成员
这个地址不一样说明:非静态成员num的地址是不一样的,派生类继承了这个num变量,父类和派生类各有一份。
那静态成员呢?
回顾:
- 静态成员变量是属于整个类的,而不是某个具体对象的属性。静态成员变量只有一份副本存在于内存中。
- 声明方式: 在类内部使用 static 关键字声明静态成员变量。
- 初始化位置: 静态成员变量必须在类外部进行初始化
基类 定义了一个 static静态成员变量,则整个继承体系里面只有一个这样的成员。无论派生出多少个派生类,都是只有一个static成员实例。
到静态成员 _one 的地址是⼀样的 ,说明派生类和基类共用同一份静态成员。
公有的情况下,父类和派生类,指定类域都可以访问静态成员
多继承及其菱形继承的问题
首先我们先搞清楚,什么是单继承和多继承。
单继承: 一个派生类只有一个直接基类时,称这个继承关系为单继承
在这个图片中,它们的关系是单继承。PostGraduate只是间接继承了Person,PostGraduate的直接基类只有一个,就是Student。
一个派生类有两个及以上直接基类时,称这个继承关系为多继承
多继承对象在内存中的模型是:先继承的基类在前面,后面继承的基类在后面,派生类成员放到最后面。
比如:Assistant这个类同时继承了一个Student类和Teacher类,继承的父类之间分别用","隔开,Assistant这个类先继承的是Student类,后继承的是Teacher类。
class Assistant :public Student, public Teacher
菱形继承是多继承的一种特殊情况。
多继承很容易造成菱形继承。(如果这两个基类Student和Teacher,都有一个共同的基类,就会形成菱形继承)
菱形继承有不太好的方面:菱形继承有数据冗余和二义性的问题,在Assistant的对象中Person成员会有两份,这样的话,会大大增加空间的消耗。
怎么看出来有两份呢?
从图中可以看出,派生类Student和Teacher当中,都有基类Person。有两份Person就会导致数据冗余(有两份,浪费空间)和二义性(有两份Person,不知道想访问哪一个)
暂时解决二义性问题的办法:指明基类
class Person
{
public:
string _name; // 姓名
};
class Student : public Person
{
protected:int _num; //学号
};
class Teacher : public Person
{
protected:int _id; // 职⼯编号
};
class Assistant : public Student, public Teacher
{
protected:string _majorCourse; // 主修课程
};
int main()
{// 编译报错:error C2385: 对“_name”的访问不明确 Assistant a;a._name = "peter";// 需要显⽰指定,访问哪个基类的成员,可以解决⼆义性问题,但是数据冗余问题⽆法解决 a.Student::_name = "xxx";a.Teacher::_name = "yyy";return 0;
}
虚继承
C++中的多继承虽然说在实际中可能应用挺广泛的,但是它应运而生的菱形继承所带来的问题有点多,这也是java中没有多继承的一个主要原因,但是C++对于之前已经创立的语法是不可以进行修改的,我们只能够向前兼容,于是我们就又创建了一个新的语法——虚继承。
虚继承就是在我们派生类继承基类的时候,在继承方式前加上一个关键字virtual即可。在C++中,虚继承(Virtual Inheritance)是一种用于解决多继承中重复继承问题的机制。其主要意义在于确保共同的基类只被继承一次,避免数据冗余和访问歧义。
class Teacher :virtual public Person//teacher类虚继承了person类
c++的语法复杂,其实多继承就是一个体现。有了多继承,就存在菱形继承,有了菱形继承就有了虚拟菱形继承,底层实现就会很复杂,性能方面就也会有一些损失,所以最好不要设计出菱形继承。多继承可以认为是C++的缺陷之一,后来的一些编程语言中基本上都没有多继承。
组合和继承
- public继承是⼀种is-a的关系。也就是说每个派⽣类对象都是⼀个基类对象。
- 组合是一种has-a的关系。假设B组合了A,每个B对象中都有一个A对象。
- 继承允许你根据基类的实现来定义派生类的实现。这种通过生成派生类的复用通常被称为⽩箱复用(这里既然说到了这个白盒测试和黑盒测试,我们先来简单地了解一下相关的知识:1 黑盒测试:不了解底层实现,是从功能的角度进行测试的,2 白盒测试(相对于黑盒测试更难):了解底层实现(代码实现),从代码运行的逻辑角度进行测试)。术语“⽩箱”是相对可视性⽽⾔:在继承⽅式中,基类的内部细节对派⽣类可⻅ 。继承⼀定程度破坏了基类的封装,基类的改变,对派生类有很⼤的影响。派生类和基类间的依赖关系很强,耦合度⾼。
- 对象组合是类继承之外的另⼀种复⽤选择。新的更复杂的功能可以通过组装或组合对象来获得。对象组合要求被组合的对象具有良好定义的接⼝。这种复用⻛格被称为⿊箱复用(black-box reuse),因为对象的内部细节是不可⻅的。对象只以“⿊箱”的形式出现。 组合类之间没有很强的依赖关系,耦合度低。优先使用对象组合有助于你保持每个类被封装。
- 优先使用组合,⽽不是继承。实际尽量多去用组合,组合的耦合度低,代码维护性好。不过也不太那么绝对,类之间的关系就适合继承(is-a)那就⽤继承,另外要实现多态,也必须要继承。类之间的关系既适合用继承(is-a)也适合组合(has-a),就用组合。
相关文章:
C++之“继承”
继续开始关于C相关的内容。C作为面向对象的语言,有三大特性:封装,继承,多态。 这篇文章我们开始学习:继承。 一、继承的概念和定义 1. 继承的概念 什么是继承呢? 字面意思理解来看:继承就是…...
java集成telegram机器人
java集成telegram机器人 最近项目需要集成telegram机器人,来实现消息推送功能,以此记录下。 1.创建telegram账号 没有账号的可以去某宝上买一个,千万不要用自己的手机号86去注册,你懂得。 2. 打开BotFather对话创建机器人获取…...
从代码学习深度学习 - 单发多框检测(SSD)PyTorch版
文章目录 前言工具函数数据处理工具 (`utils_for_data.py`)训练工具 (`utils_for_train.py`)检测相关工具 (`utils_for_detection.py`)可视化工具 (`utils_for_huitu.py`)模型类别预测层边界框预测层连接多尺度预测高和宽减半块基础网络块完整的模型训练模型读取数据集和初始化…...
因子分析——数学原理及R语言代码
这里写自定义目录标题 因子分析参数估计方法主成分法主因子法 因子旋转 代码实现Reference 因子分析 FactorAnalysis的目的是从多个高度相关的观测变量中提取出少数几个LatentFactor,这些因子代表了变量背后的共通结构,从而实现降维并提升可解释性。 假…...
CSS3 过渡与动画
在现代网页设计中,平滑的过渡和生动的动画效果已成为提升用户体验不可或缺的元素。CSS3 为我们提供了强大的 transition 和 animation 属性,让开发者能够轻松实现各种视觉效果。本文将深入探讨这两大功能的特性和应用场景。 一、CSS3 过渡(T…...
【JAVA】【重试间隔】多线程中两种常见的重试间隔
一、前言 报!! 小南啊,今日有个小任务交给你去办。就是程序中有个数据处理,总是会出错,不知道是什么原因,你去处理一下! 二、主题 围绕数据处理问题去看,从中发现,是因为…...
在现代Web应用中集成 PDF.js (pdfjs-dist 5.2 ESM): 通过 jsdelivr 实现动态加载与批注功能的思考
PDF 文档在现代 Web 应用中越来越常见,无论是作为文档预览、报告展示还是在线编辑的载体。Mozilla 的 PDF.js 是一个功能强大的 JavaScript 库,它使得在浏览器端渲染和显示 PDF 文件成为可能,无需依赖原生插件。 本文将深入探讨如何在你的项…...
android ViewModel liveData无法监听之多线程下activityViewModels不安全
我们一般的,会遇到liveData无法监听到结果,可能存在主要2种可能: liveData没有正确注册;liveData连续多次设置值,中间的值,会被丢弃,但最后一次是能监听到的。 但是我们容易忽略一种case&…...
【即插即用涨点模块】DSConv动态蛇形卷积:自适应聚焦细长弯曲的局部结构特征,助力分割高效提点【附源码+注释】
《------往期经典推荐------》 一、AI应用软件开发实战专栏【链接】 项目名称项目名称1.【人脸识别与管理系统开发】2.【车牌识别与自动收费管理系统开发】3.【手势识别系统开发】4.【人脸面部活体检测系统开发】5.【图片风格快速迁移软件开发】6.【人脸表表情识别系统】7.【…...
守护数字家园:个人博客安全防护指南
前言 在之前的文章《WordPress个人博客搭建(一)》《WordPress个人博客搭建(二)》《WordPress个人博客搭建(三)》中,我们已经在非凡云云服务器上,借助1Panel搭建起属于自己的数字庭院…...
课外活动:简单了解原生测试框架Unittest前置后置的逻辑
简单了解原生测试框架Unittest前置后置的逻辑 一、测试框架执行顺序解析 1.1 基础执行流程 import unittestclass A(unittest.TestCase):classmethoddef setUpClass(cls):print(f"【CLASS START】{cls.__name__}")def setUp(self):print(f"【TEST START】{se…...
带你玩转 Flink TumblingWindow:从理论到代码的深度探索
0.前言 在深入探讨 TumblingWindow 之前,我们先来了解一下流处理或流计算中“窗口”的基本概念。在数据流中,源会持续不断地生成数据,因此计算最终值是不可行的。 在大多数用例中,为了获取有意义的信息,最好使用两种方…...
Java线程安全问题深度解析与解决方案
一、线程安全问题的本质 并发编程的核心挑战:当多个线程同时访问共享资源时,由于操作系统的抢占式调度特性,可能导致不可预期的结果。这种因非原子操作和竞态条件引发的数据不一致问题,称为线程安全问题。 二、经典线程安全问题案…...
python实现的音乐播放器
python实现的音乐播放器 音乐播放器,原来写过一个简陋的例子,可见 https://blog.csdn.net/cnds123/article/details/137874107 那个不能拖动播放进度条上的滑块到新的位置播放。下面介绍的可以拖动播放进度条上的滑块到新的位置播放。 简单实用的音乐播放器 这个简单实用的…...
SMT贴片工艺核心优化与生产实践
内容概要 作为现代电子制造的核心环节,SMT贴片工艺的优化直接决定了产品可靠性与生产效率。本文系统性梳理工艺链中的关键控制点,从锡膏印刷精度到回流焊温度曲线,再到AOI检测技术升级,形成覆盖全流程的优化框架。针对行业普遍存…...
趣味编程:爱心
概述:五月十一号就是母亲节了,本篇博客主要是为母亲所写,这是属于程序员的浪漫,这篇博客是对母亲这么多年无微不至爱的情书。 目录 1. 效果展示 2. 源码展示 3. 代码逻辑详解 3.1 头文件与常量定义 3.2 心形曲线参数方程 3.…...
C语言—指针2
1. const 修饰变量 1.1 const修饰变量 变量被const修饰时,变量此时为常变量,本质为常量,语法上不可被修改,但是如果此时需要修改变量值,可以通过指针的方式修改。 虽然此时通过指针的方式确实修改了变量的值ÿ…...
66、微服务保姆教程(九)微服务的高可用性
微服务的高可用性与扩展 服务的高可用性 集群搭建与负载均衡。服务的故障容错与自愈。分布式事务与一致性 分布式事务的挑战与解决方案。使用 RocketMQ 实现分布式事务。微服务的监控与可观测性 metrics 和日志的收集与分析。sentinel 的监控功能。容器化与云原生 将微服务部署…...
主场景 工具栏 植物卡牌的渲染
前置知识:使用easyx图形库 1.IMAGE内存变量存储的是一张位图(图像),存储了像素数据(颜色,尺寸等) 2.loadimage(&变量名,"加载的文件路径")表示从文件中加载图像到变量中 3. saveimage("文件路径", &变…...
超详细!RxSwift 中的 BehaviorRelay 使用教程(含原理 + 示例 + 实战)
目录 前言 1.什么是 BehaviorRelay 2.基本使用方式 3.BehaviorRelay的常用API 4.BehaviorRelay 和其它类型的对比 5.BehaviorRelay的使用场景 1.绑定UITableView 2.MVVM 场景下使用 BehaviorRelay 6.使用注意事项以及建议 1.注意事项 2.使用建议总结 7.推荐阅读 前…...
【软件测试学习day7】Junit5
Junit 是单元测试框架,本期掌握 Junit5 的基础用法。 1. 注解 首先引入 Junit 依赖: <dependency><groupId>org.junit.jupiter</groupId><artifactId>junit-jupiter-api</artifactId><version>5.9.1</version…...
前端实战中的单例模式:以医疗药敏管理为例
目录 一、什么是单例模式?1. 状态共享性 —— 数据唯一,任意访问,任意修改2. 生命周期控制性 —— 自己掌控何时创建、何时销毁 二、实战分析:医疗药敏管理系统中的单例应用三、其他场景示例单例实现:ConfigManager.ts…...
如何在 Logback 日志框架中加入链路 ID
在 Logback 日志框架中加入链路 ID,能有效将同一条链路的日志串联起来,便于追踪和排查问题。 1. 生成和管理链路 ID 要保证在整个请求链路里都能获取到链路 ID,可借助 ThreadLocal 来实现。以下是一个简单的工具类示例: import…...
晶振:智能设备的“心跳”如何支撑5G与航天
在现代科技飞速发展的时代,智能设备已深度融入我们生活的方方面面,而晶振,作为智能设备的“心跳”,正默默发挥着不可替代的关键作用。无论是翱翔太空的神舟飞船,还是人们手中须臾不离的智能手机,亦或是推动…...
【HarmonyOS 5】App Linking 应用间跳转详解
目录 什么是 App Linking 使用场景 工作原理 如何开发 1.开通 App Linking 2.确定域名 3.服务端部署 applinking.json 文件 4.AGC绑定域名 5.项目配置 6.组装聚合链接 7.解析聚合链接中的参数 其他 如何获取应用ID 什么是 App Linking App Linking 是一款创建跨…...
neo4j官方示例
目录 一、准备数据 1.执行查看结果 二、操作 1.find 单个节点 2.同上,已某个属性去查询 3. 指定查询个数 4.条件查询 5.查询某个人出演的电影汇总 6.查询tom出演的电影中,还有其他演员的信息。 7.查询跟电影(Cloud Atlas)有关的演员࿰…...
基于vueflow可拖拽元素的示例(基于官网示例的单文件示例)
效果图 代码 <template><div style"width: 100%;height: calc(100vh - 84px)"><VueFlow :nodes"nodes" :edges"edges" drop"onDrop" dragover"onDragOver" dragleave"onDragLeave"><div cl…...
minio单点登录与集成(免密)
需求:系统A里,需要实现与MINIO单点登录集成,也就是说,登录了系统A,在访问MINIO时不需要再输入用户密码就可以直接访问。 具体场景如下: 在系统A的一个页面里,配置一个按钮链接,点击…...
深入理解 Docker 网络原理:构建高效、灵活的容器网络
在现代软件开发中,Docker 已经成为了容器化技术的代名词,广泛应用于开发、测试和生产环境。Docker 使得开发者能够将应用及其依赖打包成一个轻量级的容器,并通过 Docker 容器化技术来实现高效的部署与管理。 然而,在日常使用 Dock…...
Hutool中的Pair类详解
1. Pair类概述 Hutool工具库中的Pair类是一个简单的键值对数据结构,用于存储两个相关联的对象。它类似于Map的Entry,但更加轻量级,适用于需要临时存储两个相关联数据的场景。 2. Pair类的主要特点 简单轻量:不依赖复杂的数据结…...
没有Mac,我是怎么上传IPA到App Store的?
没有Mac,我是怎么上传IPA到App Store的? 最近赶一个小项目上线,写的是一个Flutter做的App。安卓版本一晚上搞定,iOS上架却差点把人整崩。 不是我技术菜,是实在太麻烦了。最关键的,是我这台Windows笔电根本…...
RISC-V hardfault分析工具,RTTHREAD-RVBACKTRACE
RV BACKTRACE 简介 本文主要讲述RV BACKTRACE 的内部主要原理 没有接触过rvbacktrace可以看下面两篇文章,理解一下如何使用RVBACKTRACE RVBacktrace RISC-V极简栈回溯组件:https://club.rt-thread.org/ask/article/64bfe06feb7b3e29.html RVBacktra…...
c语言if else语句格式(非常详细)
在C语言中,if else 语句是一种常用的条件控制结构,用于根据不同条件执行不同的代码块。 if-else 语句的基本格式 if-else 语句的基本格式如下: if (条件) { // 如果条件为真,执行这里的代码 } else { // 如果条件为假&a…...
Logback官方文档翻译章节目录
Logback官方文档翻译章节目录 第一章 Logback简介 第二章 Logback的架构(一) Logback的架构(二) Logback的架构(三) 持续更新中…...
按摩椅的机芯类型和材质
按摩椅的机芯类型和材质是影响其按摩效果、使用寿命以及舒适度的重要因素。下面我将从这两个方面详细为你解析: 一、按摩椅机芯类型 按摩椅的“机芯”相当于它的“心脏”,决定了按摩手法、力度、覆盖范围等关键性能。 常见机芯类型(按技术发…...
HarmonyOS-hdc远程网络方式连接设备
hdc工具使用手册 1 hdc简介 hdc(OpenHarmony Device Connector)是为开发人员提供的用于设备连接调试的命令行工具,pc端开发机使用命令行工具hdc,该工具需支持部署在Windows/Linux/Mac等系统上与OpenHarmony设备(或模…...
秋招准备——2.跨时钟相关
格雷码异步FIFO跨时钟域处理 格雷码 一、格雷码规律 相邻性:相邻两个数的格雷码只有一位不同,例如: 0000 → 0001(仅最低位变化)0001 → 0011(仅次低位变化)0011 → 0010(仅最低位…...
【开源版】likeshop上门家政系统PHP版全开源+uniapp前端
一.系统介绍 likeshop_上门家政系统,PHP版本更新至2.1.1最新版,全开源,适用于上门家政场景,系统拥有用户端、师傅端、无论运营还是二开都是性价比极高的100%开源家政系统。 二.搭建环境-教程 系统环境:CentOS、 运行…...
Memgraph 的安装教程
目录 Memgraph 安装步骤1. 使用 Docker 安装 Memgraph2. 使用 Memgraph Lab3. 使用 Python 客户端连接 Memgraph Memgraph 安装步骤 1. 使用 Docker 安装 Memgraph Memgraph 可以通过 Docker 快速安装和运行。以下是使用 Docker 安装 Memgraph 的步骤: 安装 Docke…...
华为网路设备学习-21 路由过滤(filter-policy)
一、路由过滤(filter-policy) 1、用于控制路由更新、接收的一个工具 2、只能过滤路由信息,无法过滤LSA 二、路由过滤(filter-policy)与动态路由协议 1、距离矢量路由协议 RIP动态路由协议 交换的是路由表࿰…...
Mac 平台 字体Unicode范围分析器
字体Unicode范围分析器 #include <CoreText/CoreText.h> // CoreText框架头文件,用于字体处理 #include <CoreFoundation/CoreFoundation.h> // CoreFoundation框架头文件 #include <stdio.h> // 标准输入输出 #include…...
Android不能下载Gradle,解决方法Could not install Gradle distribution from.......
外网下载速度太慢导致失败,换成国内镜像,可加速下载: 官网地址:https://services.gradle.org/distributions/ 腾讯云镜像 Gradle下载地址:https://mirrors.cloud.tencent.com/gradle/ 阿里云镜像 Gradle下载地址&…...
树状数组的操作问题--Python
树状数组的操作问题 一、问题引入二、解题步骤1.思维导图2.解题步骤 三、代码实现1.代码2.复杂度分析 四、个人总结 一、问题引入 请编写程序,实现树状数组区间求前缀和、单点修改的操作。 输入格式: 输入首先给出一个正整数 n(2≤n<10^…...
FEKO许可限制
随着科技的飞速发展,电磁仿真软件在多个领域发挥着越来越重要的作用。FEKO作为一款业界领先的电磁仿真软件,广泛应用于通信、雷达、航空航天、电子对抗等领域。然而,为了确保软件使用的合规性与高效性,FEKO设定了相应的许可限制。…...
第5章 深度学习和卷积神经网络
深度学习是人工智能的一种实现方法。本章我们将考察作为深度学习的代表的卷积神经网络的数学结构。 5-1小恶魔来讲解卷积神经网络的结构 深度学习是重叠了很多层的隐藏层(中间层)的神经网络。这样的神经网络使隐藏层具有一定的结构,从而更加…...
window 显示驱动开发-处理内存段(一)
视频内存管理器 (VidMm) 负责管理 GPU 的地址空间。 在此之前,内核模式显示微型端口驱动程序 (KMD) 必须通过使用内存段将 GPU 的地址空间描述为 VidMm。 KMD 创建内存段以概括和虚拟化视频内存资源。 它可以根据硬件支持的存储器类型(例如,…...
QT实现曲线图缩放、拖拽以及框选放大
.h文件 protected: void saveAxisRange();void wheelEvent(QWheelEvent *event) override;void mousePressEvent(QMouseEvent *event) override;void mouseMoveEvent(QMouseEvent *event) override;void mouseReleaseEvent(QMouseEvent *event) override;private:QPoint m_…...
龙虎榜——20250508
上证假阴包阳的走势,量能较昨天有萎缩,在前期压力附近~ 深证这两天假阴包阳的走势,60分钟未突破昨天的高点,缺口也未补等待明天的选择~ 2025年5月8日龙虎榜行业方向分析 一、核心行业方向 军工航天(政策催化地缘驱动…...
SEMI E40-0200 STANDARD FOR PROCESSING MANAGEMENT(加工管理标准)-(三)完结
10 消息服务详情 10.1 本章定义实现加工管理概念所需的消息服务。这些消息已在第8.1节中初步介绍。 协议无关性:这些服务独立于所使用的消息协议,可映射至SECS-II(SEMI E5)或其他类似协议。 10.1.1 消息服务定义内容包括&#…...
算法竞赛进阶指南.次小生成树
目录 题目算法标签: K r u s k a l Kruskal Kruskal, M S T MST MST, 倍增优化, l c a lca lca思路代码*警示后人 题目 356. 次小生成树 算法标签: K r u s k a l Kruskal Kruskal, M S T MST MST, 倍增优化, l c a lca lca 思路 因为要求的是严格次小生成树, 假设最…...