【C++篇】C++类与对象深度解析(五):友元机制、内部类与匿名对象的讲解

文章目录

前言 

💬 欢迎讨论:如果你在学习过程中有任何问题或想法,欢迎在评论区留言,我们一起交流学习。你的支持是我继续创作的动力!

👍 点赞、收藏与分享:觉得这篇文章对你有帮助!别忘了点赞、收藏并分享给更多的小伙伴哦!你们的支持是我不断进步的动力!
🚀 分享给更多人:如果你觉得这篇文章对你有帮助,欢迎分享给更多对C++感兴趣的朋友,让我们一起进步!

 4. 友元

在C++中,友元(friend)提供了一种突破类的访问限定符的机制,使得外部函数或其他类可以访问类的私有(private)受保护的成员(protected)。友元可以是友元函数友元类,而这种友元关系是在类定义中通过关键字 friend 显式声明的。

 4.1 友元涉及的基本概念


<一> 友元函数:友元函数可以访问类的私有和受保护成员,但它并不是类的成员函数。
<二>友元类:某个类的所有成员函数都可以是另一个类的友元,允许访问该类的私有和受保护成员。
<三>单向关系:友元关系是单向的,如果A类是B类的友元,B类的成员函数可以访问A类的私有成员,但A类不能访问B类的私有成员,除非B类也显式声明A类为友元。
<四>友元的局限性:虽然友元提供了便利,但它打破了类的封装性,增加了类之间的耦合,因此不宜滥用。

4.2 友元函数

友元函数是一个外部函数,但通过友元声明,它可以访问类的私有和受保护的成员。友元函数不属于类的成员函数,它可以在类的任意地方声明,而不受访问限定符(publicprivateprotected)的限制。

示例:友元函数访问两个类的私有成员 

#include<iostream>
using namespace std;// 前置声明,避免类A的友元函数不识别类B
class B;class A {// 友元声明,允许函数 func 访问A类的私有成员friend void func(const A& aa, const B& bb);private:int _a1 = 1;int _a2 = 2;
};class B {// 友元声明,允许函数 func 访问B类的私有成员friend void func(const A& aa, const B& bb);private:int _b1 = 3;int _b2 = 4;
};// 友元函数定义,能够访问A和B类的私有成员
void func(const A& aa, const B& bb) {cout << "A::_a1: " << aa._a1 << endl;  // 访问A类的私有成员cout << "B::_b1: " << bb._b1 << endl;  // 访问B类的私有成员
}int main() {A aa;B bb;func(aa, bb);  // 调用友元函数,访问A和B类的私有成员return 0;
}

解释

  • 函数 func 被声明为 A 和 B 类的友元,因此它可以访问 A 类和 B 类的私有成员变量 _a1 和 _b1
  • 虽然 func 是一个独立于类的外部函数,但通过友元声明,它获得了访问类的私有数据的权限。

4.3 友元类

友元类允许一个类的所有成员函数访问另一个类的私有和受保护成员。友元类的成员函数并不需要逐一声明为友元,只要类被声明为友元,所有的成员函数都能访问另一个类的私有和受保护成员。

示例:友元类的使用 

#include<iostream>
using namespace std;class A {// 友元类B声明,允许B类的所有成员函数访问A类的私有成员friend class B;private:int _a1 = 1;int _a2 = 2;
};class B {
public:// 可以访问A类的私有成员void func1(const A& aa) {cout << "A::_a1: " << aa._a1 << endl;  // 访问A类的私有成员cout << "B::_b1: " << _b1 << endl;     // 访问B类的私有成员}void func2(const A& aa) {cout << "A::_a2: " << aa._a2 << endl;  // 访问A类的私有成员cout << "B::_b2: " << _b2 << endl;     // 访问B类的私有成员}private:int _b1 = 3;int _b2 = 4;
};int main() {A aa;B bb;bb.func1(aa);  // 通过B类的成员函数访问A类的私有成员bb.func2(aa);  // 通过B类的成员函数访问A类的私有成员return 0;
}

