2024重生之回溯数据结构与算法系列学习之顺序表习题精讲【无论是王道考研人还真爱粉都能包会的;不然别给我家鸽鸽丢脸好嘛?】

本文涉及的产品
视觉智能开放平台,视频通用资源包5000点
视觉智能开放平台,图像通用资源包5000点
视觉智能开放平台,分割抠图1万点
简介: 顺序表的定义和基本操作之插入;删除;按值查找;按位查找习题精讲等具体详解步骤以及举例说明

欢迎各位彦祖与热巴畅游本人专栏与博客

你的三连是我最大的动力

以下图片仅代表专栏特色 [点击箭头指向的专栏名即可闪现]

专栏跑道一

➡️网络空间安全——全栈前沿技术持续深入学习

image.gif 编辑

专栏跑道二

➡️ 24 Network Security -LJS

image.gif 编辑

image.gif 编辑

image.gif 编辑

专栏跑道三


➡️ MYSQL REDIS Advance operation

image.gif 编辑

专栏跑道四

➡️HCIP;H3C-SE;CCIP——LJS[华为、华三、思科高级网络]

image.gif 编辑

专栏跑道五

➡️RHCE-LJS[Linux高端骚操作实战篇]

image.png

专栏跑道六

➡️数据结构与算法[考研+实际工作应用+C程序设计]

image.gif 编辑

专栏跑道七

➡️RHCSA-LJS[Linux初级及进阶骚技能]

image.gif 编辑

image.gif

上节回顾



课后习题精选:

(13):给定一个含n个整数的数组Q,请设计一个在时间上尽可能高效的算法,找出数组中未出现的最小正整数

image.gif 编辑

解题思路

>定义标记数组
>记录A中出现正整数情况

image.gif

具体代码实现:

#include <iostream>
#include <string.h>
using namespace std;
#define Maxsize 50
// 定义顺序表结构
typedef struct
{
    int data[Maxsize];  // 存储顺序表中的数据
    int length = 0;     // 当前顺序表的长度
} SqList;
// 插入测试数据函数
void ListInsert(SqList &L)
{
    int val = 0;
    // 从标准输入读取整数
    while (cin >> val)
    {
        // 将输入值存储到顺序表中,并更新长度
        L.data[L.length++] = val;
        // 判断是否为输入行的结束符(回车),若是则停止读取
        if (cin.get() == '\n')
        {
            break;
        }
    }
}
// 打印顺序表函数
void PrintList(SqList L)
{
    for (int i = 0; i < L.length; i++)
    {
        // 打印顺序表中的每个元素,以制表符分隔
        cout << L.data[i] << '\t';
    }
    cout << endl; // 打印换行符
}
// 找到1到n范围内的最小缺失正整数
int FindMin(SqList &L, int n)
{
    int i;
    // 动态分配一个大小为n的数组B
    int *B = new int[n];
    // 将数组B初始化为0
    memset(B, 0, sizeof(int) * n);
    // 遍历顺序表中的数据
    for (i = 0; i < n; i++)
    {
        // 若数据在1到n范围内,则在数组B中对应位置标记为1
        if (L.data[i] > 0 && L.data[i] <= n)
        {
            B[L.data[i] - 1] = 1;
        }
    }
    // 查找数组B中第一个未被标记的位置
    for (i = 0; i < n; i++)
    {
        if (B[i] == 0)
        {
            break;
        }
    }
    // 返回缺失的最小正整数
    return i + 1;
}
int main()
{
    SqList L;      // 创建一个顺序表
    ListInsert(L); // 从输入中插入测试数据
    // 查找并打印1到5范围内的最小缺失正整数
    cout << FindMin(L, 5);
}

image.gif

(12):已知一个整数序列A=(a0,a1,an-1),其中若存在则称x为A的主元素。

image.gif 编辑

解题思路:


>算法关键就是扫描数组
>标记处一个可能成为主元的元素,然后重新计数
image.gif

具体代码实现:

