C++之面向对象(上)(三)

简介: C++之面向对象(上)(三)

3.C语言和C++实现Stack的对比

1.C语言实现

#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
typedef int DataType;
typedef struct Stack
{
  DataType* arr;
  int capacity;//栈的容量
  int size;//栈中元素个数
}Stack;
void StackInit(Stack* ps)//创建栈
{
  assert(ps);
  ps->arr = (DataType*)malloc(sizeof(DataType)* 4);
  if (NULL == ps->arr)
  {
    assert(0);
    return;
  }
  ps->capacity = 4;
  ps->size = 0;
}
void StackDestroy(Stack* ps)//销毁栈
{
  assert(ps);
  if (ps->arr)
  {
    free(ps->arr);
    ps->arr = NULL;
    ps->capacity = 0;
    ps->size = 0;
  }
}
void CheckCapacity(Stack* ps)//扩容
{
  if (ps->size == ps->capacity)
  {
    int newcapacity = ps->capacity * 2;
    DataType* temp = (DataType*)realloc(ps->arr,newcapacity*sizeof(DataType));
    if (temp == NULL)
    {
      perror("realloc申请空间失败!!!");
      return;
    }
    ps->arr = temp;
    ps->capacity = newcapacity;
  }
}
void StackPush(Stack* ps, DataType data)//入栈
{
  assert(ps);
  CheckCapacity(ps);
  ps->arr[ps->size] = data;
  ps->size++;
}
int StackEmpty(Stack* ps)//栈是否为空
{
  assert(ps);
  return 0 == ps->size;
}
void StackPop(Stack* ps)//出栈
{
  if (StackEmpty(ps))
    return;
  ps->size--;
}
DataType StackTop(Stack* ps)//栈顶元素
{
  assert(!StackEmpty(ps));
  return ps->arr[ps->size - 1];
}
int StackSize(Stack* ps)//栈的元素个数
{
  assert(ps);
  return ps->size;
}
int main()//测试
{
  Stack s;
  StackInit(&s);
  StackPush(&s, 1);
  StackPush(&s, 2);
  StackPush(&s, 3);
  StackPush(&s, 4);
  printf("%d\n", StackTop(&s));
  printf("%d\n", StackSize(&s));
  StackPop(&s);
  StackPop(&s);
  printf("%d\n", StackTop(&s));
  printf("%d\n", StackSize(&s));
  StackDestroy(&s);
  return 0;
}

运行结果:

观察可得,C语言实现的Stack相关操作函数由以下共性:

  • 每个函数的第一个参数都是Stack*
  • 函数中必须要对第一个参数进行检测,避免该参数为NULL,造成程序崩溃;
  • 函数都是通过Stack*指针参数操作栈的;
  • 函数调用必须传递Stack结构体变量的地址。

结构体中只能定义存放数据的结构,操作数据的方法不能定义在结构体中,即数据和操作数据的方法是分离的。这种实现方式是复杂的,涉及到大量的指针操作,容易出错。

2.C++语言实现

#include<iostream>
using namespace std;
typedef int DataType;
class Stack
{
public:
  void Init()//创建栈
  {
    _arr = (DataType*)malloc(sizeof(DataType)* 4);
    if (NULL == _arr)
    {
      perror("malloc申请空间失败!!!");
      return;
    }
    _capacity = 4;
    _size = 0;
  }
  void Push(DataType data)//入栈
  {
    CheckCapacity();
    _arr[_size] = data;
    _size++;
  }
  void Pop()//出栈
  {
    if (Empty())
      return;
    _size--;
  }
  DataType Top()//栈顶元素
  {
    return _arr[_size - 1]; 
  }
  int Empty() //判断栈是否为空
  {
    return 0 == _size; 
  }
  int Size()//栈中元素个数
  {
    return _size; 
  }
  void Destroy()//销毁栈
  {
    if (_arr)
    {
      free(_arr);
      _arr = NULL;
      _capacity = 0;
      _size = 0;
    }
  }
private:
  void CheckCapacity()//扩容
  {
    if (_size == _capacity)
    {
      int newcapacity = _capacity * 2;
      DataType* temp = (DataType*)realloc(_arr, newcapacity *sizeof(DataType));
      if (temp == NULL)
      {
        perror("realloc申请空间失败!!!");
        return;
      }
      _arr = temp;
      _capacity = newcapacity;
    }
  }
private:
  DataType* _arr;
  int _capacity;
  int _size;
};
int main()
{
  Stack s;
  s.Init();
  s.Push(1);
  s.Push(2);
  s.Push(3);
  s.Push(4);
  printf("%d\n", s.Top());
  printf("%d\n", s.Size());
  s.Pop();
  s.Pop();
  printf("%d\n", s.Top());
  printf("%d\n", s.Size());
  s.Destroy();
  return 0;
}

运行结果:

C++中可以通过类,将数据以及操作数据的方法放在类中,通过访问权限可以控制哪些方法可以在类外进行访问,即封装。

在使用操作数据的方法时,就像使用自己的成员一样,符合人类对某一件事物的认知。而且每个方法不需要传递Stack*的参数,编译器编译之后参数会自动还原,即C++中Stack*参数是由编译器进行维护的,而C语言中需要用户自己维护。

