★ C++进阶篇 ★ 继承

news2024/9/23 16:24:14

Ciallo~(∠・ω< )⌒☆ ~ 今天,我将继续和大家一起学习C++进阶篇第一章----继承 ~

目录

一  继承的概念及定义

1.1 继承的概念

1.2 继承定义

1.2.1 定义格式

1.2.2 继承⽗类成员访问⽅式的变化 

1.3 继承类模板

二  父类和子类对象赋值兼容转换

三  继承中的作用域

3.1 隐藏规则

3.2 继承作用域相关选择题

四  子类的默认成员函数

4.1 4个常见默认成员函数

4.2 实现⼀个不能被继承的类

五  继承与友元

六  继承与静态成员

七  多继承及其菱形继承问题

7.1 继承模型

7.2 虚继承

7.3 IO库中的菱形虚拟继承

7.4 多继承中指针偏移问题

八  继承和组合


一  继承的概念及定义

1.1 继承的概念

        继承(inheritance)机制是⾯向对象程序设计使代码可以复用的最重要的⼿段,它允许我们在保持原有类特性的基础上进⾏扩展,增加⽅法(成员函数)和属性(成员变量),这样产⽣新的类,称子类。继承呈现了⾯向对象程序设计的层次结构,体现了由简单到复杂的认知过程。以前我们接触的函数层次的复⽤,继承是类设计层次的复用

        比如我们要设计两个类Student和Teacher,Student和Teacher都有姓名/地址/电话/年龄等成员变量,都有identity身份认证的成员函数,设计到两个类⾥⾯就是冗余的。当然他们也有⼀些不同的成员变量和函数,⽐如⽼师独有成员变量是职称,学⽣的独有成员变量是学号;学⽣的独有成员函数是学习,⽼师的独有成员函数是授课。那么,我们就可以把公共的成员都放到Person类中,Student和teacher都继承Person,就可以复⽤这些成员,就 不需要重复定义了,省去了很多⿇烦。

class Person
{
public:
	// 进⼊校园/图书馆/实验室刷⼆维码等⾝份认证
	void identity()
	{
		cout << "void identity()" << _name << endl;
	}
protected:
	string _name = "张三"; // 姓名
	string _address; // 地址
	string _tel; // 电话
	int _age = 18; // 年龄
};

class Student : public Person
{
public:
	// 学习
	void study()
	{
		// ...
	}
protected:
	int _stuid; // 学号
};

class Teacher : public Person
{
public:
	// 授课
	void teaching()
	{
		//...
	}
protected:
	string title; // 职称
};

1.2 继承定义

1.2.1 定义格式

下⾯我们看到Person是⽗类,也称作基类。Student是⼦类,也称作派⽣类。

1.2.2 继承⽗类成员访问⽅式的变化 
类成员 \ 继承方式public继承protected继承private继承
父类的public成员子类的public成员子类的protected成员子类的private成员
父类的protected成员子类的protected成员子类的protected成员子类的private成员
父类的private成员在子类中不可见在子类中不可见在子类中不可见

(父类的private成员在子类中不可见,其他取小的) 

  • 1. 父类private成员在子类中无论以什么方式继承都是不可见的。这里的不可⻅是指父类的私有成员还是被继承到了子类对象中,但是语法上限制⼦类对象不管在类里面还是类外⾯都不能去访问它。
  • 2. ⽗类private成员在⼦类中是不能被访问,如果父类成员不想在类外直接被访问,但需要在子类中能访问,就定义为protected。可以看出保护成员限定符是因继承才出现的。
  • 3. ⽗类的其他成员在⼦类的访问⽅式 == Min(成员在⽗类的访问限定符,继承⽅式),public > protected > private
  • 4. 使用关键字class时默认的继承⽅式是private,使⽤struct时默认的继承方式是public,不过最好显示的写出继承⽅式。
  • 5. 在实际运用中⼀般使用都是public继承,几乎很少使用protetced/private继承,也不提倡使⽤protetced/private继承,因为protetced/private继承下来的成员都只能在⼦类的类⾥⾯使⽤,实际中扩展维护性不强。

1.3 继承类模板

有了继承,我们就可以用继承实现栈

