【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)

简介: 本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。

前言

       c++是一种支持面向对象编程(OOP)的语言,而在面向对象编程当中,类和对象是核心概念,理解类和对象是学习面向对象编程的基石。面向对象编程是一种编程范式,它使用“类”来定义对象的属性和方法,完成对软件的设计。掌握类和对象的概念及其相互关系,对于深入理解并有效应用面向对象编程至关重要。


一、类的概念及定义

       类的本质是一种自定义类型,是定义对象模板的蓝图或者结构。它制定了对象可以包含的数据以及该对象可以执行的操作。在某种程度上,c++中的类可以认为是c语言结构体的升级版,不仅可以在其中定义成员变量,也可以定义成员函数,用于对成员变量进行访问或操作。


1. 类的定义格式

     接下来我们尝试定义一个类:

class MyClass
{
    void fun1()
    {
        //...
    }
    void fun2()
    {
        //...
    }
    int _x;
    float _y;
};

以上代码当中:


1. class 是定义类的关键字,Myclass 是类名,{} 中的内容是类的主体,fun1和fun2是类的成员函数,_x 和_y 是类的成员变量。

2. 注意定义最后的分号不能省略。


注意事项:


1. 为了防止命名冲突,类的成员变量在创建时一般会在变量名前加上一个特殊的标识符,例如_x,_y。

2. 定义在类中的成员函数默认具有标签inline。


       在c++当中,struct也可以用于定义类,与c语言不同的是,strcut当中可以定义函数,并且类的类型名不需要再带struct。


2. 访问限定符

       c++中有三种访问限定符,用于对类成员的访问权限进行限制,它们分别是:


public(公有):使得被修饰的成员可以在类的外部被访问和修改。

protected(保护):被修饰的成员不可在类外部进行访问,但可以在子类当中访问。

private(私有):对于被修饰的成员,无论是在类外还是在子类中,都无法访问及修改。


一般情况下,我们会对类中的成员变量修饰为private,防止它被外部修改;成员函数修饰为public,便于外部进行方法的调用。


注意:class定义的类当中,如果成员没有被这三种访问限定符修饰,则这些成员默认被private修饰;而struct中的成员默认被public修饰。


接下来,我们简单使用一下这些访问限定符:

#include <iostream>
using namespace std;
 
class MyClass
{
public:
    void fun1()
    {
        //...
    }
 
    void fun2()
    {
        //...
    }
private:
    int _x;
    float _y;
};
 
 
int main()
{
    //创建一个MyClass类对象
    MyClass a;
    a.fun1();//公有的,可以在外部访问
    a._x = 10;//私有成员,不可访问,报错
    return 0;
}

       我们在使用访问限定符时:在其后加上一个冒号,表示从此处开始到下一个访问限定符或者类结束的位置之间的所有成员都被修饰。例如,这里的fun1、fun2函数被修饰为public;_x、_y被修饰为private


       访问限定符不仅是面向对象编程的特性之一——封装的具体实现,也促进了软件设计的质量、规范性、可维护性和安全性。它们是面向对象编程中不可或缺的一部分。


二、类域

       既然学到了类,那就不得不提及类域了。我们都知道,c++一共有四大域:函数局部域、全局域、命名空间域和类域。而我们之前在类中定义的成员函数和成员变量,就属于类域。当我们在类外对类中的成员进行定义时,就需要用域限定运算符“ : : ”。举个例子:

class MyClass
{
public:
    void fun();//方法的声明
 
private:
    int _m;
};
 
void MyClass::fun()//方法的定义,要使用域限定运算符表明该方法所在的类域
{
    //...
}

那么,为什么要使用域限定运算符来表明类域呢?因为类域影响的是编译器的查找规则。如果fun函数没有声明类域,那么编译器就会从全局域去查找该函数的声明。此时如果fun函数有涉及对成员变量_m的操作,编译器从全局域找不到_m,就会发生报错。


三、类的实例化--对象

1. 实例化的概念

        与结构体的定义和创建类似,当我们定义了一个类以后,就可以用这个类在内存中创建出一个对象。所谓对象,指的就是根据类创建出的“变量”。而根据类创建对象的过程,叫做类的实例化,我们在内存中创建出的每一个对象都是类的实例。



