【C++要笑着学】友元 | 初始化列表 | 关键字explicit | 静态成员static | 内部类(二)

简介: 我是柠檬叶子C。上一章我们一步步地实现了日期类,这一章我们继续往后讲解知识点,比如说友元啊,初始化列表啊、静态成员和内部类,把这些拿出来讲一讲。还是保持最近养成的写作习惯,在讲解知识点之前,我都会用一个例子或问题进行引入,做到"循序渐进" 地讲解。

Ⅳ.  静态成员(static)


0x00 引入 - 计算类中创建了多少个类对象

如果我们要计算一个类中创建了多少个类对象,我们可以用 count 计算一下。


int count = 0;  // 全局变量
class A {
public:
  A(int a = 0)
  : _a(a) {
  count++;
  }
  A(const A& aa)
  : _a(aa._a) {
  count++;
  }
private:
  int _a;
};
void f(A a) {
  ;
}
int main(void)
{
  A a1;
  A a2 = 1;
  f(a1);
  cout << count << endl;
  return 0;
}

d355ddcd35c7df83b03c4c9becf66979_7a418261143a476fb11136fd1d964280.png



❓ 如果我不想让这个 count 可以被人在外面随便改呢?


int main(void)
{
  A a1;
  A a2 = 1;
  f(a1);
  cout << cnt << endl;
    count++;   // 我可以在类外修改它
  return 0;
}

有没有办法可以把 count 和类贴合起来呢?让这个 count 专门用来计算我 A 这个类的。


我们先试着把它定义成 —— 成员变量:

class A {
public:
  A(int a = 0)
  : _a(a) {
  _count++;
  }
  A(const A& aa)
  : _a(aa._a) {
  _count++;
  }
private:
  int _a;
  int _count = 0;  // 定义成成员变量
};

但是这样还是不行!这样的话每个对象里面都有一个 count,


我们是希望的是每个对象创建的时候去++的是同一个变量,而不是每个对象里面都有一个。


那该怎么办呢?


类里面可以定义静态成员,在成员变量前面加一个 static,就是静态成员。


我们继续往下看 ~


0x01 静态成员的概念

声明为 static 的类成员称为类的静态成员,用 static 修饰的成员变量,称为静态成员变量。


用 static 修饰的成员函数,称为静态成员函数,静态的成员变量一定要在类外进行初始化。


class A {
public:
  A(int a = 0)
  : _a(a) {
  _sCount++;
  }
  A(const A& aa)
  : _a(aa._a) {
  _sCount++;
  }
private:
  int _a;
  // 静态成员变量属于整个类,所有对象,生命周期在整个程序运行期间。
  static int _sCount;   // 这里以 _s 为前缀,是为了一眼就看出它是静态成员变量。
};


0x02 静态成员的特性

① 静态成员为所有类对象所共享,不属于某个具体的实例。


② 静态成员变量必须在类外定义,定义时不添加 static 关键字。


③ 类静态成员即可用类名 :: 静态成员变量或者对象 . 来访问。


④ 静态成员函数没有隐藏的 this 指针,不能访问任何非静态成员。


⑤ 静态成员和类的普通成员一样,也有 public、protected、private 三种访问级别,也可以具有返回值。


0x03 静态成员函数的访问

💬 如果它是公有的,我们就可以在类外对它进行访问:


class A {
public:
  A(int a = 0)
  : _a(a) {
  _sCount++;
  }
  A(const A& aa)
  : _a(aa._a) {
  _sCount++;
  }
// private:
  int _a;
  static int _sCount;
};
void f(A a) {
  ;
}
int main(void)
{
  A a1;
  A a2 = 1;
  f(a1);
  cout << A::_sCount  << endl;  // 使用类域对它进行访问
  /* 这里不是说是在 a1 里面找,这里只是帮助他突破类域罢了 */
  cout << a1._sCount << endl;
  cout << a2._sCount << endl;
  return 0;
}


但是如果它是私有的,我们可以提供一个公有的成员函数。


我们写一个公有的 GetCount 成员函数,让它返回 _sCount 的值,


这样我们就可以在类外调用该函数,就可以访问到它了。


还有没有更好的方式?让我不用对象就可以访问到它呢?


💬 静态成员函数:


static int GetCount() {
    return _sCount;
}

它的好处是没有 this 指针,它只能访问静态的成员变量。


当然,我们还可以用友元,但是未免有些没必要了。



Ⅴ.  内部类


0x00 内部类的概念