#define Container std::deque
// stack和vector的关系,既符合is-a,也符合has-a
template<class T>
class stack : public Container<T>
{
public:
	void push(const T& x)
	{
		// 父类是类模板时,需要指定⼀下类域,否则编译报错:error C3861: “push_back”: 找不到标识符
		// 因为stack<int>实例化时,也实例化vector<int>了
		// 但是模版是按需实例化,push_back等成员函数未实例化,所以找不到
		Container<T>::push_back(x);
	}
	void pop()
	{
		Container<T>::pop_back();
	}
	const T& top()
	{
		return Container<T>::back();
	}
	bool empty()
	{
		return Container<T>::empty();
	}
};

二  父类和子类对象赋值兼容转换

  • public继承的子类对象 可以赋值给 父类的对象 / 父类的指针 / 父类的引用。这⾥有个形象的说法叫切⽚或者切割。寓意把子类中父类那部分切来赋值过去。
  • 父类对象不能赋值给子类对象。
  • 父类的指针或者引用可以通过强制类型转换赋值给⼦类的指针或者引用。但是必须是⽗类的指针是指向⼦类对象时才是安全的。这⾥父类如果是多态类型,可以使用RTTI(Run-Time Type Information)的dynamic_cast 来进⾏识别后进⾏安全转换。

class Person
{
public:
	string _name; // 姓名
	string _sex; // 性别
	int _age; // 年龄
};
class Student : public Person
{
public:
	int _No; // 学号
};
int main()
{
	Student sobj;
	// 1.子类对象可以赋值给父类对象/指针/引用
	// 不产生常性临时变量,直接切给父类
	Person pobj = sobj;
	Person* pp = &sobj;
	Person& rp = sobj;
	rp._name = "澄岚";

	//2.父类对象不能赋值给⼦类对象,这里会编译报错
	//sobj = pobj; // <-
	return 0;
}

三  继承中的作用域

3.1 隐藏规则

  • 1. 在继承体系中⽗类和⼦类都有独立的作用域
  • 2. ⼦类和⽗类中有同名成员,⼦类成员将屏蔽⽗类对同名成员的直接访问,这种情况叫隐藏。(在⼦类成员函数中,可以使⽤用父类::父类成员 显⽰访问)
  • 3. 需要注意的是如果是成员函数的隐藏,只需要函数名相同就构成隐藏
  • 4. 注意在实际中在继承体系⾥⾯最好不要定义同名的成员。
class Person
{
protected:
	string _name = "⼩李⼦"; // 姓名
	int _num = 111; // ⾝份证号
};
class Student : public Person
{
public:
	void Print()
	{
		cout << " 姓名:" << _name << endl;
		cout << " ⾝份证号:" << Person::_num << endl;
		cout << " 学号:" << _num << endl;
	}
protected:
	int _num = 999; // 学号
};
int main()
{
	Student s1;
	s1.Print();
	return 0;
};

上程序Student的_num和Person的_num构成隐藏关系,输出999 。

3.2 继承作用域相关选择题

class A
{
public:
	void fun()
	{
		cout << "func()" << endl;
	}
};
class B : public A
{
public:
	void fun()
	{
		cout << "func()" << endl;
	}
};
int main()
{
	B b;
	b.fun();
	return 0;
};

3.2.1 A和B类中的两个func构成什么关系( B ) A. 重载 B. 隐藏 C.没关系

E: 只需要函数名相同就构成隐藏

3.2.2 下面程序的编译运行结果是什么( C ) A. 编译报错 B. 运行报错 C. 正常运行

E: 会调子类中的func( )

class A
{
public:
	void fun()
	{
		cout << "func()" << endl;
	}
};
class B : public A
{
public:
	void fun(int i)
	{
		cout << "func(int i)" << endl;
	}
};
int main()
{
	B b;
	b.fun();
	return 0;
};

改为此程序则编译报错,b.fun()需要加父类类域。

b.A::fun();

四  子类的默认成员函数

4.1 4个常见默认成员函数

6个默认成员函数,默认的意思就是指我们不写,编译器会变我们⾃动⽣成⼀个,那么在子类中,这几个成员函数是如何⽣成的呢?

我们用继承了Person类的student类来观察一下:

class Person
{
public:
	Person(const char* name = "zmcl")
		: _name(name)
	{
		cout << "Person()" << endl;
	}
	Person(const Person& p)
		: _name(p._name)
	{
		cout << "Person(const Person& p)" << endl;
	}
	Person& operator=(const Person& p)
	{
		cout << "Person operator=(const Person& p)" << endl;
		if (this != &p)
			_name = p._name;
		return *this;
	}
	~Person()
	{
		cout << "~Person()" << endl;
	}
protected:
	string _name; // 姓名
};
  • 1. 子类的构造函数必须调用父类的构造函数初始化父类的那⼀部分成员。如果父类没有默认的构造函数,则必须在子类构造函数的初始化列表阶段显⽰调用。
  • 2. 子类对象初始化先调用父类构造再调子类构造。

 默认生成的构造函数的行为
 1、内置类型->不确定
 2、自定义类型->调用默认构造
 3、继承父类成员看做一个整体对象,要求调用父类的默认构造

Student(const char* name, int num)
	: Person(name) // 调用父类的默认构造
	, _num(num)
{
	cout << "Student()" << endl;
}
  • 3. 子类的拷贝构造函数必须调用父类的拷贝构造完成父类的拷贝初始化。
Student(const Student& s)
	: Person(s) // 调用父类的拷贝构造
	, _num(s._num)
{
	cout << "Student(const Student& s)" << endl;
}

严格说Student拷贝构造默认生成的就够用了,如果有需要深拷贝的资源,才需要自己实现。 

  • 4. 子类的operator=必须要调用父类的operator=完成父类的复制。需要注意的是⼦类的operator=隐藏了父类的operator=,所以显⽰调用父类的operator=,需要指定父类作用域。
Student& operator= (const Student& s)
{
	cout << "Student& operator= (const Student& s)" << endl;
	if (this != &s)
	{
		// 构成隐藏,所以需要显⽰调⽤
		Person::operator =(s);
		_num = s._num;
	}
	return *this;
}

 也可以不用写~~

  • 5. 子类的析构函数会在被调用完成后自动调用父类的析构函数清理父类成员。因为这样才能保证子类对象先清理子类成员再清理父类成员的顺序
  • 6. 子类对象析构清理先调用子类析构再调父类的析构
  • 7. 因为多态中⼀些场景析构函数需要构成重写,重写的条件之⼀是函数名相同。那么编译器会对析构函数名进⾏特殊处理,处理成destructor(),所以父类析构函数不加 virtual的情况下,子类析构函数和父类析构函数构成隐藏关系
// 严格说Student析构默认生成的就够用了
// 如果有需要显示释放的资源,才需要自己实现
// 析构函数都会被特殊处理成destructor()
~Student()
{
    cout << "~Student()" << endl;
    // 子类的析构和父类析构函数也构成隐藏关系
	// 规定:不需要显示调用,子类析构函数之后,会自动调用父类析构
	// 这样保证析构顺序,先子后父,显示调用取决于实现的人,不能保证先子后父
}

4.2 实现⼀个不能被继承的类

方法1:父类的构造函数私有,子类的构成必须调⽤父类的构造函数,但是父类的构成函数私有化以后,子类看不⻅就不能调⽤了,那么子类就⽆法实例化出对象。

方法2:C++11新增了⼀个final关键字,final修改父类,子类就不能继承了。

// C++11的⽅法
class Base final
{
public:
	void func1() { cout << "Base::func1" << endl; }
protected:
	int _a = 1;
private:
	// C++98的⽅法
	/*Base()
	{}*/
};
class Derive :public Base //继承不了
{
	void func2() { cout << "Derive::func2" << endl; }
protected:
	int _b = 2;
};

五  继承与友元

友元关系不能继承,也就是说父类友元不能访问子类私有和保护成员 。

class Student;// 前置声明
class Person
{
public:
	friend void Display(const Person& p, const Student& s);// 友元关系不能被继承
protected:
	string _name; // 姓名
};
class Student : public Person
{
	//friend void Display(const Person& p, const Student& s); // <-解决⽅案
protected:
	int _stuNum; // 学号
};
void Display(const Person& p, const Student& s)
{
	cout << p._name << endl;
	cout << s._stuNum << endl;// 编译报错:error C2248: “Student::_stuNum”: 无法访问 protected 成员
	// 解决⽅案:Display也变成Student 的友元即可
}

六  继承与静态成员

父类定义了static静态成员,则整个继承体系里面只有⼀个这样的成员。无论派⽣出多少个子类,都只有⼀个static成员实例

