【C++入门到精通】C++入门 —— 类和对象(了解类和对象)(二)

简介: 【C++入门到精通】C++入门 —— 类和对象(了解类和对象)

4.类的作用域


       在C++中,类的作用域(Scope)指的是类声明所定义的标识符的可见性和访问性范围。类的作用域可以分为两个方面:


       1. 类的成员作用域:类的成员作用域指的是在类定义内部声明的成员的可见性和访问范围。成员包括数据成员(变量)和成员函数。在类内部声明的成员默认是私有的(private),只能在类内部访问。可以使用访问修饰符(public、protected、private)来控制成员的访问权限。例如:


class MyClass {
public:
    int publicMember;    // 公共成员,在类外部可访问
protected:
    int protectedMember; // 保护成员,在派生类内和类内部可访问,类外部不可访问
private:
    int privateMember;   // 私有成员,仅在类内部可访问,类外部不可访问
};

       2. 对象的作用域:对象的作用域指的是创建的类对象对标识符的可见性和访问范围。在创建类对象时,对象名在局部作用域中定义,并且只在该作用域内有效。例如:


void myFunction() {
    MyClass obj;
    obj.publicMember = 10;   // 可以访问公共成员
    // obj.privateMember = 20; // 错误,无法访问私有成员
}

       在作用域结束时,对象和对象的成员都会被销毁。需要注意的是,类的静态成员具有特殊的作用域规则。静态成员不与特定的对象关联,而是与类本身相关联。静态成员在类的作用域中可见,可以通过类名和作用域解析运算符(::)来访问。例如:


class MyClass {
public:
    static int staticMember;  // 静态成员
};
int MyClass::staticMember = 0;  // 静态成员定义和初始化
void myFunction() {
    MyClass::staticMember = 10;   // 通过类名访问静态成员
}

       总的来说,C++中类的作用域决定了在不同的上下文中如何访问类的成员。在类内部,可以直接访问所有成员;在对象作用域下,可以通过对象名访问公共成员;在类作用域下,可以通过类名访问静态成员。


5.类的实例化


       用类类型创建对象的过程,称为类的实例化,类是对对象进行描述的,是一个模型一样的东西,限定了类有哪些成员,定义出一个类并没有分配实际的内存空间来存储它。一个类可以实例化出多个对象,实例化出的对象 占用实际的物理空间,存储类成员变量。


⭕访问类的成员


1.调用成员函数:

通过对象名(或指针/引用)和成员访问运算符".“(或”->")来调用类的成员函数。可以使用类定义的公共成员函数对对象进行操作。


2.访问成员变量:

通过对象名(或指针/引用)和成员访问运算符".“(或”->")来访问类的成员变量。可以使用类定义的公共成员变量获取或修改对象的状态。


// 类定义
class MyClass {
public:
    int myInt;  // 成员变量
    void myFunction() {
        cout << "Hello from MyClass!" << endl;
    }  // 成员函数
};
int main() {
    // 声明并创建对象
    MyClass obj;
    // 使用对象调用成员函数和访问成员变量
    obj.myInt = 42;
    obj.myFunction();
    return 0;
}


       在上述示例中,通过"MyClass"类定义了一个类,包括一个整型的成员变量"myInt"和一个成员函数"myFunction"。在main函数中,声明一个名为"obj"的对象,通过对象名"obj"访问成员变量和调用成员函数。


6.类对象模型


🥝类对象的存储方式


  类的成员变量和成员函数的存储方式有所不同。


1. 类成员变量的存储方式:

  ⭕类的成员变量通常存储在类的对象中。

  ⭕对象中的成员变量在类的实例化时被创建,并与对象的实例一起分配内存。

  ⭕每个对象都有自己的成员变量副本,它们存储在对象的内存空间中。


2. 类成员函数的存储方式:

  ⭕类的成员函数通常不会存储在类的对象中,而是存储在类的代码段中。

  ⭕成员函数是类的公共接口的一部分,它们被所有类的实例共享。因此,所有对象在共享的代码段中使用相同的成员函数实现。

  ⭕不同对象间共享相同的成员函数,节省了内存空间。


示例代码:


#include <iostream>
using namespace std;
class MyClass {
public:
    int myInt;  // 成员变量
    void myFunction() {
        cout << "Hello from MyClass!" << endl;
    }  // 成员函数
};
int main() {
    MyClass obj1;  // 创建两个对象
    MyClass obj2;
    obj1.myInt = 42;  // 对象1设置成员变量
    obj2.myInt = 10;  // 对象2设置成员变量
    obj1.myFunction();  // 调用成员函数
    obj2.myFunction();
    return 0;
}


       在上述示例中,每个对象(`obj1`和`obj2`)都包含一个独立的成员变量`myInt`,它们分别存储在各自对象的内存空间中。然而,`myFunction`成员函数只存在一份实现,被所有对象共享。


       需要注意的是,成员函数在调用时会自动传递一个隐式参数,即指向调用对象的指针(`this`指针),使得成员函数能够访问对象的成员变量和其他成员函数。这个`this`指针指向调用该成员函数的对象的内存地址(在本文章后面会提及‘this’指针)


       总的来看:类的成员变量存储在类的对象中,而类的成员函数存储在类的代码段中,被所有对象共享。