#include <iostream>
using namespace std;
#define Maxsize 50
// 定义顺序表结构
typedef struct
{
    int data[Maxsize];  // 存储顺序表中的数据
    int length = 0;     // 当前顺序表的长度
} SqList;
// 插入测试数据
void ListInsert(SqList &L)
{
    int val = 0;
    // 从标准输入读取整数
    while (cin >> val)
    {
        // 将输入值存储到顺序表中,并更新长度
        L.data[L.length++] = val;
        // 判断是否为输入行的结束符(回车),若是则停止读取
        if (cin.get() == '\n')
        {
            break;
        }
    }
}
// 打印顺序表
void PrintList(SqList L)
{
    for (int i = 0; i < L.length; i++)
    {
        // 打印顺序表中的每个元素,以制表符分隔
        cout << L.data[i] << '\t';
    }
    cout << endl; // 打印换行符
}
// 查找出现次数超过一半的元素
int MainValue(SqList &L, int n)
{
    int i, c, count = 1;
    c = L.data[0]; // 初始候选元素为第一个元素
    // 遍历顺序表中的数据,确定可能的多数元素
    for (i = 1; i < n; i++)
    {
        if (L.data[i] == c)
        {
            count++; // 如果当前元素等于候选元素,则计数器加1
        }
        else
        {
            if (count > 0)
            {
                count--; // 如果当前元素与候选元素不同且计数器大于0,则计数器减1
            }
            else
            {
                c = L.data[i]; // 选择新的候选元素
                count = 1;    // 重置计数器为1
            }
        }
    }
    // 再次遍历确认候选元素是否真的出现次数超过一半
    if (count > 0)
    {
        for (i = count = 0; i < n; i++)
        {
            if (L.data[i] == c)
            {
                count++; // 统计候选元素的实际出现次数
            }
        }
    }
    // 如果候选元素的出现次数超过一半,则返回该元素,否则返回-1
    if (count > n / 2)
    {
        return c;
    }
    else
    {
        return -1;
    }
}
int main()
{
    SqList L;      // 创建一个顺序表
    ListInsert(L); // 从输入中插入测试数据
    // 查找并打印出现次数超过一半的元素
    cout << MainValue(L, 5);
}

image.gif

(11):一个长度为L的升序序列S,处在第L/2个位置的数称为S的中位数,例如,若序列则S1的中位数是15,两个序列的中位数是11,现在有两个等长升序A和B

image.gif 编辑

解题思路

>分成3种情况,相等、小于、大于
>等于直接返回
>小于、大于分别减半
image.gif

具体代码实现

#include <iostream>
using namespace std;
#define Maxsize 50
// 定义顺序表结构
typedef struct
{
    int data[Maxsize];  // 存储数据的数组
    int length = 0;     // 顺序表的当前长度
} SqList;
// 插入测试数据
void ListInsert(SqList &L)
{
    int val = 0;
    while (cin >> val) // 从标准输入读取数据
    {
        L.data[L.length++] = val; // 将读取的值插入顺序表,并增加长度
        if (cin.get() == '\n') // 检测到换行符,结束输入
        {
            break;
        }
    }
}
// 打印顺序表
void PrintList(SqList L)
{
    for (int i = 0; i < L.length; i++)
    {
        cout << L.data[i] << '\t'; // 打印顺序表中的每个元素
    }
    cout << endl;
}
// 题目功能函数:查找两个排序数组中的中位数
int SearchMid(SqList &L1, SqList &L2, int n)
{
    int s1 = 0, d1 = n - 1; // L1 的开始和结束下标
    int m1, s2 = 0, d2 = n - 1; // L2 的开始和结束下标
    int m2;
    while (s1 != d1 || s2 != d2) // 当两个数组还没有缩小到只有一个元素时
    {
        m1 = (s1 + d1) / 2; // L1 的中间下标
        m2 = (s2 + d2) / 2; // L2 的中间下标
        if (L1.data[m1] == L2.data[m2]) // 如果中间元素相等,则找到中位数
        {
            return L1.data[m1];
        }
        if (L1.data[m1] < L2.data[m2]) // 如果 L1 中的中间元素小于 L2 中的中间元素
        {
            if ((s1 + d1) % 2 == 0) // 如果 L1 的当前区间大小为偶数
            {
                s1 = m1; // 将 L1 的起始位置更新为 m1
                d2 = m2; // 将 L2 的结束位置更新为 m2
            }
            else // 如果 L1 的当前区间大小为奇数
            {
                s1 = m1 + 1; // 将 L1 的起始位置更新为 m1 + 1
                d2 = m2; // 将 L2 的结束位置更新为 m2
            }
        }
        else // 如果 L1 中的中间元素大于 L2 中的中间元素
        {
            if ((s2 + d2) % 2 == 0) // 如果 L2 的当前区间大小为偶数
            {
                d1 = m1; // 将 L1 的结束位置更新为 m1
                s2 = m2; // 将 L2 的起始位置更新为 m2
            }
            else // 如果 L2 的当前区间大小为奇数
            {
                d1 = m1; // 将 L1 的结束位置更新为 m1
                s2 = m2 + 1; // 将 L2 的起始位置更新为 m2 + 1
            }
        }
    }
    // 返回最终中位数,取两个数组中当前位置的较小值
    return L1.data[s1] < L2.data[s2] ? L1.data[s1] : L2.data[s2];
}
int main()
{
    SqList L1, L2;  // 创建两个顺序表
    ListInsert(L1); // 插入测试数据到 L1
    ListInsert(L2); // 插入测试数据到 L2
    cout << SearchMid(L1, L2, 5); // 输出两个排序数组中位数
}