class Person
{
public:
	string _name;
	static int _count;
};
int Person::_count = 0;
class Student : public Person
{
protected:
	int _stuNum;
};
int main()
{
	Person p;
	Student s;
	// 这⾥的运行结果可以看到⾮静态成员_name的地址是不⼀样的
	// 说明⼦类继承下来了,父⼦类对象各有⼀份
	cout << &p._name << endl;  // 0000007A40FFF738
	cout << &s._name << endl;  // 0000007A40FFF778

	// 这⾥的运⾏结果可以看到静态成员_count的地址是⼀样的
	// 说明⼦类和父类共⽤同⼀份静态成员
	cout << &p._count << endl;  // 00007FF718EA44C4
	cout << &s._count << endl;  // 00007FF718EA44C4

	// 公有的情况下,父⼦类指定类域都可以访问静态成员
	cout << Person::_count << endl;
	cout << Student::_count << endl;
	return 0;
}

七  多继承及其菱形继承问题

7.1 继承模型

  • 单继承:⼀个⼦类只有⼀个直接父类时称这个继承关系为单继承

  • 多继承:⼀个⼦类有两个或以上直接父类时称这个继承关系为多继承,多继承对象在内存中的模型是,先继承的⽗类在前⾯,后⾯继承的⽗类在后⾯,⼦类成员在放到最后⾯。

  • 菱形继承:菱形继承是多继承的⼀种特殊情况。菱形继承的问题,从下⾯的对象成员模型构造,可以看出菱形继承有数据冗余和⼆义性的问题,在Assistant的对象中Person成员会有两份。⽀持多继承就⼀定会有菱形继承,像Java就直接不⽀持多继承,规避掉了这⾥的问题,所以实践中我们也是不建议设计出菱形继承这样的模型的。

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 = "zmcl";

	// 需要显示指定访问哪个父类的成员可以解决⼆义性问题,但是数据冗余问题无法解决
	a.Student::_name = "xxx";
	a.Teacher::_name = "yyy";
	return 0;
}

7.2 虚继承

使用 virtual 关键字可以解决数据冗余和⼆义性。

class Person
{
public:
string _name; // 姓名
};

// 使⽤虚继承Person类
class Student : virtual public Person
{
protected:
	int _num; //学号
};

// 使⽤虚继承Person类
class Teacher : virtual public Person
{
protected:
	int _id; // 职⼯编号
};

// 教授助理
class Assistant : public Student, public Teacher
{
protected:
	string _majorCourse; // 主修课程
};

int main()
{
	// 使⽤虚继承,可以解决数据冗余和⼆义性
	Assistant a;
	a._name = "zmcl";
	return 0;
}

 

哪个类产生数据冗余和⼆义性,那个类加virtual。

7.3 IO库中的菱形虚拟继承

源码:

template<class CharT, class Traits = std::char_traits<CharT>>
class basic_ostream : virtual public std::basic_ios<CharT, Traits>
{};

template<class CharT, class Traits = std::char_traits<CharT>>
class basic_istream : virtual public std::basic_ios<CharT, Traits>
{};

7.4 多继承中指针偏移问题

先继承的在前面~

class Base1 { public: int _b1; };
class Base2 { public: int _b2; };
class Derive : public Base1, public Base2 { public: int _d; };

int main()
{
    Derive d;
    Base1* p1 = &d;
    Base2* p2 = &d;
    Derive* p3 = &d;
    return 0;
}

以上程序 p1 == p3 != p2  

八  继承和组合

  • public继承是⼀种is-a的关系。也就是说每个⼦类对象都是⼀个⽗类对象。
  • 组合是⼀种has-a的关系。假设B组合了A,每个B对象中都有⼀个A对象。
  • 继承允许你根据父类的实现来定义子类的实现。这种通过⽣成⼦类的复⽤通常被称为白箱复用 (white-box reuse)。术语“⽩箱”是相对可视性⽽⾔:在继承⽅式中,⽗类的内部细节对子类可见。继承⼀定程度破坏了⽗类的封装,⽗类的改变,对⼦类有很⼤的影响。子类和父类间的依赖关系很强,耦合度高
  • 对象组合是类继承之外的另⼀种复用选择。新的更复杂的功能可以通过组装或组合对象来获得。对象组合要求被组合的对象具有良好定义的接⼝。这种复⽤⻛格被称为黑箱复用(black-box reuse), 因为对象的内部细节是不可⻅的。对象只以“⿊箱”的形式出现。 组合类之间没有很强的依赖关系,耦合度低。优先使⽤对象组合有助于你保持每个类被封装。
  • 优先使用组合,而不是继承。实际尽量多去用组合,组合的耦合度低,代码维护性好。不过也不太那么绝对,类之间的关系就适合继承(is-a)那就⽤继承,另外要实现多态,也必须要继承。类之间的关系既适合用继承(is-a)也适合组合(has-a),就用组合。
  • 很多⼈说C++语法复杂,其实多继承就是⼀个体现。有了多继承,就存在菱形继承,有了菱形继承就有菱形虚拟继承,底层实现就很复杂,性能也会有⼀些损失,所以最好不要设计出菱形继承。多继承可以认为是C++的缺陷之⼀,后来的⼀些编程语⾔都没有多继承,如Java。