7543297b86f936fe9c5c32354a861254_b2ebe0d261f74599b5dca9415973e77c.png


如果在  类中定义  类,我们称  是  的内部类。


class A 
{
  class B {
  ;
  };
};

0x01 内部类特性

此时这个内部类是一个独立的类,它不属于外部类,


更不能通过外部类的对象去调用内部类,外部类对内部类没有任何特权。


但是,内部类就是外部类的友元类,


内部类可以通过外部类的对象参数来访问外部类中的所有成员,像极了殖民行为。


💬 B 是 A 的内部类:


class A {
private:
  static int _s_a1;
  int _a2;
public:
  class B {   // B天生就是A的友元
  public:
  void foo(const A& a) {
    cout << _s_a1 << endl;   // ✅ 
    cout << a._a2 << endl;   // ✅ 
  }
  private:
  int _b1;
  };
};

0x02 详细探索内部类

❓ 我们用 sizeof 计算 A 类的大小,得到的结果会是什么?


#include <iostream>
using namespace std;
class A {
private:
  static int _s_a1; 
  int _a2;
public:
  class B {
  private:
  int _b1;
  };
};
int A::_s_a1 = 1;
int main(void)
{
  cout << "A的大小为: " << sizeof(A) << endl;
  return 0;
}

58e6e21b726932cbfcc8b2c12760ba37_5184dac5342844c68746786dbf2788fb.png

39482b1c8d3f4e692fcc02e32461e10e_7627878fe5aa425986bac929df01fe5e.png


① 内部类 B 和在全局定义是基本一样的,它只是受外部类 A 类域的限制,定义在 A 的类域中。

c2306ffbbf8a0832789b37348df96ccb_7aef7ffcf66b4b8c996130fb2b428ea1.png

class A {
private:
  static int _s_a1; 
  int _a2;
public:
  class B {
  private:
  int _b1;
  };
};
int A::_s_a1 = 1;
int main(void)
{
  A aa;
  A::B bb;  // 用A的类域指定即可(前提是公有的)
  return 0;
}

65bed94daf9fc907e1e3c928ee74dcdf_f8c527d8ad944b0ea82ec1bb17d8810e.png


② 内部类 B 天生就是外部类 A 的友元,也就是 B 中可以访问 A 的私有,A 不能访问 B 的私有(或保护)。


所以,A 类型的对象里没有 B,跟 B 没什么关系,计算 sizeof 当然也不会带上B。


💬 舔的好!那我就勉为其难地,让你做我的朋友吧:


class A {
private:
  static int _s_a1; 
  int _a2;
public:
  class B {  // B天生就是A的友元
  friend class A;  // 声明A是B的友元
  private:
  int _b1;
  };
};

如此一来,A 和 B 就互通了。

相关文章
|
18天前
|
安全 编译器 C++
C++ `noexcept` 关键字的深入解析
`noexcept` 关键字在 C++ 中用于指示函数不会抛出异常,有助于编译器优化和提高程序的可靠性。它可以减少代码大小、提高执行效率,并增强程序的稳定性和可预测性。`noexcept` 还可以影响函数重载和模板特化的决策。使用时需谨慎,确保函数确实不会抛出异常,否则可能导致程序崩溃。通过合理使用 `noexcept`,开发者可以编写出更高效、更可靠的 C++ 代码。
25 0
|
2月前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
113 5
|
3月前
|
存储 设计模式 编译器
【C++篇】C++类与对象深度解析(五):友元机制、内部类与匿名对象的高级应用
【C++篇】C++类与对象深度解析(五):友元机制、内部类与匿名对象的高级应用
41 2
|
3月前
|
存储 编译器 程序员
【C++】C++特性揭秘:引用与内联函数 | auto关键字与for循环 | 指针空值(二)
【C++】C++特性揭秘:引用与内联函数 | auto关键字与for循环 | 指针空值
|
2月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
61 2
|
2月前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
112 4
|
2月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
152 4
|
3月前
|
存储 编译器 对象存储
【C++打怪之路Lv5】-- 类和对象(下)
【C++打怪之路Lv5】-- 类和对象(下)
35 4
|
3月前
|
编译器 C语言 C++
【C++打怪之路Lv4】-- 类和对象(中)
【C++打怪之路Lv4】-- 类和对象(中)
33 4
|
3月前
|
存储 安全 C++
【C++打怪之路Lv8】-- string类
【C++打怪之路Lv8】-- string类
33 1