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



目录
相关文章
|
3天前
|
设计模式 安全 编译器
【C++11】特殊类设计
【C++11】特殊类设计
22 10
|
8天前
|
C++
C++友元函数和友元类的使用
C++中的友元(friend)是一种机制,允许类或函数访问其他类的私有成员,以实现数据共享或特殊功能。友元分为两类:类友元和函数友元。类友元允许一个类访问另一个类的私有数据,而函数友元是非成员函数,可以直接访问类的私有成员。虽然提供了便利,但友元破坏了封装性,应谨慎使用。
39 9
|
3天前
|
存储 编译器 C语言
【C++基础 】类和对象(上)
【C++基础 】类和对象(上)
|
12天前
|
编译器 C++
【C++】string类的使用④(字符串操作String operations )
这篇博客探讨了C++ STL中`std::string`的几个关键操作,如`c_str()`和`data()`,它们分别返回指向字符串的const char*指针,前者保证以&#39;\0&#39;结尾,后者不保证。`get_allocator()`返回内存分配器,通常不直接使用。`copy()`函数用于将字符串部分复制到字符数组,不添加&#39;\0&#39;。`find()`和`rfind()`用于向前和向后搜索子串或字符。`npos`是string类中的一个常量,表示找不到匹配项时的返回值。博客通过实例展示了这些函数的用法。
|
12天前
|
存储 C++
【C++】string类的使用③(非成员函数重载Non-member function overloads)
这篇文章探讨了C++中`std::string`的`replace`和`swap`函数以及非成员函数重载。`replace`提供了多种方式替换字符串中的部分内容,包括使用字符串、子串、字符、字符数组和填充字符。`swap`函数用于交换两个`string`对象的内容,成员函数版本效率更高。非成员函数重载包括`operator+`实现字符串连接,关系运算符(如`==`, `&lt;`等)用于比较字符串,以及`swap`非成员函数。此外,还介绍了`getline`函数,用于按指定分隔符从输入流中读取字符串。文章强调了非成员函数在特定情况下的作用,并给出了多个示例代码。
|
12天前
|
C++
【C++】string类的使用④(常量成员Member constants)
C++ `std::string` 的 `find_first_of`, `find_last_of`, `find_first_not_of`, `find_last_not_of` 函数分别用于从不同方向查找目标字符或子串。它们都返回匹配位置,未找到则返回 `npos`。`substr` 用于提取子字符串,`compare` 则提供更灵活的字符串比较。`npos` 是一个表示最大值的常量,用于标记未找到匹配的情况。示例代码展示了这些函数的实际应用,如替换元音、分割路径、查找非字母字符等。
|
12天前
|
C++
C++】string类的使用③(修改器Modifiers)
这篇博客探讨了C++ STL中`string`类的修改器和非成员函数重载。文章介绍了`operator+=`用于在字符串末尾追加内容,并展示了不同重载形式。`append`函数提供了更多追加选项,包括子串、字符数组、单个字符等。`push_back`和`pop_back`分别用于在末尾添加和移除一个字符。`assign`用于替换字符串内容,而`insert`允许在任意位置插入字符串或字符。最后,`erase`函数用于删除字符串中的部分内容。每个函数都配以代码示例和说明。
|
12天前
|
安全 编译器 C++
【C++】string类的使用②(元素获取Element access)
```markdown 探索C++ `string`方法:`clear()`保持容量不变使字符串变空;`empty()`检查长度是否为0;C++11的`shrink_to_fit()`尝试减少容量。`operator[]`和`at()`安全访问元素,越界时`at()`抛异常。`back()`和`front()`分别访问首尾元素。了解这些,轻松操作字符串!💡 ```
|
12天前
|
存储 编译器 Linux
【C++】string类的使用②(容量接口Capacity )
这篇博客探讨了C++ STL中string的容量接口和元素访问方法。`size()`和`length()`函数等价,返回字符串的长度;`capacity()`提供已分配的字节数,可能大于长度;`max_size()`给出理论最大长度;`reserve()`预分配空间,不改变内容;`resize()`改变字符串长度,可指定填充字符。这些接口用于优化内存管理和适应字符串操作需求。
|
17天前
|
C++
【C++】日期类Date(详解)②
- `-=`通过复用`+=`实现,`Date operator-(int day)`则通过创建副本并调用`-=`。 - 前置`++`和后置`++`同样使用重载,类似地,前置`--`和后置`--`也复用了`+=`和`-=1`。 - 比较运算符重载如`&gt;`, `==`, `&lt;`, `&lt;=`, `!=`,通常只需实现两个,其他可通过复合逻辑得出。 - `Date`减`Date`返回天数,通过迭代较小日期直到与较大日期相等,记录步数和符号。 ``` 这是236个字符的摘要,符合240字符以内的要求,涵盖了日期类中运算符重载的主要实现。