【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 就互通了。

相关文章
|
20天前
|
C++
C++友元函数和友元类的使用
C++中的友元(friend)是一种机制,允许类或函数访问其他类的私有成员,以实现数据共享或特殊功能。友元分为两类:类友元和函数友元。类友元允许一个类访问另一个类的私有数据,而函数友元是非成员函数,可以直接访问类的私有成员。虽然提供了便利,但友元破坏了封装性,应谨慎使用。
46 9
|
18天前
|
编译器 数据安全/隐私保护 C++
C++一分钟之-属性友元与访问控制
【7月更文挑战第9天】C++中的友元机制允许非成员函数或类访问私有和保护成员,打破了封装性。友元需在类内声明,常见的错误包括忘记声明、过度使用及误解友元的非继承性。要避免错误,应明确声明友元,限制其使用,并理解其局限。示例展示了如何声明和使用友元函数来访问私有数据。谨慎使用友元以保持代码的健壮性和可维护性。
13 1
|
23天前
|
编译器 C++
【C++】类和对象⑤(static成员 | 友元 | 内部类 | 匿名对象)
📚 C++ 知识点概览:探索类的`static`成员、友元及应用🔍。
|
25天前
|
编译器 C++
【C++】详解初始化列表,隐式类型转化,类静态成员,友元
【C++】详解初始化列表,隐式类型转化,类静态成员,友元
|
28天前
|
存储 编译器 C++
【C++】类和对象④(再谈构造函数:初始化列表,隐式类型转换,缺省值
C++中的隐式类型转换在变量赋值和函数调用中常见,如`double`转`int`。取引用时,须用`const`以防修改临时变量,如`const int& b = a;`。类可以有隐式单参构造,使`A aa2 = 1;`合法,但`explicit`关键字可阻止这种转换。C++11起,成员变量可设默认值,如`int _b1 = 1;`。博客探讨构造函数、初始化列表及编译器优化,关注更多C++特性。
|
3天前
|
C++
什么是析构函数,它在C++类中起什么作用
什么是析构函数,它在C++类中起什么作用?
20 11
|
3天前
|
C++
能不能说一个C++类的简单示例呀?能解释一下组成部分更好了
能不能说一个C++类的简单示例呀?能解释一下组成部分更好了
25 10
|
15天前
|
设计模式 安全 编译器
【C++11】特殊类设计
【C++11】特殊类设计
36 10
|
15天前
|
存储 编译器 C语言
【C++基础 】类和对象(上)
【C++基础 】类和对象(上)