C++ 笔记(18)— 类和对象(this 指针、指向类的指针、类静态成员变量和函数)

1. this 指针

C++ 中,每一个对象都能通过 this 指针来访问自己的地址。 this 指针是所有成员函数的隐含参数。因此,在成员函数内部,它可以用来指向调用对象。

友元函数没有 this 指针,因为友元不是类的成员。只有成员函数才有 this 指针。

在类中,关键字 this 包含当前对象的地址,换句话说,其值为 &object 。当您在类成员方法中调用其他成员方法时,编译器将隐式地传递 this 指针—函数调用中不可见的参数:

class Human
{
    private:
        void Talk(string msg)
        {
            cout << msg << endl;
        }
    public:
        void IntroduceSelf()
        {
            Talk("hello world"); // Talk(this, "hello world")
        }
}

方法 IntroduceSelf() 使用私有成员 Talk() 在屏幕上显示一句话。实际上,编译器将在调用 Talk 时嵌入 this 指针,即 Talk(this, "hello world")

从编程的角度看, this 的用途不多,且大多数情况下都是可选的。见下:

void SetAge(int humansAge)
{
    this->age = humansAge; // same as age = humansAge
}

调用静态方法时,不会隐式地传递 this 指针,因为静态函数不与类实例相关联,而由所有实例共享。要在静态函数中使用实例变量,应显式地声明一个形参,并将实参设置为 this 指针。

因为 this 的目的总是指向“这个”对象,所以 this 是一个常量指针,我们不允许改变 this 中保存的地址。

2. 指向类的指针

一个指向 C++ 类的指针与指向结构的指针类似,访问指向类的指针的成员,需要使用成员访问运算符 -> ,就像访问指向结构的指针一样。与所有的指针一样,您必须在使用指针之前,对指针进行初始化。

#include <iostream>
#include <string>
using namespace std;
 
class Box
{
   public:
      // 构造函数声明
      Box(double, double, double);
      double Volume();

   private:
      double length;     // Length of a box
      double breadth;    // Breadth of a box
      double height;     // Height of a box
};

Box::Box(double l=2.0, double b=2.0, double h=2.0)
{ 
    cout <<"Constructor called." << endl;
    length = l;
    breadth = b;
    height = h;
}

double Box::Volume()
{
    return length * breadth * height;
}

int main(void)
{
   Box Box1(3.3, 1.2, 1.5);    // Declare box1
   Box Box2(8.5, 6.0, 2.0);    // Declare box2
   Box *ptrBox;                // Declare pointer to a class.

   // 保存第一个对象的地址
   ptrBox = &Box1;

   // 现在尝试使用成员访问运算符来访问成员
   cout << "Volume of Box1: " << ptrBox->Volume() << endl;

   // 保存第二个对象的地址
   ptrBox = &Box2;

   // 现在尝试使用成员访问运算符来访问成员
   cout << "Volume of Box2: " << ptrBox->Volume() << endl;
  
   return 0;
}

3. 类静态成员

可以使用 static 关键字来把类成员定义为静态的。当我们声明类的成员为静态时,这意味着无论创建多少个类的对象,静态成员都只有一个副本。

静态成员在类的所有对象中是共享的。如果不存在其他的初始化语句,在创建第一个对象时,所有的静态数据都会被初始化为零。

我们不能把静态成员的初始化放置在类的定义中,但是可以在类的外部通过使用范围解析运算符 :: 来重新声明静态变量从而对它进行初始化,如下面的实例所示。

3.1 静态成员变量

静态成员数据也可以分为共有的(Public)和私有的(Private),静态成员数据的声明方法为:

static data_type data_name;

由于静态成员数据不是仅仅属于某一个具体对象的,所以它不能再构造函数中被初始化。否则岂不是每创建一个对象,静态成员数据都要被初始化一次?

如果类的头文件会被直接或间接地重复包含,则静态成员数据也会被重复初始化。为了避免这个问题,我们可以将类的声明和定义分离。

如果类的头文件绝对不会被重复包含,那么把静态成员数据的初始化放在类的头文件中也是勉强可以接受的。

静态成员数据的初始化语句为:

数据类型 类名::静态成员数据=初始值;

