【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指针.



目录
相关文章
|
10天前
|
编译器 C++
C++ 类构造函数初始化列表
构造函数初始化列表以一个冒号开始,接着是以逗号分隔的数据成员列表,每个数据成员后面跟一个放在括号中的初始化式。
56 30
|
24天前
|
C++
C++(十一)对象数组
本文介绍了C++中对象数组的使用方法及其注意事项。通过示例展示了如何定义和初始化对象数组,并解释了栈对象数组与堆对象数组在初始化时的区别。重点强调了构造器设计时应考虑无参构造器的重要性,以及在需要进一步初始化的情况下采用二段式初始化策略的应用场景。
|
24天前
|
存储 编译器 C++
C ++初阶:类和对象(中)
C ++初阶:类和对象(中)
|
24天前
|
C++
C++(十六)类之间转化
在C++中,类之间的转换可以通过转换构造函数和操作符函数实现。转换构造函数是一种单参数构造函数,用于将其他类型转换为本类类型。为了防止不必要的隐式转换,可以使用`explicit`关键字来禁止这种自动转换。此外,还可以通过定义`operator`函数来进行类型转换,该函数无参数且无返回值。下面展示了如何使用这两种方式实现自定义类型的相互转换,并通过示例代码说明了`explicit`关键字的作用。
|
24天前
|
存储 设计模式 编译器
C++(十三) 类的扩展
本文详细介绍了C++中类的各种扩展特性,包括类成员存储、`sizeof`操作符的应用、类成员函数的存储方式及其背后的`this`指针机制。此外,还探讨了`const`修饰符在成员变量和函数中的作用,以及如何通过`static`关键字实现类中的资源共享。文章还介绍了单例模式的设计思路,并讨论了指向类成员(数据成员和函数成员)的指针的使用方法。最后,还讲解了指向静态成员的指针的相关概念和应用示例。通过这些内容,帮助读者更好地理解和掌握C++面向对象编程的核心概念和技术细节。
|
1月前
|
存储 算法 编译器
c++--类(上)
c++--类(上)
|
1月前
|
编译器 C++
virtual类的使用方法问题之C++类中的非静态数据成员是进行内存对齐的如何解决
virtual类的使用方法问题之C++类中的非静态数据成员是进行内存对齐的如何解决
|
1月前
|
编译器 C++
virtual类的使用方法问题之在C++中获取对象的vptr(虚拟表指针)如何解决
virtual类的使用方法问题之在C++中获取对象的vptr(虚拟表指针)如何解决
|
24天前
|
存储 C++
C++(五)String 字符串类
本文档详细介绍了C++中的`string`类,包括定义、初始化、字符串比较及数值与字符串之间的转换方法。`string`类简化了字符串处理,提供了丰富的功能如字符串查找、比较、拼接和替换等。文档通过示例代码展示了如何使用这些功能,并介绍了如何将数值转换为字符串以及反之亦然的方法。此外,还展示了如何使用`string`数组存储和遍历多个字符串。
|
1月前
|
存储 C++
C++ dll 传 string 类 问题
C++ dll 传 string 类 问题
20 0