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

简介: 顺序表的定义和基本操作之插入;删除;按值查找;按位查找习题精讲等具体详解步骤以及举例说明

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

你的三连是我最大的动力

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

专栏跑道一

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

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

,,,

相关文章
|
7月前
|
机器学习/深度学习 算法 数据挖掘
没发论文的注意啦!重磅更新!GWO-BP-AdaBoost预测!灰狼优化、人工神经网络与AdaBoost集成学习算法预测研究(Matlab代码实现)
没发论文的注意啦!重磅更新!GWO-BP-AdaBoost预测!灰狼优化、人工神经网络与AdaBoost集成学习算法预测研究(Matlab代码实现)
227 0
|
6月前
|
机器学习/深度学习 运维 算法
【微电网多目标优化调度】多目标学习者行为优化算法MOLPB求解微电网多目标优化调度研究(Matlab代码实现)
【微电网多目标优化调度】多目标学习者行为优化算法MOLPB求解微电网多目标优化调度研究(Matlab代码实现)
304 1
|
12月前
|
算法 数据可视化 开发者
为什么要学习数据结构与算法
今天,我向大家介绍一门非常重要的课程——《数据结构与算法》。这门课不仅是计算机学科的核心,更是每一位开发者从“小白”迈向“高手”的必经之路。
为什么要学习数据结构与算法
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
753 77
|
负载均衡 算法
架构学习:7种负载均衡算法策略
四层负载均衡包括数据链路层、网络层和应用层负载均衡。数据链路层通过修改MAC地址转发帧;网络层通过改变IP地址实现数据包转发;应用层有多种策略,如轮循、权重轮循、随机、权重随机、一致性哈希、响应速度和最少连接数均衡,确保请求合理分配到服务器,提升性能与稳定性。
2755 11
架构学习:7种负载均衡算法策略
|
存储 C++
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
【数据结构——树】哈夫曼树(头歌实践教学平台习题)【合集】目录 任务描述 相关知识 测试说明 我的通关代码: 测试结果:任务描述 本关任务:编写一个程序构建哈夫曼树和生成哈夫曼编码。 相关知识 为了完成本关任务,你需要掌握: 1.如何构建哈夫曼树, 2.如何生成哈夫曼编码。 测试说明 平台会对你编写的代码进行测试: 测试输入: 1192677541518462450242195190181174157138124123 (用户分别输入所列单词的频度) 预
518 14
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
|
存储 人工智能 算法
【C++数据结构——图】最短路径(头歌教学实验平台习题) 【合集】
任务描述 本关任务:编写一个程序,利用Dijkstra算法,实现带权有向图的最短路径。 相关知识 为了完成本关任务,你需要掌握:Dijkst本关任务:编写一个程序,利用Dijkstra算法,实现带权有向图的最短路径。为了完成本关任务,你需要掌握:Dijkstra算法。带权有向图:该图对应的二维数组如下所示:Dijkstra算法:Dijkstra算法是指给定一个带权有向图G与源点v,求从v到G中其他顶点的最短路径。Dijkstra算法的具体步骤如下:(1)初始时,S只包含源点,即S={v},v的距离为0。
216 15
|
Java C++
【C++数据结构——树】二叉树的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现二叉树的基本运算。​ 相关知识 创建二叉树 销毁二叉树 查找结点 求二叉树的高度 输出二叉树 //二叉树节点结构体定义 structTreeNode{ intval; TreeNode*left; TreeNode*right; TreeNode(intx):val(x),left(NULL),right(NULL){} }; 创建二叉树 //创建二叉树函数(简单示例,手动构建) TreeNode*create
444 12
|
C++
【C++数据结构——树】二叉树的性质(头歌实践教学平台习题)【合集】
本文档介绍了如何根据二叉树的括号表示串创建二叉树,并计算其结点个数、叶子结点个数、某结点的层次和二叉树的宽度。主要内容包括: 1. **定义二叉树节点结构体**:定义了包含节点值、左子节点指针和右子节点指针的结构体。 2. **实现构建二叉树的函数**:通过解析括号表示串,递归地构建二叉树的各个节点及其子树。 3. **使用示例**:展示了如何调用 `buildTree` 函数构建二叉树并进行简单验证。 4. **计算二叉树属性**: - 计算二叉树节点个数。 - 计算二叉树叶子节点个数。 - 计算某节点的层次。 - 计算二叉树的宽度。 最后,提供了测试说明及通关代
237 10
|
算法 C++
【C++数据结构——图】最小生成树(头歌实践教学平台习题) 【合集】
【数据结构——图】最小生成树(头歌实践教学平台习题)目录 任务描述 相关知识 测试说明 我的通关代码: 测试结果:【合集】任务描述 本关任务:编写一个程序求图的最小生成树。相关知识 为了完成本关任务,你需要掌握:1.建立邻接矩阵,2.Prim算法。建立邻接矩阵 上述带权无向图对应的二维数组,根据它建立邻接矩阵,如图1建立下列邻接矩阵。注意:INF表示无穷大,表示整数:32767 intA[MAXV][MAXV];Prim算法 普里姆(Prim)算法是一种构造性算法,从候选边中挑
216 10

热门文章

最新文章