【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】

简介: 1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法

 

目录😋

任务描述

相关知识

1. 相关排序和查找算法的原理

2. C++ 类与成员函数的定义

3. 数组作为类的成员变量的处理

4. 函数参数传递与返回值处理

编程要求

测试说明

通关代码

测试结果


任务描述

本关任务:

将直接插入排序、直接选择排序、冒泡排序、顺序查找函数封装到数组类 Array 中,作为成员函数。

相关知识

为了完成本关任务,你需要掌握:

  1. 相关排序和查找算法的原理
  2. C++ 类与成员函数的定义
  3. 数组作为类的成员变量的处理
  4. 函数参数传递与返回值处理

1. 相关排序和查找算法的原理

  • 直接插入排序
  • 原理:把待排序的记录按其关键码值的大小逐个插入到一个已经排好序的有序序列中,直到所有的记录插入完为止,就像人们按大小顺序整理手中的扑克牌一样。具体过程是从第二个元素开始,依次将每个元素与前面已排好序的元素从后往前进行比较,找到合适的位置插入该元素,使得插入后前面的序列依然有序。
  • 示例代码(非成员函数形式,便于理解原理):
void insertionSort(int arr[], int n) {
    for (int i = 1; i < n; i++) {
        int key = arr[i];
        int j = i - 1;
        while (j >= 0 && arr[j] > key) {
            arr[j + 1] = arr[j];
            j--;
        }
        arr[j + 1] = key;
    }
}
  • image.gif
  • 直接选择排序
  • 原理:首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾,以此类推,直到所有元素均排序完毕。
  • 示例代码(非成员函数形式):
void selectionSort(int arr[], int n) {
    for (int i = 0; i < n - 1; i++) {
        int minIndex = i;
        for (int j = i + 1; j < n; j++) {
            if (arr[j] < arr[minIndex]) {
                minIndex = j;
            }
        }
        if (minIndex!= i) {
            int temp = arr[i];
            arr[i] = arr[minIndex];
            arr[minIndex] = temp;
        }
    }
}
  • image.gif
  • 冒泡排序
  • 原理:它重复地走访过要排序的数列,一次比较两个元素,如果它们的顺序错误就把它们交换过来,走访数列的工作是重复地进行直到没有再需要交换的元素为止,即排序完成。就好像水里的气泡,轻的(小的数值)会逐渐往上冒(移到数列前面)。
  • 示例代码(非成员函数形式):
void bubbleSort(int arr[], int n) {
    for (int i = 0; i < n - 1; i++) {
        for (int j = 0; j < n - i - 1; j++) {
            if (arr[j] > arr[j + 1]) {
                int temp = arr[j];
                arr[j] = arr[j + 1];
                arr[j + 1] = temp;
            }
        }
    }
}
  • image.gif
  • 顺序查找
  • 原理:从数组的第一个元素开始,逐个元素与要查找的目标元素进行比较,直到找到目标元素或者遍历完整个数组为止。它是一种简单直观的查找方法,适用于无序数组等各种情况。
  • 示例代码(非成员函数形式):
int sequentialSearch(int arr[], int n, int target) {
    for (int i = 0; i < n; i++) {
        if (arr[i] == target) {
            return i;
        }
    }
    return -1;  // 表示未找到目标元素
}
  • image.gif

2. C++ 类与成员函数的定义

  • 类的基本结构
class Array {
private:
    int* data;  // 可以用来存储数组元素的指针,这里假设存储整数数组
    int size;   // 数组的大小
public:
    Array(int arr[], int n);  // 构造函数声明,用于初始化数组对象
    // 在这里声明要封装的排序和查找成员函数,如
    void insertionSort();
    void selectionSort();
    void bubbleSort();
    int sequentialSearch(int target);
};
  • image.gif
  • 需要了解如何定义一个 Array 类,包括类的声明部分(一般在 .h 文件中定义类的成员变量和成员函数的声明)和类的实现部分(一般在 .cpp 文件中实现成员函数的具体代码逻辑)。例如:
  • 成员函数的定义与调用
  • 要掌握如何在类的实现文件中正确地定义这些成员函数,并且在函数内部能够正确地访问类的私有成员变量(如通过 this 指针来访问当前对象的 datasize 等成员)。例如,插入排序成员函数在类中的实现大致如下:
void Array::insertionSort() {
    for (int i = 1; i < this->size; i++) {
        int key = this->data[i];
        int j = i - 1;
        while (j >= 0 && this->data[j] > key) {
            this->data[j + 1] = this->data[j];
            j--;
        }
        this->data[j + 1] = key;
    }
}
  • image.gif
  • 同时,要清楚如何在类外部创建 Array 类的对象,并调用这些封装好的成员函数来对数组进行相应的排序或查找操作,比如:
int main() {
    int arr[] = {5, 3, 4, 6, 2};
    Array myArray(arr, 5);
    myArray.insertionSort();
    // 可以继续调用其他排序或查找成员函数进行相应操作
    return 0;
}
  • image.gif

3. 数组作为类的成员变量的处理

  • 内存管理方面
  • 当把数组作为类的成员变量时(如上面示例中的 data 指针成员),要考虑数组内存的分配与释放问题。在构造函数中,可能需要根据传入的数组大小动态分配内存来存储数组元素(一般使用 new 关键字),在类的析构函数中,要记得释放之前分配的内存(使用 delete[] 关键字),避免内存泄漏。例如:
Array::Array(int arr[], int n) {
    this->size = n;
    this->data = new int[n];
    for (int i = 0; i < n; i++) {
        this->data[i] = arr[i];
    }
}
Array::~Array() {
    delete[] this->data;
}
  • image.gif
  • 正确访问数组元素
  • 在成员函数中,要通过正确的方式使用类中的数组成员变量来实现排序和查找逻辑,比如使用 this->data[i] 的形式来访问数组中第 i 个元素,确保操作的是当前对象所关联的数组内容。

4. 函数参数传递与返回值处理

  • 参数传递
  • 对于排序成员函数,一般不需要额外的参数传入(因为操作的对象就是类中存储的数组成员变量),但像顺序查找成员函数,就需要传入要查找的目标元素作为参数,要理解值传递、指针传递、引用传递等不同参数传递方式的特点及适用场景,选择合适的方式来传递参数,保证函数功能的正确实现且避免不必要的内存开销等问题。
  • 返回值处理
  • 排序成员函数通常不需要返回值(因为它们直接对类中的数组进行原地排序操作),而顺序查找成员函数需要返回查找目标元素在数组中的索引,如果没找到则返回合适的值(如 -1)来表示查找失败,要正确处理这些返回值情况,以便在调用函数的地方根据返回值进行相应的后续操作。

综上所述,掌握以上这些知识要点,就可以顺利地将相关排序和查找函数封装到 Array 类中作为成员函数了。

编程要求

在右侧编辑器补充代码,实现 3 种排序算法。

测试说明

平台会对你编写的代码进行测试:

测试输入:

无;

预期输出:

1 2 3 6 8

1.9 3.2 4.1 5.6

A c g

2

-1

开始你的任务吧,祝你成功!


通关代码