我们写一段代码体现类的实例化:

#include <iostream>
using namespace std;
 
//类的定义
class MyClass
{
public:
    void fun() 
    {
        //...
    }
private:
    int _x;
};
 
int main()
{
    MyClass a;//类的实例化,创建一个对象叫a
    return 0;
}

在上述代码中,我们对类进行定义时,编译器并没有为其开辟内存空间,就像是造房子的图纸,类就是一个模板,而对象则是根据这个模板建造出的“房子”,创建对象时才会分配内存空间。


2. 对象的内存大小

       既然创建对象时才分配内存空间,那么对象所占内存空间的大小是多少呢?


首先我们写一段程序,用sizeof来计算对象的内存大小:

#include <iostream>
using namespace std;
 
class X
{
public:
    void fun() 
    {
        //...
    }
private:
    int _x;
};
 
int main()
{
    X x;
    cout << sizeof(x) << endl;
    return 0;
}

运行结果:



可以看到,对象x所占空间是4个字节。从代码当中得知,这个类中包含一个函数fun和一个整形变量_x,而整形的大小是4个字节,所以说对于对象而言,成员变量的内存是包含在其中的,而成员函数不在对象当中存储,而是在代码段当中。其次,c++规定,对象的成员变量才存储时要符合结构体的内存对齐规则。


规则如下:


1.结构体的第一个成员对齐到和结构体的起始地址的偏移量为0的地址处,也就是说第一个成员的偏移量记为0。


2.其他的成员要对齐到该成员的对齐数整数倍的地址处。


(对齐数:编译器默认对齐数与该成员内存大小的较小值;在VS环境中,默认对齐数是8;linux系统中,没有默认对齐数,对齐数就是该成员内存大小)


3.结构体的总大小为结构成员中最大的对齐数的整数倍。


当类中仅有成员函数或者空类的情况:

#include <iostream>
using namespace std;
 
class A
{
    void fun()
    {
        //...
    }
};
 
class B
{
 
};
 
int main()
{
    cout << sizeof(A) << endl;
    cout << sizeof(B) << endl;
    return 0;
}

运行结果:



当类中只有成员函数或者类为空类时,其所创建的对象大小为1字节,纯属占位作用。


四、this指针

       首先来看一段代码:

#include <iostream>
using namespace std;
 
class MyClass
{
public:
    MyClass(int a = 0, float b = 0, char c = 0)//构造函数,用于初始化对象的成员变量,后续会给大家介绍
    {
        _a = a;
        _b = b;
        _c = c;
    }
    void Print()
    {
        cout << _a << endl;
        cout << _b << endl;
        cout << _c << endl;
    }
private:
    int _a;
    float _b;
    char _c;
};
 
int main()
{
    MyClass m = { 1,5.5,'w' };
    m.Print();
}

运行结果:



以上程序中,我们首先用MyClass创建了一个对象m,并且对其进行了初始化。之后,我们打印了一下其中三个成员变量的值。这里不难发现,Print函数是没有参数的。那既然没有参数,那么编译器是怎么知道要打印的是哪个对象的成员变量呢?

       实际上,这里的Print函数的参数的第一个位置,存在一个隐含的this指针



当我们调用对象的成员函数时,本质是将该对象的地址赋值给this指针,隐含的this指针总是指向该对象,不可改变。也就是说,该函数调用当中的this指针指向的是对象m。所以它的本质是通过隐含的this指针,就访问到了对象m的成员。


要注意:

1. 在函数的实参和形参中,这个this指针会自动在参数第一个位置生成,我们不能显示地写出来;但是在函数体内我们可以使用this指针。

2. this指针只能在成员函数内部使用。

3. this指针只是一个形参,并不存储在对象当中。


       this指针的用处:


1. 当我们需要使成员函数返回该对象的地址,就可以return this;

2. 当函数内的局部变量与类的成员变量名发生冲突时,就可以在类成员前加上this->,便于区分。


小练习:


1. 以下代码的运行结果是?

#include <iostream>
using namespace std;
 
class MyClass
{
public:
    void Print()
    {
        cout << "hehe" << endl;
    }
private:
    int _a;
};
 
