【C++初阶】类和对象修炼上

简介: 【C++初阶】类和对象修炼上

1.面向对象和面向过程

C语言是面向过程的,关注的是过程,把一个事情拆分成几个步骤,把步骤写成函数,最后通过调用函数来完成。

C++是面向对象的,关注的是对象,把一个事情拆分成几个对象,抓住对象之间的关系,最后通过对象交互来完成

b9fd313bc9ca37cdb1fd0539c0f3e2b0.jpg

以洗衣服为例:


C语言是面向过程的语言,关注的是过程,把洗衣服这件事拆分成浸泡,漂洗,脱水,晾干等过程,把过程写成函数,最后调用函数来完成;


4e01244b00881bfa8d53d79c5f937b07.png


C++是面向对象的语言,关注的是对象,把洗衣服这件事拆分成人,衣服,洗衣机等对象,抓住对象之间的关系,最后通过对象交互来完成。


e0c3fc146e22741bd953bd6e46292f0c.png


ps: 这里我们刚进新手村,不太理解两者区别没关系


2.class的引入和对象的实例化

如果我们要完整的描述小明这个人,你会怎么描述呐?


小明的属性:姓名,身高,体重,年龄等

小明的行为:吃饭,睡觉,打豆豆等

C++就是采用class关键字来定义我们的类,乍一看是不是和我们学过的struct有一丢丢像呐,这个问题我们后面就会讲.

class classname
{
  //类体:由成员变量和成员函数组成
};

class为定义类的关键字,classname为类名,{}里为类的主体,类体由成员变量和成员函数组成


类体的组成:类中的变量叫做成员变量或类的属性,类中的函数叫做成员函数或类的方法.


类只是一个类型,并不是一个实体,从类得到实体的过程就被称为类的实例化.


如果把类定义为图纸,那么类的实例化就是拿着图纸建造房子,对象就是我们建造出来的房子.

0564577079783436a5dfab428e395d2b.png

3.class对比struct

C++中的class和C语言中的struct对比:

四个角度:

属性:class兼容struct中成员变量的定义

方法:class中增加了成员函数的定义(struct中没有成员函数的定义)

数据:class中成员变量和成员函数在同一个作用域中,成员函数可以直接访问成员变量;

struct中成员变量和函数不在同一个作用域中,函数不能直接访问成员变量,得传参.

默认访问限定符修饰:class为了体现封装性,约束访问成员变量,将默认访问限定符修饰为私有;而struct默认使用者编程素质较高,自由访问成员变量,将默认访问限定符修饰为共有。

struct Stack
{
  //默认public共有:
  int* a;
  int size;
  int capacity;
};
class Stack
{
  //默认private私有:
public:
  void Init(int capacity)
  {
    ;
  }
private:
  int* _a;
  int _size;
  int _capacity;
};

ps:

  1. C++兼容C的语法,所以也是支持struct Stack来定义对象的,同时C++还支持直接使用Stack定义变量
  2. 函数如果被定义在类中,编译器就会默认把这个成员函数定义为内联函数

4.访问限定符

951e2b781a104cf740cd9b59d6cae212.png

ps: 访问限定符限制的是域外面能不能访问,在类里面,只要是共有的,无论是成员变量还是成员函数,都可以访问.

5.声明和定义分离

先来看一个问题:下图的语法错误原因是什么?

//类只是声明
class A
{
public:
  int _a;
};
int main()
{
  A::_a = 1;//红色警告
}

为什么上面的代码中A::_a=1会报错呐?


即使成员变量使用了访问限定符public修饰,主函数中_a使用了域作用限定符A限定,但是因为这时候的 _a只是一种声明,声明的话就意味着此时并没有开辟空间,因此并不能存放数值1;就好比是类只是图纸,不能住人"1",只有类实例化出对象后,在房子里才能住人.


正确代码:

//类只是声明
class A
{
public:
  int _a;
};
int main()
{
  A a;//定义
  a._a = 1;
}

那么对于这个class类,我们如果要实现声明和定义分离,我们该怎么做呐?

844c2e7536cdd6cd600b326cf8f6391e.png

ps:

  1. 声明和定义分离:方便浏览类的结构
  2. 域作用限定符限定:防止命名冲突
  3. 缺省值声明和定义只在声明中写

6.封装

C++的三大特性:封装,继承,多态

但是C++并不只是有这几个特性,毕竟四大名著实际上有很多名著,只是这四本比较好而已.

封装是一种管理,为了更方便管理我们的类.

封装:隐藏属性,公开行为接口

也就是将想给你访问成员函数的定义成私有,不想给你访问成员变量的定义成私有,将成员变量定义成私有之后,在类外你不能随意访问我们的成员变量,这样就不用担心成员变量被修改了,你要想修改成员变量的话,只有通过我提供给你的共有的成员函数来间接访问.

7.类中成员函数的存储位置

首先我们得知道,类就像一张图纸,对象就是按照图纸建造出来得房子.


