成员变量和成员函数分开存储

  • 在C++中,类内的成员变量和成员函数分开存储
  • 只有非静态成员变量才属于类的对象上
  • 空对象占用一个字节的内存空间,非空对象占用的内存空间大小为所有非静态成员变量占用空间大小之和(内存大小需要对齐) ```cpp

    include

    using namespace std;

// 成员变量 和 成员函数 分开存储

class Person { public: int m_A; // 非静态成员变量 属于类的对象上 static int m_B; // 静态成员变量 不属于类的对象上

  1. void func() {} // 非静态成员函数 不属于类的对象上
  2. void func2() {} // 静态成员函数 不属于类的对象上
  3. float age;

};

void test1() { Person p; // 空对象占用的内存空间为:1 // C++编译器会给每个空对象也分配一个字节空间,是为了区分不同空对象占内存的位置 // 每个空对象也应该有一个独一无二的内存地址 cout << “size of p = “ << sizeof(p) << endl; }

void test2() { Person p; cout << “size of p = “ << sizeof(p) << endl; // 8 cout << “size of Person = “ << sizeof(Person) << endl; // 8 } int main(void) { test2();

  1. return 0;

}

  1. <a name="Hss8U"></a>
  2. ## this指针概念
  3. - 通过上一节的学习我们知道C++成员变量和成员函数是分开存储的
  4. - 每一个非静态成员函数只会诞生一份函数实例,也就是说多个同类型的对象会共用一块代码
  5. - 那么问题是:这一块代码是如何区分哪个对象调用自己的呢?
  6. <a name="VJauF"></a>
  7. ### this指针的定义:
  8. - C++通过提供特殊的对象指针,this指针,解决上述问题。this指针指向被调用的成员函数所属的对象
  9. - this指针是隐含每一个非静态成员函数内的一种指针
  10. - this指针不需要定义,直接使用即可
  11. <a name="VYptg"></a>
  12. ### this指针的用途:
  13. - 当形参和成员变量同名时,可用this指针来区分
  14. - 在类的非静态成员函数中返回对象本身,可使用return *this
  15. ```cpp
  16. #include <iostream>
  17. using namespace std;
  18. class Person {
  19. public:
  20. Person(int age) {
  21. // this指针指向 被调用的成员函数 所属的对象
  22. this->age = age;
  23. }
  24. // 链式编程思想
  25. Person& PersonAddAge(Person &p) {
  26. this->age += p.age;
  27. return *this; // 返回对象引用
  28. }
  29. int age;
  30. };
  31. // 1.解决名称冲突
  32. void test1() {
  33. Person p1(18);
  34. cout << "p1的年龄为:" << p1.age << endl;
  35. }
  36. // 2.返回对象本身用*this
  37. void test2() {
  38. Person p1(10);
  39. Person p2(18);
  40. p2.PersonAddAge(p1).PersonAddAge(p1).PersonAddAge(p1);
  41. cout << "p2的年龄为:" << p2.age << endl;
  42. }
  43. int main(void) {
  44. //test1();
  45. test2();
  46. return 0;
  47. }

空指针访问成员函数

  • C++中空指针也是可以调用成员函数的,但是也要注意有没有用到this指针
  • 如果用到this指针,需要加以判断保证代码的健壮性

    示例:

    ```cpp

    include

    using namespace std;

// 空指针调用成员函数 class Person { public: void showClassName() { cout << “this is a Person class” << endl; }

  1. void showPersonAge() {
  2. if (this == NULL) { // 防止空指针异常
  3. return;
  4. }
  5. // 报错原因是传入的指针是NULL
  6. cout << "age = " << this->age << endl;
  7. }
  8. int age;

};

void test01() { Person* p = NULL; p->showClassName(); // 空指针调用成员函数 p->showPersonAge(); }

int main(void) { test01();

  1. return 0;

}

  1. <a name="qqwrm"></a>
  2. ## const修饰成员函数
  3. <a name="MMj2w"></a>
  4. ### 常函数:
  5. - 成员函数后加const后我们称这个函数为常函数
  6. - 常函数内不可以修改成员属性
  7. - 成员属性声明时加关键字mutable后,在常函数中依然可以修改,在常对象中也可以修改
  8. <a name="IG7sM"></a>
  9. ### 常对象:
  10. - 声明对象前加const称该对象为常对象
  11. - 常对象只能调用常函数
  12. <a name="m6FPb"></a>
  13. ### 示例:
  14. ```cpp
  15. #include <iostream>
  16. using namespace std;
  17. // 常函数
  18. class Person {
  19. public:
  20. // this指针的本质 是指针常量 指针的指向是不可以修改的
  21. // const Person* const this;
  22. // 在成员函数后面加上const,修饰的是this指针,让指针指向的值也不可以修改
  23. void showPerson() const {
  24. this->m_B = 100;
  25. //this->m_A = 100;
  26. //this = NULL; // this指针是不可以修改指针的指向的
  27. }
  28. void showName() {
  29. cout << "hello" << endl;
  30. }
  31. int m_A;
  32. mutable int m_B; // 特殊变量,即使在常函数中,也可以修改这个值
  33. };
  34. void test01() {
  35. Person p;
  36. p.showPerson();
  37. }
  38. // 常对象
  39. void test02() {
  40. const Person p; // 在对象前加const,变为常对象
  41. //p.m_A = 100;
  42. p.m_B = 100; // m_B是特殊值,在常对象下也可以修改
  43. p.showPerson();
  44. //p.showName(); // 常对象只能调用常函数,不能调用普通成员函数,因为普通成员函数可以修改普通属性
  45. }
  46. int main(void) {
  47. return 0;
  48. }