静态成员数据是某一个类所具有的属性,而不是某一个对象的属性,所以它的存在并不依赖于对象。
示例代码:

#include <iostream>
#include <string>
using namespace std;
 
class Box
{
   public:
      Box(double, double, double);  // 构造函数声明
      double Volume();
      static int objectCount;

   private:
      double length;     // Length of a box
      double breadth;    // Breadth of a box
      double height;     // Height of a box
};

// 初始化类 Box 的静态成员
int Box::objectCount = 0;

Box::Box(double l=2.0, double b=2.0, double h=2.0)
{ 
    cout <<"Constructor called." << endl;
    length = l;
    breadth = b;
    height = h;
    objectCount++; // 每次创建对象时增加 1
}

double Box::Volume()
{
    return length * breadth * height;
}

int main(void)
{
    Box Box1(3.3, 1.2, 1.5);    // 声明 box1
    Box Box2(8.5, 6.0, 2.0);    // 声明 box2

    // 输出对象的总数
    // Total objects: 2
    cout << "Total objects: " << Box::objectCount << endl;

    return 0;
}

3.2 静态成员函数

既然成员数据可以属于某一个类而不属于某一个具体的对象,那么成员函数也可以这样。在 C++ 中静态成员函数也是属于某一个类而不属于某一个具体的对象的。

静态成员函数的声明方法为:

static 返回值类型 函数名(参数表);

不过在定义静态成员函数时,却不能出现 static

如果把函数成员声明为静态的,就可以把函数与类的任何特定对象独立开来。静态成员函数即使在类对象不存在的情况下也能被调用,静态函数只要使用类名加范围解析运算符 :: 就可以访问。

由于静态成员函数是属于类的,不是属于某一个具体的对象,所以它分不清楚到底是哪个对象的非静态成员数据,故而不能访问非静态成员数据。

所以静态成员函数只能访问静态成员变量数据、其他静态成员函数和类外部的其他函数。

静态成员函数有一个类范围,他们不能访问类的 this 指针。您可以使用静态成员函数来判断类的某些对象是否已被创建。

静态成员函数与普通成员函数的区别:

  • 静态成员函数没有 this 指针,只能访问静态成员(包括静态成员变量和静态成员函数)。
  • 普通成员函数有 this 指针,可以访问类中的任意成员;而静态成员函数没有 this 指针。
#include <iostream>
#include <string>
using namespace std;
 
class Box
{
   public:
      Box(double, double, double);  // 构造函数声明
      double Volume();
      static int objectCount;
      static int getCount();

   private:
      double length;     // Length of a box
      double breadth;    // Breadth of a box
      double height;     // Height of a box
};

// 初始化类 Box 的静态成员
int Box::objectCount = 0;

// 在定义静态成员函数时,却不能出现 `static`
int Box::getCount()
{
    return objectCount;
}
Box::Box(double l=2.0, double b=2.0, double h=2.0)
{ 
    cout <<"Constructor called." << endl;
    length = l;
    breadth = b;
    height = h;
    objectCount++; // 每次创建对象时增加 1
}

double Box::Volume()
{
    return length * breadth * height;
}

int main(void)
{
   // 在创建对象之前输出对象的总数
   cout << "Inital Stage Count: " << Box::getCount() << endl;
 
   Box Box1(3.3, 1.2, 1.5);    // 声明 box1
   Box Box2(8.5, 6.0, 2.0);    // 声明 box2
 
   // 在创建对象之后输出对象的总数
   cout << "Final Stage Count: " << Box2.getCount() << endl;
 
   return 0;
}

其中 Box2.getCount() 等价于 Box::getCount()。即 类型::静态成员函数(参数列表) 等价于 对象名.静态成员函数(参数列表),但更推荐前者,比较符合静态成员函数定义。

输出结果:

Inital Stage Count: 0
Constructor called.
Constructor called.
Final Stage Count: 2

类中特殊成员变量的初始化问题:

  • 常量变量:必须通过构造函数参数列表进行初始化。
  • 引用变量:必须通过构造函数参数列表进行初始化。
  • 普通静态变量:要在类外通过"::"初始化。
  • 静态整型常量:可以直接在定义的时候初始化。
  • 静态非整型常量:不能直接在定义的时候初始化。要在类外通过"::"初始化。
  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值