image.gif

(10) :一个长度为L的升序序列S,处在第L/2个位置的数称为S的中位数,例如,若序列则S1的中位数是15,两个序列的中位数是11,现在有两个等长升序A和B

image.gif 编辑

解题思路:

>将子数组逆转3次
image.gif

具体代码实现:

#include <iostream>
using namespace std;
#define Maxsize 50
// 定义顺序表结构
typedef struct
{
    int data[Maxsize];  // 存储顺序表中的数据
    int length = 0;     // 当前顺序表的长度
} SqList;
// 插入测试数据
void ListInsert(SqList &L)
{
    int val = 0;
    // 从标准输入读取整数
    while (cin >> val)
    {
        // 将输入值存储到顺序表中,并更新长度
        L.data[L.length++] = val;
        // 判断是否为输入行的结束符(回车),若是则停止读取
        if (cin.get() == '\n')
        {
            break;
        }
    }
}
// 打印顺序表
void PrintList(SqList L)
{
    for (int i = 0; i < L.length; i++)
    {
        // 打印顺序表中的每个元素,以制表符分隔
        cout << L.data[i] << '\t';
    }
    cout << endl; // 打印换行符
}
// 逆转数组的部分元素
void Reverse(SqList &L, int left, int right)
{
    // 将指定区间 [left, right] 的元素交换以实现逆转
    for (int i = 0; i < (right - left + 1) / 2; i++)
    {
        int temp = L.data[i + left];
        L.data[i + left] = L.data[right - i];
        L.data[right - i] = temp;
    }
}
// 逆转顺序表的两部分以及整个数组
void ReverseList(SqList &L, int p)
{
    // 逆转前 p 个元素
    Reverse(L, 0, p - 1);        
    // 逆转从 p 到最后的元素
    Reverse(L, p, L.length - 1); 
    // 逆转整个数组
    Reverse(L, 0, L.length - 1); 
}
int main()
{
    SqList L;      // 创建一个顺序表
    ListInsert(L); // 从输入中插入测试数据
    // 逆转顺序表中的前 3 个元素,以及从第 3 个元素开始的其余部分,然后逆转整个数组
    ReverseList(L, 3);
    // 打印逆转后的顺序表
    PrintList(L);
}
image.gif

,,,