#include <iostream>
using namespace std;
template <class T>
class Array
{
    T* alist;
    int size;
public:
    Array() {size = 0;}
    Array(int sz) {alist = new T[sz]; size = sz;}
    Array(T a[], int sz) {
        size = sz;
        alist = new T[size];
        for (int i = 0; i < size; i ++)
            alist[i] = a[i];
    }
    ~Array() {size = 0; delete []alist;}
    int getSize() {return size;}
    T& operator [](int i) {return alist[i];}
    void output(){
        for(int i = 0; i < size; i ++)
            cout << alist[i] << " ";
        cout << endl;
    }
    void insertSort();
    void selectSort();
    void bubbleSort();
    int seqSearch(T key);
};
/**********  Begin  **********/
// 在数组中顺序查找第一个key元素,若找到则返回其索引值(是数组中第几个,从0算起),找不到则返回-1
template <class T>
int Array<T>::seqSearch(T key) {
    for(int i = 0;i < size;i++){
        if(alist[i] == key){
            return i;
        }
    }
    return -1;
}
// 插入排序
template <class T>
void Array<T>::insertSort(){
    T temp;
    int j;
    for(int i = 1;i < size;i++){
        temp = alist[i];
        for(j = i;j > 0 && temp < alist[j - 1];j--){
            alist[j] = alist[j - 1];
        }
        alist[j] = temp;
    }
}
// 选择排序
template <class T>
void Array<T>::selectSort(){
    int m;
    T temp;
    for(int i = 0;i < size - 1;i++){
        m=i;
        for(int j = i + 1;j < size;j++){
            if(alist[j] < alist[m]){
                m = j;
            }
        }
        if(m != i){
            temp = alist[i];
            alist[i] = alist[m];
            alist[m] = temp;
        }
    }
}
template <class T>
void Array<T>::bubbleSort(){
    T temp;
    for(int i = 0;i < size - 1;i++){
        for(int j = 0;j < size - 1 - i;j++){
            if(alist[j] > alist[j + 1]){
                temp = alist[j];
                alist[j] = alist[j + 1];
                alist[j + 1] = temp;
            }
        }
    }
}
/**********  End  **********/
int main()
{
    int a[5] = {3, 6, 1, 8, 2};
    Array<int> iArray(a, 5);
    iArray.insertSort();
    iArray.output();
    double d[4] = {4.1, 3.2, 5.6, 1.9};
    Array<double> dArray(d, 4);
    dArray.selectSort();
    dArray.output();
    char c[3] = {'g', 'c', 'A'};
    Array<char> cArray(c, 3);
    cArray.bubbleSort();
    cArray.output();
    int b[4] = {-1, 2, 0, 5};
    Array<int> bArray(b, 4);
    cout<< bArray.seqSearch(0)<< endl;
    cout<< bArray.seqSearch(3)<< endl;
    return 0;
}

image.gif


测试结果

image.gif 编辑

image.gif 编辑

目录
相关文章
|
4月前
|
C++
基本二叉树与排序二叉树(C++源码)
本程序实现二叉树基本操作与二叉排序树应用。支持前序建树、四种遍历、求深度、叶子数、第K层节点数及查找功能;并实现二叉排序树的构建、中序输出与查找比较次数统计,分析不同插入顺序对树形态和查找效率的影响。
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
639 77
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
284 19
|
存储 人工智能 算法
【C++数据结构——图】最短路径(头歌教学实验平台习题) 【合集】
任务描述 本关任务:编写一个程序,利用Dijkstra算法,实现带权有向图的最短路径。 相关知识 为了完成本关任务,你需要掌握:Dijkst本关任务:编写一个程序,利用Dijkstra算法,实现带权有向图的最短路径。为了完成本关任务,你需要掌握:Dijkstra算法。带权有向图:该图对应的二维数组如下所示:Dijkstra算法:Dijkstra算法是指给定一个带权有向图G与源点v,求从v到G中其他顶点的最短路径。Dijkstra算法的具体步骤如下:(1)初始时,S只包含源点,即S={v},v的距离为0。
197 15
|
C++
【C++数据结构——树】二叉树的性质(头歌实践教学平台习题)【合集】
本文档介绍了如何根据二叉树的括号表示串创建二叉树,并计算其结点个数、叶子结点个数、某结点的层次和二叉树的宽度。主要内容包括: 1. **定义二叉树节点结构体**:定义了包含节点值、左子节点指针和右子节点指针的结构体。 2. **实现构建二叉树的函数**:通过解析括号表示串,递归地构建二叉树的各个节点及其子树。 3. **使用示例**:展示了如何调用 `buildTree` 函数构建二叉树并进行简单验证。 4. **计算二叉树属性**: - 计算二叉树节点个数。 - 计算二叉树叶子节点个数。 - 计算某节点的层次。 - 计算二叉树的宽度。 最后,提供了测试说明及通关代
224 10
|
12月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
10月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
411 12
|
8月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
219 0
|
8月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
346 0
|
11月前
|
设计模式 安全 C++
【C++进阶】特殊类设计 && 单例模式
通过对特殊类设计和单例模式的深入探讨,我们可以更好地设计和实现复杂的C++程序。特殊类设计提高了代码的安全性和可维护性,而单例模式则确保类的唯一实例性和全局访问性。理解并掌握这些高级设计技巧,对于提升C++编程水平至关重要。
209 16