三、相关练习题

  1. 问题:C++中structclass的区别是什么?
    答:C++需要兼容C语言,所以C++中struct可以当成结构体使用。另外C++中struct还可以用来定义类。和class定义类是一样的,区别是struct定义的类默认访问权限是public,class定义的类默认访问权限是private。
  2. 结构体怎么对齐? 为什么要进行内存对齐?
    答:(1)内存对齐是什么?
    内存对齐是指编译器将每个数据单元安排在适当的位置的位置上。
    (2)结构体怎么对齐?
    结构体对齐是依照结构体内存对齐规则(本文有讲解)。
    (3)为什么要进行内存对齐?
    ①平台原因(移植原因)
    某些编译器不能对任意内存位置进行操作,所以要将数据存储在可被操作的位置。
    ②性能原因
    如果没有内存对齐,对数据的访问要进行两次,有内存对齐,对数据的访问只要进行一次。
    ③总结来说,内存对齐就是用空间换时间。
    ④如果想要既节省时间又节省空间,就可以将占内存小的变量集中定义。
  3. 如何让结构体按照指定的对齐参数进行对齐?能否按照3、4、5即任意字节对齐?
    可以通过使用#pragma pack()对默认对齐数进行修改和恢复,实现让结构体按照指定的对齐参数进行对齐,因此,可以按照任意字节对齐。
  4. 什么是大小端?如何测试某台机器是大端还是小端?
    (1)什么是大小端?
    大端是指,数据存储时,高位字节存储在低地址,低位字节存储在高地址;
    小端是指,数据存储时,高位字节存储在高地址,低位字节存储在低地址;

    (2)如何判断某台机器是大端还是小端?
    利用联合体改变一个成员变量,其他成员变量也会被修改的特点,判断大小端:
//判断当前计算机的大小端
int is_little_endian()
{
  union Un
  {
    int a;
    char b;
  };
  union Un i;
  i.a = 1;
  return i.b;
}
int main()
{
  int ret = is_little_endian();
  if (ret == 1)
  {
    printf("小端\n");
  }
  if (ret == 0)
  {
    printf("大端\n");
  }
}

运行结果:


总结

以上就是今天要讲的内容,本文介绍了C++中的类的相关概念。本文作者目前也是正在学习C++相关的知识,如果文章中的内容有错误或者不严谨的部分,欢迎大家在评论区指出,也欢迎大家在评论区提问、交流。

最后,如果本篇文章对你有所启发的话,希望可以多多支持作者,谢谢大家!

目录
打赏
0
0
0
0
0
分享
相关文章
|
2月前
|
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
76 19
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
65 13
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
62 5
|
2月前
|
【C++面向对象——输入输出流】处理二进制文件(头歌实践教学平台习题)【合集】
本任务要求使用C++读取二进制文件并在每行前添加行号后输出到控制台。主要内容包括: 1. **任务描述**:用二进制方式打开指定文件,为每一行添加行号并输出。 2. **相关知识**: - 流类库中常用的类及其成员函数(如`iostream`、`fstream`等)。 - 标准输入输出及格式控制(如`cin`、`cout`和`iomanip`中的格式化函数)。 - 文件的应用方法(文本文件和二进制文件的读写操作)。 3. **编程要求**:编写程序,通过命令行参数传递文件名,使用`getline`读取数据并用`cout`输出带行号的内容。 4. **实验步骤**:参考实验指
59 5
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
48 5
【C++面向对象——类的多态性与虚函数】计算图像面积(头歌实践教学平台习题)【合集】
本任务要求设计一个矩形类、圆形类和图形基类,计算并输出相应图形面积。相关知识点包括纯虚函数和抽象类的使用。 **目录:** - 任务描述 - 相关知识 - 纯虚函数 - 特点 - 使用场景 - 作用 - 注意事项 - 相关概念对比 - 抽象类的使用 - 定义与概念 - 使用场景 - 编程要求 - 测试说明 - 通关代码 - 测试结果 **任务概述:** 1. **图形基类(Shape)**:包含纯虚函数 `void PrintArea()`。 2. **矩形类(Rectangle)**:继承 Shape 类,重写 `Print
53 4
【C++面向对象——类的多态性与虚函数】编写教学游戏:认识动物(头歌实践教学平台习题)【合集】
本项目旨在通过C++编程实现一个教学游戏,帮助小朋友认识动物。程序设计了一个动物园场景,包含Dog、Bird和Frog三种动物。每个动物都有move和shout行为,用于展示其特征。游戏随机挑选10个动物,前5个供学习,后5个用于测试。使用虚函数和多态实现不同动物的行为,确保代码灵活扩展。此外,通过typeid获取对象类型,并利用strstr辅助判断类型。相关头文件如&lt;string&gt;、&lt;cstdlib&gt;等确保程序正常运行。最终,根据小朋友的回答计算得分,提供互动学习体验。 - **任务描述**:编写教学游戏,随机挑选10个动物进行展示与测试。 - **类设计**:基类
42 3
【C++篇】继承之韵:解构编程奥义,领略面向对象的至高法则
【C++篇】继承之韵:解构编程奥义,领略面向对象的至高法则
119 11
【C++核心】一文理解C++面向对象(超级详细!)
这篇文章详细讲解了C++面向对象的核心概念,包括类和对象、封装、继承、多态等。
46 2
【C++】初识面向对象:类与对象详解
【C++】初识面向对象:类与对象详解