相关文章
|
2月前
|
机器学习/深度学习 算法 C++
【DFS/回溯算法】2016年蓝桥杯真题之路径之谜详解
题目要求根据城堡北墙和西墙箭靶上的箭数,推断骑士从西北角到东南角的唯一路径。每步移动时向正北和正西各射一箭,同一格不重复经过。通过DFS回溯模拟“拔箭”过程,验证路径合法性。已知箭数约束路径唯一,最终按编号输出行走顺序。
|
4月前
|
机器学习/深度学习 算法 数据挖掘
没发论文的注意啦!重磅更新!GWO-BP-AdaBoost预测!灰狼优化、人工神经网络与AdaBoost集成学习算法预测研究(Matlab代码实现)
没发论文的注意啦!重磅更新!GWO-BP-AdaBoost预测!灰狼优化、人工神经网络与AdaBoost集成学习算法预测研究(Matlab代码实现)
153 0
|
4月前
|
算法
回溯算法的基本思想
本节介绍回溯算法,通过图1中从A到K的路径查找示例,说明其与穷举法的异同。回溯算法通过“回退”机制高效试探各种路径,适用于决策、优化和枚举问题。
115 0
|
3月前
|
机器学习/深度学习 运维 算法
【微电网多目标优化调度】多目标学习者行为优化算法MOLPB求解微电网多目标优化调度研究(Matlab代码实现)
【微电网多目标优化调度】多目标学习者行为优化算法MOLPB求解微电网多目标优化调度研究(Matlab代码实现)
210 1
|
5月前
|
存储 缓存 算法
什么是回溯算法
回溯算法是一种通过尝试所有可能路径寻找问题解的策略,采用深度优先搜索与状态重置机制。它适用于组合、排列、棋盘等需枚举所有可能解的问题,核心思想包括DFS遍历、剪枝优化与状态恢复。尽管时间复杂度较高,但通过合理剪枝可显著提升效率,是解决复杂搜索问题的重要方法。
236 0
|
9月前
|
算法 数据可视化 开发者
为什么要学习数据结构与算法
今天,我向大家介绍一门非常重要的课程——《数据结构与算法》。这门课不仅是计算机学科的核心,更是每一位开发者从“小白”迈向“高手”的必经之路。
为什么要学习数据结构与算法
|
9月前
|
算法 Java
算法系列之回溯算法求解数独及所有可能解
数独求解的核心算法是回溯算法。回溯算法是一种通过逐步构建解决方案并在遇到冲突时回退的算法。具体来说,我们尝试在空格中填入一个数字,然后递归地继续填充下一个空格。如果在某个步骤中发现无法继续填充,则回退到上一步并尝试其他数字。
354 11
算法系列之回溯算法求解数独及所有可能解
|
10月前
|
算法 Java
算法系列之回溯算法
回溯算法(Backtracking Algorithm)是一种通过穷举来解决问题的方法,它的核心思想是从一个初始状态出发,暴力搜索所有可能的解决方案,遇到正确解将其记录,直到找到了所有的解或者尝试了所有的可能为止。
328 4
算法系列之回溯算法
|
11月前
|
负载均衡 算法
架构学习:7种负载均衡算法策略
四层负载均衡包括数据链路层、网络层和应用层负载均衡。数据链路层通过修改MAC地址转发帧;网络层通过改变IP地址实现数据包转发;应用层有多种策略,如轮循、权重轮循、随机、权重随机、一致性哈希、响应速度和最少连接数均衡,确保请求合理分配到服务器,提升性能与稳定性。
2284 11
架构学习:7种负载均衡算法策略
|
11月前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】顺序表的基本运算(头歌实践教学平台习题)【合集】
本文档介绍了线性表的基本运算任务,涵盖顺序表和链表的初始化、销毁、判定是否为空、求长度、输出、查找元素、插入和删除元素等内容。通过C++代码示例详细展示了每一步骤的具体实现方法,并提供了测试说明和通关代码。 主要内容包括: - **任务描述**:实现顺序表的基本运算。 - **相关知识**:介绍线性表的基本概念及操作,如初始化、销毁、判定是否为空表等。 - **具体操作**:详述顺序表和链表的初始化、求长度、输出、查找、插入和删除元素的方法,并附有代码示例。 - **测试说明**:提供测试输入和预期输出,确保代码正确性。 - **通关代码**:给出完整的C++代码实现,帮助完成任务。 文档
349 5

热门文章

最新文章