🥝计算类对象大小


       在C++中,可以使用`sizeof`运算符来计算类对象的大小,以下是一个示例,展示了如何计算类对象的大小:


#include <iostream>
using namespace std;
class MyClass {
public:
    int myInt;
    double myDouble;
    char myChar;
};
int main() {
    MyClass obj;
    size_t size = sizeof(obj);
    cout << "Size of MyClass object: " << size << " bytes" << endl;
    return 0;
}

       需要注意的是,类对象的大小可能会受到编译器、编译选项和对齐规则的影响,因此不同的环境下可能会得到不同的结果。此外,成员变量的顺序和对齐方式等因素也可能影响类对象的大小。


    结论:一个类的大小,实际就是该类中”成员变量”之和,当然要注意内存对齐

         🚨注意:空类比较特殊,编译器给了空类一个字节来唯一标识这个类的对象。


🥝结构体内存对齐规则


       在C++中,结构体(struct)的内存对齐规则被用来优化内存访问效率,减少存储碎片和内存访问的开销。结构体的内存对齐规则如下:


1.对齐原则:

  1. 结构体的起始地址必须是其最宽基本类型成员大小的倍数。
  2. 结构体的每个成员相对于结构体的起始地址的偏移量必须是其自身大小的倍数。


2.默认对齐方式:


  1. 结构体成员的默认对齐方式通常是按照其自身大小进行对齐。
  2. 基本数据类型的默认对齐方式通常是按照其字节大小进行对齐。


3.强制对齐:

  1. 可以使用特定的对齐属性来显式指定结构体成员的对齐方式,通过在成员声明前加上"attribute((aligned(x)))“(GCC编译器)或"alignas(x)”(标准C++11及以后)来实现。
  2. 强制对齐可以用来解决特定的对齐需求,比如和硬件相关的数据结构。

       需要注意的是,不同的编译器可能会对结构体的对齐规则有所差异,尤其是在不同的编译器选项或编译器版本中。可以使用编译器提供的对齐指令或预处理器的宏来控制结构体的对齐方式。


7. this 指针


       ⭕概念


 this指针是一个隐式的指针,它指向当前对象的地址。它只能在非静态成员函数中使用,用于访问对象的成员变量和成员函数。当你在一个成员函数中使用成员变量或调用成员函数时,编译器会自动将this指针作为一个隐式参数传递给该函数。这样,你就可以使用this指针来访问当前对象的成员。简单说就是:当使用C++编写类的成员函数时,隐含地提供了一个指向当前对象的指针,即this指针。它是指向当前对象的常量指针。


      ⭕this指针的特点


🔴 this指针是一个隐式的指针,无需显式声明。它在每个非静态成员函数内部都是可用的。


🔴 this指针被自动传递给类的非静态成员函数。这意味着在调用成员函数时,编译器会自动在参数列表中添加一个额外的隐式参数,不需要用户传递。


🔴 this指针是一个常量指针,不能修改它所指向的对象。这是因为它指向了当前对象本身,它的值在对象的生命周期内不会改变。


🔴 当使用this指针访问成员变量或调用成员函数时,可以使用箭头运算符( - >)来替代点运算符( . )。


  下面是一个使用this指针的代码:


class MyClass {
public:
    void setX(int x) {
        this->x = x;  // 使用this指针访问成员变量
    }
    int getX() {
        return this->x;  // 使用this指针返回成员变量的值
    }
    void printAddress() {
        cout << "The address of this object is: " << this << endl;  // 使用this指针打印对象的地址
    }
private:
    int x;
};
int main() {
    MyClass obj;
    obj.setX(10);
    cout << "The value of x is: " << obj.getX() << endl;
    obj.printAddress();
    return 0;
}


       在上面的代码中,`setX`函数使用this指针来访问成员变量' x ' ,' getX '函数使用this指针返回成员变量的值,' printAddress ' 函数使用this指针打印对象的地址。在main函数中,我们创建了一个MyClass对象obj ,并调用了这些成员函数。


       🚨需要注意的是,当成员变量和参数或局部变量同名时,this指针可以用于区分它们,以明确指示要操作的是成员变量。


目录
相关文章
|
4月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
95 0
|
4月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
173 0
|
4月前
|
存储 安全 编译器
c++入门
c++作为面向对象的语言与c的简单区别:c语言作为面向过程的语言还是跟c++有很大的区别的,比如说一个简单的五子棋的实现对于c语言面向过程的设计思路是首先分析解决这个问题的步骤:(1)开始游戏(2)黑子先走(3)绘制画面(4)判断输赢(5)轮到白子(6)绘制画面(7)判断输赢(8)返回步骤(2) (9)输出最后结果。但对于c++就不一样了,在下五子棋的例子中,用面向对象的方法来解决的话,首先将整个五子棋游戏分为三个对象:(1)黑白双方,这两方的行为是一样的。(2)棋盘系统,负责绘制画面。
49 0
|
6月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
184 12
|
7月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
7月前
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
8月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
7月前
|
设计模式 安全 C++
【C++进阶】特殊类设计 && 单例模式
通过对特殊类设计和单例模式的深入探讨,我们可以更好地设计和实现复杂的C++程序。特殊类设计提高了代码的安全性和可维护性,而单例模式则确保类的唯一实例性和全局访问性。理解并掌握这些高级设计技巧,对于提升C++编程水平至关重要。
131 16
|
8月前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
7月前
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。