~完~

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

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

相关文章

Spring -- 事务

Spring中事务的操作分为两类:(1)编程式事务 – 手动写代码操作事务(2)声明式事务 – 利用注解开启事务和提交事务 1. 编程式事务 准备Controller RestController RequestMapping("/user") public class UserInfoController {Autowiredprivate UserInfoService use…

用Manim创建条形图【BarChart】

BarChart是Manim库中用于创建条形图的函数。它允许用户通过一组值创建一个条形图&#xff0c;其参数可以调整条形的外观和布局。 BarChart(values, bar_namesNone, y_rangeNone, x_lengthNone, y_lengthNone, bar_colors[#003f5c, #58508d, #bc5090, #ff6361, #ffa600],bar_w…

信息安全服务资质CCRC认证需要哪些条件?

CCRC认证是中国信息安全认证中心开展的信息安全服务资质认证&#xff0c;涵盖7个方向&#xff0c;包括安全集成、安全运维、风险评估、应急处理、软件安全开发、灾难备份与恢复、网络安全审计。每个认证方向的资质级别分为一级、二级、三级&#xff0c;一级最高&#xff0c;三级…

网络基础篇~路由、网络类型和相关命令

一、安装Cisco模拟器 功能&#xff1a;Cisco Packet Tracer&#xff0c;是Cisco官方提供的网络模拟器&#xff0c;专门用于教学和学习网络技术。它提供了广泛的Cisco设备模拟&#xff0c;并具有一个直观的界面&#xff0c;可用于创建、配置和模拟网络拓扑。 &#xff08;一&…

基于vue框架的Android共享停车位i00t4(程序+源码+数据库+调试部署+开发环境)系统界面在最后面。

系统程序文件列表 项目功能&#xff1a;用户,车位主,车位出租,车位预约,取消预约 开题报告内容 基于Vue框架的Android共享停车位系统 开题报告 一、研究背景及意义 1.1 研究背景 随着城市化进程的加速&#xff0c;私家车数量急剧增加&#xff0c;停车难问题日益凸显&…

玛雅房产系统源码开发与技术功能解析

引言 随着房地产市场的蓬勃发展&#xff0c;房产管理系统&#xff08;Real Estate Management System, REMS&#xff09;作为提升行业效率、优化资源配置的关键工具&#xff0c;其重要性日益凸显。房产系统源码开发不仅涉及复杂的业务逻辑处理&#xff0c;还融合了先进的软件开…

无人机之热成像篇

一、定义 无人机热成像技术是指将热成像相机安装在无人机云台上&#xff0c;通过无人机的高空飞行能力和云台的稳定性&#xff0c;结合红外热成像技术对目标区域进行非接触式的温度测量和图像采集。该技术利用物体发出的红外辐射来生成图像&#xff0c;通过测量物体表面温度分布…

08、MySQL-事务

目录 1、事务简介 2、事务操作 2.1 方式一 2.2 方式二 3、事务四大特性 4、并发事务问题 5、事务隔离级别 1、事务简介 事务是一组操作的集合&#xff0c;它是一个不可分割的工作单位&#xff0c;事务会把所有的操作作为一个整体一起向系统提交或撤销操作请求&#xff0c…

第27课 Scratch入门篇:放大的数字

放大的数字 故事背景&#xff1a; 舞台上输入 12345&#xff0c;数字竟然能显示很大&#xff0c;奇妙的数字如何显示的&#xff1f; 程序原理&#xff1a; 重点是如何利用克隆的知识&#xff0c;通过角色造型编号来显示具体的数字。 开始编程 1、在角色绘制中&#xff0c;…

Learn ComputeShader 05 Using noise in the shader

首先实现一个简单的噪声效果 实现原理也很简单&#xff0c;只是在每个线程使用随机函数获得一个随机值 float random (float2 pt) {const float a 12.9898;const float b 78.233;const float c 43758.543123;return frac(sin(dot(pt, float2(a, b))) * c ); }[numthreads(8…

STM32-USART时序与寄存器状态分析

一、时序分析 在UART&#xff08;通用异步收发传输&#xff09;通信中&#xff0c;信号线上的状态分为两种&#xff1a;逻辑1&#xff08;高电平&#xff09;和逻辑0&#xff08;低电平&#xff09;。在空闲状态下&#xff0c;数据线应保持逻辑高电平。UART协议中的各个信号位具…

拥抱变革:旗晟智能巡检机器人系统重塑高风险行业巡检模式

随着工业自动化的快速发展&#xff0c;特别是在石油、化工、煤矿等高风险行业中&#xff0c;传统的巡检方式已难以满足现代企业的需求。高频次、全天候、重复的人工巡检不仅效率低下&#xff0c;还面临着人身安全、数据准确性和运营成本等多方面的挑战。针对这些问题&#xff0…

大模型算力基础设施技术趋势、关键挑战与发展路径

文章目录 前言一、大模型技术发展趋势1.1 大语言模型1.2 多模态模型1.3 长序列模型1.4 混合专家模型二、大模型算力基础设施发展问题与挑战2.1 可用算力规模亟需算力利用效率提升2.2 集群性能提升依赖跨尺度、多层次互联三、大模型算力基础设施高质量发展路径总结前言 从大模型…

SpringCloud与SpringBoot之间的关系解析

Spring Cloud和Spring Boot是两个独立的项目&#xff0c;分别用于构建微服务架构和快速构建Java应用程序。它们之间有着密切的关系&#xff0c;可以相互配合使用。 Spring Boot简介 Spring Boot是一个用于快速构建Java应用程序的框架。它简化了Spring应用程序的开发过程&#x…

Redis,MongoDB,Memcached未授权访问漏洞(及其修复方法)

一. Redis Redis 默认情况下&#xff0c;会绑定在 0.0.0.0:6379 &#xff0c;如果没有进⾏采⽤相关的策略&#xff0c;⽐如添加防 ⽕墙规则避免其他⾮信任来源 ip 访问等&#xff0c;这样将会将 Redis 服务暴露到公⽹上&#xff0c;如果在没有设 置密码认证&#xff08;⼀般为空…

【haproxy】haproxy基本配置信息

一、负载均衡 LB&#xff1a; LoadBalancing&#xff08;负载均衡&#xff09;由多个主机组成&#xff0c;每个主机只承担一部分访问 负载均衡:Load Balance,简称LB&#xff0c;是一种服务或基于硬件设备等实现的高可用反向代理技术&#xff0c;负载均衡将特定的业务(web服务、…

新书速览|Python数据可视化:科技图表绘制(送书)

《Python数据可视化:科技图表绘制》 本书内容 《Python数据可视化:科技图表绘制》结合编者多年的数据分析与科研绘图经验&#xff0c;详细讲解Python语言及包括Matplotlib在内的多种可视化包在数据分析与科研图表制作中的使用方法与技巧。《Python数据可视化:科技图表绘制》分为…

WWDg 正点原子版

窗口看门狗 超时时间计算如下 f1系列&#xff0c;wwdg的时钟源频率是36mhz&#xff0c;时钟频率➗&#xff08;4096*分频系数&#xff09;就是得到实际的频率&#xff0c;也就是一秒钟能计数好多次&#xff0c;那么频率的倒数也就是计数一次需要的时间&#xff0c;所以频率的倒…

【Linux】阻塞信号|信号原理

常见的信号术语 信号递达&#xff08;Delivery&#xff09;: 信号实际被执行处理的过程;&#xff08;当一个信号被递达给进程时&#xff0c;该信号的处理动作已经开始执行实际执行信号的处理动作&#xff09;&#xff1b; 信号未决&#xff08;Pending&#xff09;: 信号从产生…

快速上手Spring Boot

快速上手Spring Boot (qq.com)