同一个类实例化出来得对象,比如外卖员他们都有各自的属性信息,但是他们的行为都是一样的


所以对于成员函数的存储位置的布局,C++中采取的是一种共享的策略.


猜测1:类实例化出来的每一个对象都存放各自的成员变量和成员函数


缺点:可以,但是没必要,当对象比较多,同一个类实例化出来的对象比较多,就会造成不必要的空间浪费,猜想不合理


0c065dea642c32785d55da81ac5b5bac.png

猜想2:同一个类实例化出的对象都只存放各自的成员变量,成员函数放在公共代码段中


优点:节约了空间.猜想合理且成立


44f6a54f55e98e12529271d1cb6fd963.png


ps:其实仔细一想我们也能理解,我们在C语言中写函数的目的就是为了防止重复造轮子,打印你这个数组和打印我这个数组,其实都可以只调用一个函数ArrayPrint(),只需要传入各自的数组名和数组元素个数即可打印出各自的数组(只不过在C++中这里的参数变成了隐含的this指针,后面第9点会讲)


8.类/对象的大小


通过上面的知识点,我们已经知道:成员函数并没有存储在每一个实例化出的对象中,所以,对象/类的大小就只是包括成员变量的大小之和,当然要满足"内存对齐规则"


只是包括成员变量的话,其实就是和C语言中struct结构体中的内存对齐规则一样,这里就不多赘述;


值得一提的是空类的大小是多少呐?


class A
{
public:
  void Print()
  {
    ;
  }
};
class B
{
};
int main()
{
  A a;
  cout << &a << endl;
  B b;
  cout << &b << endl;
  return 0;
}


4d728d31a9b4afdde720b67eded0c59b.png


通过打印结果我们看得出,实际上,空类的大小并不是0,这里为了避免空类实例化出来了对象,对象在取地址时出现的都为空,(在内存中没有开辟空间却定义出了变量的尴尬问题),所以C++中编译器给空类和空类实例化出的对象都给与了一个字节空间的大小.


9.隐含的this指针


还记得第7点我们讲的类中成员函数的存储布局吗?那里我们知道了成员函数存储的位置是公共代码段,成员函数是共享使用的:


C语言中对于打印数组1和数组2,只需显式传入各自的数组名和数组大小即可打印出各自的数组.


C++中则是隐式地传入的是调用Print()函数的对象的地址,函数用隐式地用一个this指针来接收.

class Date
{
public:
  void Print()
  {
    cout << _year << endl;
  }
  //隐式的接收:
  //  void Print(const Date* this)
  //{
  //  cout << _year << endl;
  //}
private:
  int _year;
  int _month;
  int _day;
};
int main()
{
  Date d1;
  d1.Print();
  //隐式的传地址:
  //d1.Print(&d1)
  Date d2;
  d2.Print();
  //同理
  return 0;
}

ps:这里的传地址和this接收,都不能显式地写出来,但是允许在成员函数中使用,但是一般能不显式地写出this,例如:

void Print()
  {
    cout << this->_year << endl;
  }

这样的话,哪个对象调用的这个成员函数,就传哪个对象的地址,这样就能实现不同的对象调用相同的成员函数,却实现了对各自的对象的成员变量进行操作.


到了这里我们来看看两个问题:


问题1:


在第五点的问题我们进行变式,(这里的成员函数明明是定义好了的),那么导致下面图片的语法错误的原因是什么?


802ec7a74883d4ac596b77e88d7f6a09.png

实际上,这里的成员函数的确是定义好了的,但是呐,这里就是因为没有哪一个对象来调用Print()函数,所以也就没有办法传隐式的参数,从而出现了这个语法问题.


问题2:下面两个代码,运行结果是什么?


7cff7f5e3360c3878d69251404d4f54a.png


答案是代码1正常运行,代码2运行崩溃


或许有人说:这个p是空,p->这里不是就是错的吗?之前我们在第7点已经给大家讲过,成员函数不是存放在对象中,而是存放在了公共代码段中,不是你的,肯定不在你那里找喽,p->Print()只是传递了p的地址.同时在成员函数内部的this接收到的都是nullptr,所以代码2在打印_a的时候实际是this-> _a,也就是*nullptr-> _a,就出现了对空指针解引用的问题,所以运行崩溃了,而代码1只是打印的一个常量字符串,所以能正常运行.


this指针的特性:


this指针的类型:类类型* const,即成员函数中,不能给this指针赋值。


只能在“成员函数”的内部使用


this指针本质上是“成员函数”的形参,当对象调用成员函数时,将对象地址作为实参传递给


this形参。所以对象中不存储this指针。


this指针是“成员函数”第一个隐含的指针形参,一般情况由编译器通过ecx寄存器自动传

递,不需要用户传递


其实:我们细想一下,我们在学C语言的时候,对于传参,我们一般都是选择传一个变量(比如栈或数组)的地址,在C++中,为了解决这样一个规律性的东西就设计出了this指针.



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