int main()
{
    MyClass* a = nullptr;
    a->Print();
}

答案:


正常运行,打印“hehe”。原因是:这里创建类指针a,并且调用函数Print。可以看到程序中虽然使用了“->”,但是并没有对空指针a进行解引用,本质是将a传递给了形参this指针。而函数当中并没有访问成员变量,只是打印了“hehe”,所以不会发生问题,程序正常运行。


2. 以下代码的运行结果是:

#include <iostream>
using namespace std;
 
class MyClass
{
public:
    void Print()
    {
        cout << _a << endl;
    }
private:
    int _a;
};
 
int main()
{
    MyClass* a = nullptr;
    a->Print();
}

答案:


运行崩溃。和上一道题相同,本质也是将a传给了形参this,但是函数内部却访问了成员变量_a,我们都知道_a本质是由this指针解引用访问到的,但是此时的形参this是空指针,所以就出现了对空指针解引用的问题,运行崩溃。


总结

       今天,我们初入了c++类和对象的大门,学习了类的概念及定义、类实例化出对象,以及this指针的概念及作用。学习并理解这些知识,对于我们理解面向对象编程的特性之一--封装有很大帮助。如果你觉得博主讲的还不错,就请留下一个小小的赞在走哦,感谢大家的支持❤❤❤

相关文章
|
3月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
134 26
|
4月前
|
C语言 C++
c与c++的内存管理
再比如还有这样的分组: 这种分组是最正确的给出内存四个分区名字:栈区、堆区、全局区(俗话也叫静态变量区)、代码区(也叫代码段)(代码段又分很多种,比如常量区)当然也会看到别的定义如:两者都正确,记那个都选,我选择的是第一个。再比如还有这样的分组: 这种分组是最正确的答案分别是 C C C A A A A A D A B。
65 1
|
4月前
|
机器学习/深度学习 存储 算法
基于 C++ 布隆过滤器算法的局域网上网行为控制:URL 访问过滤的高效实现研究
本文探讨了一种基于布隆过滤器的局域网上网行为控制方法,旨在解决传统黑白名单机制在处理海量URL数据时存储与查询效率低的问题。通过C++实现URL访问过滤功能,实验表明该方法可将内存占用降至传统方案的八分之一,查询速度提升约40%,假阳性率可控。研究为优化企业网络管理提供了新思路,并提出结合机器学习、改进哈希函数及分布式协同等未来优化方向。
95 0
|
7月前
|
存储 Linux C语言
C++/C的内存管理
本文主要讲解C++/C中的程序区域划分与内存管理方式。首先介绍程序区域,包括栈(存储局部变量等,向下增长)、堆(动态内存分配,向上分配)、数据段(存储静态和全局变量)及代码段(存放可执行代码)。接着探讨C++内存管理,new/delete操作符相比C语言的malloc/free更强大,支持对象构造与析构。还深入解析了new/delete的实现原理、定位new表达式以及二者与malloc/free的区别。最后附上一句鸡汤激励大家行动缓解焦虑。
|
7月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
7月前
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
7月前
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
3月前
|
存储
阿里云轻量应用服务器收费标准价格表:200Mbps带宽、CPU内存及存储配置详解
阿里云香港轻量应用服务器,200Mbps带宽,免备案,支持多IP及国际线路,月租25元起,年付享8.5折优惠,适用于网站、应用等多种场景。
864 0
|
3月前
|
存储 缓存 NoSQL
内存管理基础:数据结构的存储方式
数据结构在内存中的存储方式主要包括连续存储、链式存储、索引存储和散列存储。连续存储如数组,数据元素按顺序连续存放,访问速度快但扩展性差;链式存储如链表,通过指针连接分散的节点,便于插入删除但访问效率低;索引存储通过索引表提高查找效率,常用于数据库系统;散列存储如哈希表,通过哈希函数实现快速存取,但需处理冲突。不同场景下应根据访问模式、数据规模和操作频率选择合适的存储结构,甚至结合多种方式以达到最优性能。掌握这些存储机制是构建高效程序和理解高级数据结构的基础。
251 0

热门文章

最新文章