解释

  • B 类被声明为 A 类的友元类,因此 B 类的所有成员函数都可以访问 A 类的私有成员 _a1 和 _a2
  • 通过友元类声明,不需要逐个将 B 类的成员函数声明为 A 类的友元,只要 B 类是 A 类的友元,B 类的所有成员函数都可以访问 A 类的私有数据。

4.4 友元的特性与限制

  • 单向关系:友元关系是单向的,如果 A 是 B 的友元,那么 B 类的成员可以访问 A 类的私有成员,但 A 类不能访问 B 类的私有成员,除非 B 类也将 A 类声明为友元。

示例:单向友元关系

class A;class B {friend class A;  // B 声明 A 为友元
private:int _b1 = 1;
};class A {
public:void accessB(B& bb) {// A 可以访问 B 的私有成员cout << "B::_b1: " << bb._b1 << endl;}
};int main() {A aa;B bb;aa.accessB(bb);  // A 类访问 B 的私有成员return 0;
}
  • 不具有传递性:友元关系不具有传递性。如果 A 是 B 的友元,B 是 C 的友元,A 不能访问 C 类的私有成员。

  • 友元增加耦合性:虽然友元机制提供了访问类私有成员的便利,但过度使用友元会导致类与类之间的耦合增加,破坏了类的封装性。因此,友元不宜滥用,应该谨慎使用。

4.5 友元函数与类的实际应用

友元在某些情况下能提供方便,比如当需要两个类之间进行紧密合作时,使用友元可以简化代码,减少冗长的接口设计。

#include<iostream>
using namespace std;class Account;class Transaction {
public:void deposit(Account& account, double amount);void withdraw(Account& account, double amount);
};class Account {friend class Transaction;  // 声明 Transaction 类为友元类
public:Account(double balance) : _balance(balance) {}void showBalance() const {cout << "Balance: " << _balance << endl;}private:double _balance;
};void Transaction::deposit(Account& account, double amount) {account._balance += amount;  // 直接访问 Account 类的私有成员
}void Transaction::withdraw(Account& account, double amount) {if (amount <= account._balance) {account._balance -= amount;} else {cout << "Insufficient balance" << endl;}
}int main() {Account myAccount(1000.0);Transaction trans;trans.deposit(myAccount, 500.0);  // 存款myAccount.showBalance();          // 输出:1500trans.withdraw(myAccount, 200.0); // 取款myAccount.showBalance();          // 输出:1300return 0;
}

解释:

Transaction 类被声明为 Account 类的友元类,因此 Transaction 类的成员函数 deposit 和 withdraw 可以直接访问 Account 类的私有成员 _balance。

这种情况下,友元机制简化了类与类之间的合作,不必通过公共接口访问私有数据,减少了不必要的代码冗。

总结


友元机制在C++中提供了一种打破类封装的方式,允许外部函数或类访问类的私有和受保护成员。它通过friend关键字来声明友元函数或友元类,使得类之间的合作更加简便。
友元函数和友元类都有其特定的用途,友元函数可以访问多个类的私有成员,而友元类则使得另一个类的所有成员函数都可以访问当前类的私有数据。
友元关系是单向的,不具有传递性,过度使用友元会破坏类的封装性和增加类的耦合性,应该谨慎使用。

5. 内部类

内部类(Nested Class)是指一个类定义在另一个类的内部。在C++中,内部类和外部类是独立的类,尽管它们之间有一定的联系,但内部类不属于外部类的对象,它有自己的内存布局和独立性。使用内部类通常是为了封装简化类之间的关联

5.1 内部类的基本概念


1.独立性:尽管内部类是定义在外部类的内部,但它是一个独立的类。外部类的对象并不包含内部类的对象。也就是说,创建外部类的对象时,并不会自动创建内部类的对象,内部类需要单独实例化。
2.友元关系:内部类默认是外部类的友元类,这意味着内部类可以访问外部类的私有成员。
3.封装:使用内部类可以将一些只在外部类内部使用的逻辑封装起来,使代码更加紧凑和可控。内部类可以定义在 private 或 protected 访问限定符下,限制其他类对其的访问。

5.2 内部类的使用示例

#include<iostream>
using namespace std;class A {
private:static int _k;  // 外部类的静态成员int _h = 1;     // 外部类的非静态成员public:// 定义内部类 Bclass B {public:// 内部类方法可以访问外部类的私有成员,因为 B 是 A 的友元类void foo(const A& a) {cout << "A::_k = " << _k << endl;   // 访问外部类的静态成员cout << "A::_h = " << a._h << endl; // 访问外部类的非静态成员}};
};// 初始化外部类的静态成员
int A::_k = 1;int main() {cout << "Size of A: " << sizeof(A) << endl;  // 输出 A 类的大小A::B b;  // 创建内部类 B 的对象A aa;    // 创建外部类 A 的对象b.foo(aa);  // 使用内部类对象调用其方法,访问外部类的私有成员return 0;
}

解释:

内部类 B 被定义在外部类 A 的 public 区域中,但它依然是 A 的友元类,可以访问 A 类的私有成员变量 _k 和 _h。
创建了 A::B b 来实例化内部类 B,然后通过内部类的成员函数 foo 访问外部类对象的私有成员。
sizeof(A) 表示 A 类的大小,由于 A 只有一个整数成员 _h,因此其大小为4字节。

5.3 封装与访问权限

内部类作为外部类的一部分,可以被放置在 private 或 protected 访问区域中,这样可以控制内部类的可见性。

示例:将内部类放在 private 区域 

#include<iostream>
using namespace std;class Outer {
private:class Inner {  // 内部类定义在 private 区域public:void display() {cout << "Inner class method called." << endl;}};public:void createInner() {Inner in;   // 外部类的方法中可以创建内部类的对象in.display();}
};int main() {Outer outer;outer.createInner();  // 通过外部类的方法调用内部类的方法// Outer::Inner in;  // 错误!内部类在 private 区域,外部无法访问return 0;
}

解释

  • 在这个例子中,内部类 Inner 定义在 Outer 类的 private 区域,外部类的方法 createInner() 可以创建 Inner 类的对象并调用其方法。
  • 尝试在外部直接访问 Inner 类会导致编译错误,因为它是 private 的。

5.4 内部类的封装与应用场景

使用内部类的一个常见场景是当两个类紧密相关时,可以将一个类封装到另一个类中。这样做的目的是让外部类管理内部类的访问,使得内部类只为外部类所用。

场景:内部类作为外部类的专属工具类

#include<iostream>
using namespace std;class Manager {
private:class Task {public:void performTask() {cout << "Performing task." << endl;}};public:void assignTask() {Task t;  // 外部类方法可以使用内部类t.performTask();}
};int main() {Manager mgr;mgr.assignTask();  // 调用外部类的方法,执行内部类中的任务逻辑return 0;
}

解释:

这里,Task 类被封装在 Manager 类的 private 区域,表示 Task 只为 Manager 类服务,外部无法直接访问它。
这是一种封装技术,用于使 Task 类专属于 Manager 类,外部无法创建 Task 对象,只能通过 Manager 类的方法来间接使用它。


5.5 内部类的友元关系

内部类默认是外部类的友元类,这意味着内部类可以访问外部类的私有和受保护成员。这种设计允许内部类和外部类之间进行紧密的合作,使得内部类可以像外部类的成员函数一样访问其内部数据。

示例:内部类访问外部类的私有成员 

#include<iostream>
using namespace std;class Container {
private:int _data = 100;public:// 定义内部类class Helper {public:void showData(const Container& c) {cout << "Container::_data = " << c._data << endl;  // 访问外部类的私有成员}};
};int main() {Container c;Container::Helper h;  // 创建内部类对象h.showData(c);  // 调用内部类的方法,访问外部类的私有成员return 0;
}

解释

  • Helper 类作为 Container 的内部类,默认是 Container 的友元,因此它可以访问 Container 类的私有成员 _data
  • 通过内部类的对象 h,可以调用 showData 方法来访问外部类 Container 的私有数据。

5.6 应用:求 1 + 2 + 3 + … + n

#include<iostream>
using namespace std;class Solution {// 内部类 Sum,用于进行累加操作class Sum {public:Sum() {_ret += _i;  // 每创建一个对象,累加一次当前的 _i++_i;        // 自增 i}};static int _i;    // 用于计数的静态变量static int _ret;  // 用于存储结果的静态变量public:int Sum_Solution(int n) {Sum arr[n];  // 创建 n 个 Sum 对象,触发累加逻辑return _ret; // 返回累加的结果}
};// 初始化静态变量
int Solution::_i = 1;
int Solution::_ret = 0;int main() {Solution sol;cout << "Sum of 1 to 5: " << sol.Sum_Solution(5) << endl;  // 1 + 2 + 3 + 4 + 5 = 15return 0;
}

解释

  • 内部类 Sum 在创建对象时会自动进行累加操作,创建 n 个 Sum 对象等价于对 1 到 n 进行

累加。

  • 静态变量 _i 用于记录当前的计数,_ret 用于存储累加的结果。

总结

内部类是一种封装机制,允许将类定义在另一个类的内部,从而限制内部类的可见性或封装内部逻辑。内部类与外部类独立,但它默认可以访问外部类的私有成员。
内部类的主要优势是封装性和紧密耦合。当一个类主要是为了另一个类服务时,将其设计为内部类可以减少外部依赖和接口冗余。
内部类可以用于实现复杂的逻辑封装、类间的紧密合作、计算封装等多个场景,但应谨慎使用,避免过度增加类的复杂性。

 6. 匿名对象

匿名对象是C++中的一种特殊对象,和普通的有名对象不同,匿名对象没有名字,仅在表达式中被使用,生命周期非常短暂。它的生命周期只限于当前语句,当语句执行结束后,匿名对象就会自动被销毁并调用析构函数。匿名对象的典型用法是临时定义对象,完成某项任务后立即销毁。

6.1 匿名对象的基本概念
匿名对象的定义:匿名对象是通过直接调用构造函数创建的对象,而没有为其指定名字。形式上,它看起来像 A() 或 A(1) 这样的表达式。

生命周期:匿名对象的生命周期非常短暂,只有在当前表达式结束时存在,表达式执行完毕后,匿名对象立即调用析构函数被销毁。

应用场景:匿名对象通常用于临时性操作,例如快速调用某个对象的成员函数或操作符,而不需要将该对象保存在变量中。

匿名对象 vs 有名对象
有名对象:对象名(实参)

例:B pro(10);(有名对象)
生命周期:与作用域相关,当作用域结束时对象销毁。
匿名对象:类型(实参)

例:B(5);(匿名对象)
生命周期:只在当前表达式有效,随后立即销毁。

6.2 匿名对象的创建与销毁

在C++中,通过 B() 或 B(10) 这样的语法直接调用构造函数来创建匿名对象,匿名对象没有名字,生命周期仅限于当前行,结束后立即调用析构函数进行销毁.

示例:

#include<iostream>
using namespace std;class A {
public:// 构造函数A(int a = 0) : _a(a) {cout << "A(int a) 构造函数被调用, _a = " << _a << endl;}// 析构函数~A() {cout << "~A() 析构函数被调用, _a = " << _a << endl;}private:int _a;
};int main() {A aa1;  // 有名对象 aa1 的创建// 不能这样定义对象,因为编译器无法确定是函数声明还是对象定义// A aa1();// 创建匿名对象并立即销毁A();    A(1);   A aa2(2);  // 有名对象 aa2 的创建,生命周期为整个作用域// 匿名对象用于调用函数,完成任务后立即销毁Solution().Sum_Solution(10);return 0;
}

输出:

A(int a) 构造函数被调用, _a = 0
~A() 析构函数被调用, _a = 0
A(int a) 构造函数被调用, _a = 1
~A() 析构函数被调用, _a = 1
A(int a) 构造函数被调用, _a = 2
~A() 析构函数被调用, _a = 2

解释

  • A() 和 A(1) 创建的是匿名对象,它们在当前语句结束后立即调用析构函数。
  • 有名对象 aa1 和 aa2 是在整个作用域内存在的,它们在作用域结束时调用析构函数。
  • 匿名对象 的使用场景之一是调用某个方法或操作符后立即销毁,不占用额外的资源。

6.3 匿名对象的应用场景

6.3.1 匿名对象用于临时调用成员函数

匿名对象的一个常见应用场景是用来临时调用某个类的成员函数,执行完任务后不需要该对象的存在。

class Solution {
public:int Sum_Solution(int n) {return n * (n + 1) / 2;}
};int main() {// 使用匿名对象调用 Sum_Solution 函数int result = Solution().Sum_Solution(10);  // 匿名对象创建后立即销毁cout << "Sum of 1 to 10: " << result << endl;return 0;
}

解释

  • 匿名对象 Solution() 被创建,用于调用 Sum_Solution 函数。函数调用结束后,匿名对象立即销毁,不再占用资源。
  • 这是一种常见的设计模式,适用于不需要保存对象状态的场景。

6.3.2 匿名对象避免对象命名

示例:返回匿名对象

class A {
public:A(int a) : _a(a) {cout << "A(int a) 构造函数被调用, _a = " << _a << endl;}~A() {cout << "~A() 析构函数被调用, _a = " << _a << endl;}private:int _a;
};// 函数返回一个匿名对象
A createA() {return A(100);  // 返回匿名对象
}int main() {createA();  // 调用 createA 函数,返回的匿名对象立即销毁return 0;
}

输出:

A(int a) 构造函数被调用, _a = 100
~A() 析构函数被调用, _a = 100

解释

  • 函数 createA 返回一个匿名对象,返回后立即销毁。
  • 匿名对象在不需要进一步使用的情况下,能够有效减少对象创建和销毁的负担。

6.4 匿名对象的注意事项

  1. 生命周期短暂:匿名对象的生命周期只在当前语句结束时有效,不能跨语句使用匿名对象。如果需要在多行代码中使用对象,必须创建有名对象。

    错误示例:

A obj = A(1);  // 正确,有名对象 obj
A(1).foo();    // 匿名对象调用方法
// A(1);        // 错误:匿名对象无法在下一行使用

2.编译器解析问题:在C++中,有些语法可能导致编译器误判为函数声明而不是对象创建。因此,注意避免如下情况:

错误示例:

A aa1();  // 被误判为函数声明,实际上不是对象的创建

正确用法:

A aa1(1);  // 明确创建对象

 3.匿名对象的返回值优化(RVO):现代C++编译器通常会对匿名对象进行优化,在返回对象时避免多余的拷贝操作。这种优化称为返回值优化(RVO)

总结

  • 匿名对象是没有名字的临时对象,生命周期非常短暂,通常用于一次性操作,如临时调用成员函数或返回值。
  • 匿名对象在表达式结束后立即调用析构函数销毁,适用于不需要持久化对象的场景。
  • 匿名对象避免了额外的命名和管理开销,在简化代码的同时提高了代码的简洁性和可读性。

相信通过这篇文章你对C++类与对象高级部分的有了初步的了解。如果此篇文章对你学习C++有帮助,期待你的三连,你的支持就是我创作的动力!!!

下一篇文章再会.

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

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

相关文章

时序数据库 TDengine 支持集成开源的物联网平台 ThingsBoard

Thingsboard 中“设备配置”和“设备”的关系是一对多的关系&#xff0c;通过设备配置为每个设备设置不同的配置&#xff0c;每个设备都会有一个与其关联的设备配置文件。等等&#xff0c;这不就是TDengine 中超级表的概念&#xff1a; 超级表是一种特殊的表结构&#xff0c;用…

【功能安全】汽车功能安全个人认证证书

目录 1、证书 2、课程信息 &#x1f4d6; 推荐阅读 1、证书 汽车功能安全工程师去拿类似莱茵、SGS、南德颁发的证书&#xff0c;如下&#xff1a; 2、课程信息 一般上什么课程了&#xff0c;课程信息大概如下&#xff1a; 汽车功能安全工程师认证课 &#xff08;3天&#…

数据库作业2

建立并使用mydb8_worker&#xff0c;并在其中建表&#xff1a; &#xff08;1&#xff09;显示所有职工的基本信息 &#xff08;2&#xff09;去重查询 &#xff08;3&#xff09;所有职工人数 &#xff08;4&#xff09;最高最低工资 &#xff08;5&#xff09;平均工资和总工…

2024“龙信杯“电子数据取证竞赛-手机取证题目Writeup

文章关键词&#xff1a;电子数据取证、电子物证、手机取证、服务器取证、介质取证 案情介绍 近期&#xff0c;某公安机关正式受理了一起受害者报案案件。受害者陈述称&#xff0c;其通过微信平台结识了一名自称为相亲中介服务的客服人员。该客服人员诱骗受害者参与所谓的“相亲…

【AscendC算子开发】笔记1 算子开发哲学

重看这门课&#xff0c;有很多内容的认识更深了&#xff0c;做一些记录。 为什么不能将网络节点融合 这个问题关联到另一个问题&#xff1a;为什么我们需要激活函数&#xff1f; 使用线性的神经元堆叠得到的方程最后也是线性方程&#xff0c;无法表征非线性的信息&#xff0c…

一文1800字从0到1浅谈web性能测试!

什么是性能测试&#xff1f; web性能应该注意些什么&#xff1f; 性能测试&#xff0c;简而言之就是模仿用户对一个系统进行大批量的操作&#xff0c;得出系统各项性能指标和性能瓶颈&#xff0c;并从中发现存在的问题&#xff0c;通过多方协助调优的过程。而web端的性能测试…

基于知识图的电影推荐系统

&#x1f3ac; 毕设灵感——“基于知识图谱的电影推荐系统”&#x1f680; &#x1f449; 如果你的毕业设计还没有灵感&#xff0c;那么这个基于知识图谱的电影推荐系统绝对值得参考&#xff01;这不是普通的推荐系统&#xff0c;而是通过知识图谱大数据的结合&#xff0c;来为…

Genmo 的 Mochi1 AI 视频生成技术:内容创作的新纪元

Genmo 的 Mochi1 AI 视频生成技术&#xff1a;内容创作的新纪元 随着 AI 技术的快速发展&#xff0c;AI 视频生成工具已经成为许多创作者的重要工具。Genmo 最新推出的 Mochi1 技术&#xff0c;作为一款开源的 AI 视频生成工具&#xff0c;为内容创作者提供了极具创新性的视频…

详解:CentOS 7 NAT模式的网络配置

打开虚拟机发现并没有网络选项 一、解决虚拟机ping通主机 解决方法1&#xff1a; 启动VMware NAT Service &#xff08;1&#xff09;首先查看主机的ip地址&#xff0c;winR打开搜索框&#xff0c;输入cmd &#xff08;2&#xff09;输入ipconfig查看网络配置信息 ipcon…

Bug:通过反射修改@Autowired注入Bean的字段,明确存在,报错 NoSuchFieldException

【BUG】通过Autowired注入了一个Bean SeqNo&#xff0c;测试的时候需要修改其中的字段。通过传统的反射&#xff0c;无论如何都拿不到信息&#xff0c;关键是一方面可以通过IDEA跳转&#xff0c;一方面debug也确实能看到这个字段。但是每次调用set方法报错&#xff1a;NoSuchFi…

腾讯云轻量服务器Lighthouse的前世今生

目录 序一、名字的由来二、Lighthouse的定位是什么&#xff0c;与CVM的差异化有哪些三、Lighthouse是如何实现简单易用的四、Lighthouse对于开发者有哪些具体的利好 序 印象中&#xff0c;腾讯云轻量应用服务器Lighthouse是在2020年正式上线的。 在其一经推出后&#xff0c;就…

【C++进阶】之C++11的简单介绍(三)

&#x1f4c3;博客主页&#xff1a; 小镇敲码人 &#x1f49a;代码仓库&#xff0c;欢迎访问 &#x1f680; 欢迎关注&#xff1a;&#x1f44d;点赞 &#x1f442;&#x1f3fd;留言 &#x1f60d;收藏 &#x1f30f; 任尔江湖满血骨&#xff0c;我自踏雪寻梅香。 万千浮云遮碧…

Vue项目实战-新能源汽车可视化(一)(持续更新中)

一.项目代码 1.App.vue <template><!-- 模板--><div id"wrapper"><div style"width: 100%"><el-tabs v-model"activeName" id"tabs"> <!-- 标签栏里包含了三个标签面板&#xff0c;分别是研发与维…

web前端-html:简单创建表格表单相结合的网页

效果&#xff1a; <body><form action"这里如果为空表单提交后不会有任何操作"method"get"<label for"edit">用户名</label><input type"text" name"用户名" id"最好不要空&#xff0c;id属性…

【论文学习与撰写】论文里配图的题注、多张图片同列排版格式等

目录 1、插入题注 2、多张图排版 1、插入题注 word--引用--插入题注&#xff0c;就会出来这个 直接点确定的话&#xff0c;是会出来图1/图2/图3.。。。之类的 那是因为标签设置的是 图 如图新建标签为&#xff1a; 图 1. 那么&#xff0c;插入题注之后&#xff0c;就会…

【分立元件】贴片电阻的额定功率

贴片电阻器通过电流后将会发热。而贴片电阻的额定功率(Power Rating)是在额定环境温度中可在连续工作状态下使用的最大功率值。 此外,由于使用温度的上限是确定的,因此在高于额定环境温度的条件下使用时,需要按照以下的功率降额曲线来降低功率。额定环境温度是能够…

ionic Capacitor 生成 Android 应用

官方文档 https://ionic.nodejs.cn/developing/android/ https://capacitorjs.com/docs/getting-started 1、创建新的 Capacitor 应用程序 空目录下面 npm init capacitor/app2、install Capacitor npm install npm start在这里插入图片描述 3、生成dist目录 npm run buil…

SpringBoot篇(缓存层)

目录 前言 缓存是什么&#xff1f; 一、SpringBoot内置缓存解决方案 1. 简介 2. 手机验证码案例 二、SpringBoot整合Ehcache缓存 1. 简介 2. 总结 三、SpringBoot整合Redis缓存 1. 简介 2. 总结 四、SpringBoot整合Memcached缓存 1. 简介 2. 安装 3. 变更缓存为M…

5G 现网信令参数学习(1) - MIB

MIB消息中的参数 systemFrameNumber 000101B, subCarrierSpacingCommon scs30or120, ssb-SubcarrierOffset 6, dmrs-TypeA-Position pos2, pdcch-ConfigSIB1 { controlResourceSetZero 10, searchSpaceZero 4 }, cellBarred notBarred, intraFreqReselection allowed, sp…

1024程序员节特惠题解!

#题外话&#xff1a;/ #先看题目 题目传送门https://www.luogu.com.cn/problem/P1035#思路&#xff1a;直接模拟&#xff08;Shift1&#xff09; #代码&#xff1a; #include <bits/stdc.h> using namespace std; int main(){double cnt0,k,sum0,x;cin>>k;while…