重生之“我打数据结构,真的假的?”--1.顺序表(无习题)

news2024/10/22 22:45:45

在这里插入图片描述

C语言中的顺序表详细总结

1. 概述

顺序表(Sequential List)是一种线性数据结构,用于存储具有相同数据类型的一组元素。顺序表采用一段连续的存储空间,使用数组来实现,能够高效地支持随机访问操作。在 C 语言中,顺序表的实现通常基于数组,并且用户需要手动管理内存。顺序表适合用来解决需要快速访问元素的场景,尤其是当元素的数量较为稳定、不需要频繁插入或删除时。本文将详细讨论顺序表的定义、实现、各种操作的具体实现代码,以及顺序表的优缺点和实际应用场景。

2. 顺序表的基本概念

2.1 顺序表的定义

顺序表是一种存储线性表的顺序存储结构,其存储单元采用一段连续的内存区域,可以直接通过索引来访问任意元素。这使得顺序表在进行随机访问时效率非常高,时间复杂度为 O(1)。然而,由于内存是连续的,所以在插入或删除元素时,可能需要移动大量的数据,因此插入和删除操作的时间复杂度较高。

2.2 顺序表的特点

  • 连续存储:顺序表的元素存储在连续的内存空间中。
  • 随机访问:可以通过下标直接访问元素,时间复杂度为 O(1)。
  • 内存分配:顺序表的内存大小通常在初始化时分配,若需要动态扩展,则需要重新分配内存。

3. 顺序表的基本操作实现

顺序表的基本操作包括初始化、插入、删除、查找和遍历。以下我们通过 C 语言代码实现这些操作,以帮助理解顺序表的工作原理。

3.1 顺序表的数据结构定义

首先,定义顺序表的结构体。该结构体包含一个指针指向存储数据的数组,以及顺序表的当前长度和最大容量。

#include <stdio.h>
#include <stdlib.h>

#define INITIAL_CAPACITY 10

// 顺序表结构体定义
typedef struct {
    int *data;       // 存储数据的数组
    int length;      // 当前顺序表的长度
    int capacity;    // 顺序表的容量
} SequentialList;

在上述代码中,我们定义了一个名为 SequentialList 的结构体,其中 data 是一个指向 int 类型数组的指针,length 表示当前顺序表中的元素个数,capacity 表示顺序表的最大容量。

3.2 初始化顺序表

接下来,实现初始化顺序表的函数。该函数分配一段内存作为顺序表的存储空间,并初始化其长度和容量。

// 初始化顺序表
SequentialList* initList(int capacity) {
    SequentialList *list = (SequentialList*)malloc(sizeof(SequentialList));
    if (list == NULL) {
        printf("内存分配失败\n");
        exit(1);
    }
    list->data = (int*)malloc(sizeof(int) * capacity);
    if (list->data == NULL) {
        printf("内存分配失败\n");
        free(list);
        exit(1);
    }
    list->length = 0;
    list->capacity = capacity;
    return list;
}

该函数首先为 SequentialList 结构体本身分配内存,然后为数据数组分配内存,并设置初始长度为 0,容量为传入的参数值。

3.3 插入元素

顺序表支持在指定位置插入元素。如果插入的位置无效或者顺序表已满,则需要进行相应处理。

// 插入元素
int insertElement(SequentialList *list, int index, int value) {
    if (index < 0 || index > list->length) {
        printf("插入位置无效\n");
        return 0;
    }
    // 如果顺序表已满,扩展容量
    if (list->length == list->capacity) {
        int newCapacity = list->capacity * 2;
        int *newData = (int*)realloc(list->data, sizeof(int) * newCapacity);
        if (newData == NULL) {
            printf("内存扩展失败\n");
            return 0;
        }
        list->data = newData;
        list->capacity = newCapacity;
    }
    // 从插入位置开始,向后移动元素
    for (int i = list->length; i > index; i--) {
        list->data[i] = list->data[i - 1];
    }
    // 插入新元素
    list->data[index] = value;
    list->length++;
    return 1;
}

该函数首先检查插入位置是否合法,然后判断顺序表是否已满,若已满则通过 realloc 扩展顺序表的容量。接着,将插入位置之后的元素依次后移,最后将新元素插入到指定位置。

3.4 删除元素

顺序表的删除操作同样涉及到移动元素。删除指定位置的元素后,需要将后续元素前移,以保持顺序表的连续性。

// 删除元素
int deleteElement(SequentialList *list, int index) {
    if (index < 0 || index >= list->length) {
        printf("删除位置无效\n");
        return 0;
    }
    // 从删除位置开始,向前移动元素
    for (int i = index; i < list->length - 1; i++) {
        list->data[i] = list->data[i + 1];
    }
    list->length--;
    return 1;
}

该函数首先检查删除位置是否合法,然后将删除位置之后的所有元素向前移动一个位置,最后减少顺序表的长度。

3.5 查找元素

查找元素的操作可以分为按值查找和按索引查找。

  • 按值查找:找到指定值在顺序表中的位置。
// 查找元素(按值查找)
int findElementByValue(SequentialList *list, int value) {
    for (int i = 0; i < list->length; i++) {
        if (list->data[i] == value) {
            return i;  // 返回找到的索引
        }
    }
    return -1;  // 未找到
}

该函数遍历顺序表中的所有元素,找到与指定值匹配的元素,并返回其索引。如果没有找到,返回 -1。

  • 按索引查找:获取指定索引处的元素。
// 查找元素(按索引查找)
int getElementByIndex(SequentialList *list, int index, int *value) {
    if (index < 0 || index >= list->length) {
        printf("索引无效\n");
        return 0;
    }
    *value = list->data[index];
    return 1;
}

该函数检查索引是否合法,然后通过索引获取元素的值。

3.6 遍历顺序表

遍历顺序表中的所有元素并打印出来。

// 遍历顺序表
void traverseList(SequentialList *list) {
    for (int i = 0; i < list->length; i++) {
        printf("%d ", list->data[i]);
    }
    printf("\n");
}

该函数从头到尾遍历顺序表中的所有元素,并将它们打印到控制台。

4. 顺序表的优缺点

4.1 优点

  1. 随机访问效率高:顺序表支持通过下标访问任意元素,时间复杂度为 O(1),这使得其在需要频繁随机访问的场景中表现优异。
  2. 内存紧凑:顺序表中的元素存储在连续的内存空间中,因此不存在指针的额外内存开销。
  3. 遍历效率高:由于顺序表使用连续的存储空间,遍历顺序表时可以很好地利用 CPU 缓存,提高访问效率。

4.2 缺点

  1. 插入和删除效率低:在顺序表中插入或删除元素时,可能需要移动大量的元素,时间复杂度为 O(n)。
  2. 内存分配不灵活:顺序表需要预先分配连续的内存,当需要扩展容量时,可能需要重新分配内存并复制原有数据,成本较高。
  3. 空间利用率问题:如果预分配的容量过大,会造成内存浪费;如果容量不足,需要频繁扩展,会影响性能。

5. 顺序表的应用场景

顺序表适用于以下场景:

  • 频繁随机访问:顺序表支持 O(1) 的随机访问,适合需要频繁访问任意位置元素的场景。
  • 元素数量相对固定:如果元素数量相对固定,不需要频繁插入和删除,顺序表是一个较好的选择。
  • 需要遍历操作:由于顺序表的元素存储在连续的内存空间中,遍历顺序表时可以充分利用 CPU 缓存,提高效率。

6. 示例代码汇总

下面是一个完整的示例代码,展示了顺序表的基本操作,包括初始化、插入、删除、查找和遍历。

#include <stdio.h>
#include <stdlib.h>

#define INITIAL_CAPACITY 10

// 顺序表结构体定义
typedef struct {
    int *data;
    int length;
    int capacity;
} SequentialList;

// 初始化顺序表
SequentialList* initList(int capacity) {
    SequentialList *list = (SequentialList*)malloc(sizeof(SequentialList));
    if (list == NULL) {
        printf("内存分配失败\n");
        exit(1);
    }
    list->data = (int*)malloc(sizeof(int) * capacity);
    if (list->data == NULL) {
        printf("内存分配失败\n");
        free(list);
        exit(1);
    }
    list->length = 0;
    list->capacity = capacity;
    return list;
}

// 插入元素
int insertElement(SequentialList *list, int index, int value) {
    if (index < 0 || index > list->length) {
        printf("插入位置无效\n");
        return 0;
    }
    if (list->length == list->capacity) {
        int newCapacity = list->capacity * 2;
        int *newData = (int*)realloc(list->data, sizeof(int) * newCapacity);
        if (newData == NULL) {
            printf("内存扩展失败\n");
            return 0;
        }
        list->data = newData;
        list->capacity = newCapacity;
    }
    for (int i = list->length; i > index; i--) {
        list->data[i] = list->data[i - 1];
    }
    list->data[index] = value;
    list->length++;
    return 1;
}

// 删除元素
int deleteElement(SequentialList *list, int index) {
    if (index < 0 || index >= list->length) {
        printf("删除位置无效\n");
        return 0;
    }
    for (int i = index; i < list->length - 1; i++) {
        list->data[i] = list->data[i + 1];
    }
    list->length--;
    return 1;
}

// 查找元素(按值查找)
int findElementByValue(SequentialList *list, int value) {
    for (int i = 0; i < list->length; i++) {
        if (list->data[i] == value) {
            return i;
        }
    }
    return -1;
}

// 查找元素(按索引查找)
int getElementByIndex(SequentialList *list, int index, int *value) {
    if (index < 0 || index >= list->length) {
        printf("索引无效\n");
        return 0;
    }
    *value = list->data[index];
    return 1;
}

// 遍历顺序表
void traverseList(SequentialList *list) {
    for (int i = 0; i < list->length; i++) {
        printf("%d ", list->data[i]);
    }
    printf("\n");
}

// 主函数
int main() {
    SequentialList *list = initList(INITIAL_CAPACITY);
    insertElement(list, 0, 10);
    insertElement(list, 1, 20);
    insertElement(list, 2, 30);
    traverseList(list);

    deleteElement(list, 1);
    traverseList(list);

    int index = findElementByValue(list, 30);
    if (index != -1) {
        printf("元素 30 的索引为: %d\n", index);
    }

    int value;
    if (getElementByIndex(list, 1, &value)) {
        printf("索引 1 处的元素为: %d\n", value);
    }

    // 释放内存
    free(list->data);
    free(list);

    return 0;
}

7. 总结

顺序表是一种使用连续内存存储线性数据的结构,适合需要快速随机访问的应用场景。通过本文的总结,介绍了顺序表的定义、实现、基本操作、优缺点及应用场景。顺序表的实现虽然简单,但其对内存的要求较高,适用于元素数量固定、插入和删除操作较少的情况。在实际开发中,顺序表是基础数据结构之一,可以有效帮助理解和构建更复杂的数据结构。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2221116.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于ssm的校园车辆管理系统的设计与实现

文未可获取一份本项目的java源码和数据库参考。 一、选题背景与意义 &#xff08;一&#xff09;选题背景 自改革开放以来&#xff0c;国家开始加大力度对高等校园进行投入&#xff0c;深化了教育体制的改革&#xff0c;全国许多高等校园应运而生&#xff0c;越来越多的孩子走…

C++ —— 《模板进阶详解》,typedef和class的区别以及用法,非类型模板参数,模板的特化,模板的分离编译

目录 1.非类型模板参数 2.模板特化 2.1 概念 2.2 函数模板特化 2.3 类模板特化 2.3.1 全特化 2.3.2 偏特化 3 模板分离编译 3.1 什么是分离编译 3.2 模板的分离编译 4.模板总结 在讲解模板进阶之前&#xff0c;我想先简单单独聊聊class和typename的用法 我们在平时…

Mac使用Git仓库上传文件夹方法

环境&#xff1a;mac 仓库&#xff1a;GitLink Q: 首先我们先了解到&#xff0c;远程下载&#xff08;用ssh&#xff09;仓库会默认下载到电脑的哪里呢&#xff1f; A: 如下图&#xff0c;路径为/Users/bb&#xff0c;&#xff08;bb是我的用户名&#xff09; 那么我们在将a文…

筑牢理性防线,“卡游启智,理性护航”青少年健康消费倡议发布

新华网北京10月22日电&#xff08;记者周靖杰&#xff09;10月21日&#xff0c;国内针对青少年卡牌盲盒消费的倡议在京发布。 “卡游启智&#xff0c;理性护航”青少年健康消费倡议在京正式发布 近年来&#xff0c;卡牌盲盒在青少年中悄然流行&#xff0c;买卡、拆包、集卡成为…

临时配置linux Bridge网桥

Linux Bridge&#xff08;网桥&#xff09;是用纯软件实现的虚拟交换机&#xff0c;有着和物理交换机相同的功能&#xff0c;例如二层交换&#xff0c;MAC地址学习等。因此我们可以把tun/tap&#xff0c;veth pair等设备绑定到网桥上&#xff0c;就像是把设备连接到物理交换机上…

数据可视化示例

用eclipse软件 java 代码&#xff1a; <!DOCTYPE html> <html> <head> <meta charset"UTF-8"> <!-- 引入ECharts脚本&#xfe63;-> <script src"js/echarts.js"></script> <title&#xff1e;网格…

红日靶场(三)1、环境介绍及环境搭建

1、靶场介绍 红日靶场03是一个用于安全测试和渗透测试的虚拟化环境&#xff0c;可以帮助用户通过模拟攻击和防御场景来提升网络安全技能。该靶场包含了多个虚拟机和网络配置&#xff0c;用户可以在其中进行各种安全测试活动&#xff0c;如信息收集、漏洞利用、权限提升等。 2…

Java项目-基于Springboot的高校党务系统项目(源码+说明).zip

作者&#xff1a;计算机学长阿伟 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、ElementUI等&#xff0c;“文末源码”。 开发运行环境 开发语言&#xff1a;Java数据库&#xff1a;MySQL技术&#xff1a;SpringBoot、Vue、Mybaits Plus、ELementUI工具&#xff1a;IDEA/…

ACL访问控制

要求&#xff1a; PC1与PC2不能通信。PC1可以和PC3通信。PC2可以和PC3通信。 1. VLAN配置 根据拓扑图的连接&#xff0c;PC1、PC2、PC3属于不同的VLAN。我们需要确保交换机上的端口已经正确划分到不同的VLAN。假设交换机接口的VLAN配置已经完成&#xff08;其他博文有)&…

在VMware中安装LInux

1.打开VMware选择新建虚拟机 2.这里建议初学者选择“典型&#xff08;推荐&#xff09;”。 3.已提前准备好 Linux 系统的映像文件&#xff08;.iso 文件&#xff09;&#xff0c;此处可选择“安装程序光盘映像文件”&#xff0c;并通过“浏览”按钮找到要安装 Linux 系统的 i…

“金秋敬老月 浓浓孝老情”2024中益孝行大爱进万家敬老月公益活动走进涿州

为贯彻实施积极应对人口老龄化国家战略&#xff0c;弘扬中华民族孝亲敬老传统美德&#xff0c;10月22日&#xff0c;中益老龄事业发展中心和涿州市东城坊镇人民政府在安享城养老院联合开展“金秋敬老月&#xff0c;浓浓孝老情”2024中益孝行大爱进万家敬老月公益活动。中益老龄…

android openGL ES详解——混合

一、混合概念 混合是一种常用的技巧&#xff0c;通常可以用来实现半透明。但其实它也是十分灵活的&#xff0c;你可以通过不同的设置得到不同的混合结果&#xff0c;产生一些有趣或者奇怪的图象。混合是什么呢&#xff1f;混合就是把两种颜色混在一起。具体一点&#xff0c;就…

《Python游戏编程入门》注-第2章2

《Python游戏编程入门》的“2.2.5 绘制线条”中提到了通过pygame库绘制线条的方法。 1 相关函数介绍 通过pygame.draw模块中的line()函数来绘制线条&#xff0c;该函数的格式如下所示。 line(surface, color, start_pos, end_pos, width1) -> Rect 其中&#xff0c;第一…

面试题:Redis(八)

1. 面试题 2. 锁的特性 单机版同一个jvm虚拟机内&#xff0c;synchronized或者Lock接口 分布式多个不同jvm虚拟机&#xff0c;单机的线程锁机制不再起作用&#xff0c;资源类在不同的服务器之间共享 一个靠谱分布式锁所需的条件 3. 手写分布式锁 3.1 独占性&#xff08;线程安…

VScode运行C语言终端输出中文乱码问题解决方案

VScode运行C语言输出中文乱码问题解决方案 由于 VSCode 的终端是对系统的 cmd 命令行工具的调用&#xff0c;而 cmd 的默认编码为 GBK。当我们在 VSCode 中以 UTF-8 编码进行代码编写且代码里含有中文字符时&#xff0c;在终端运行代码便会出现中文乱码现象。要解决此问题&…

传统企业应该如何突破管理瓶颈?

传统企业应该如何突破管理瓶颈&#xff1f; 【导读】 作为传统企业&#xff0c;有很多传承的传统机制&#xff0c;然而在市场机制下&#xff0c;越来越能够深刻感受到外部市场变化快的特点&#xff0c;在逐步适应以市场为导向的环境下&#xff0c;传统企业自身如何做好管理工…

C/C++每日一练:实现一个环形队列

队列&#xff08;queue&#xff09; 队列是一种先进先出&#xff08;FIFO&#xff0c;First In First Out&#xff09; 的数据结构&#xff0c;类似于排队的场景。最先进入队列的元素最先被处理&#xff0c;而后加入的元素则排在队列的末尾。 常见的队列操作&#xff1a; 入队…

【linux 多进程并发】0301 Linux创建后台服务进程,daemon进程,自己的进程可以被一号进程接管啦

0301 Linux创建后台进程 ​专栏内容&#xff1a; postgresql使用入门基础手写数据库toadb并发编程 个人主页&#xff1a;我的主页 管理社区&#xff1a;开源数据库 座右铭&#xff1a;天行健&#xff0c;君子以自强不息&#xff1b;地势坤&#xff0c;君子以厚德载物. 一、概述…

Matlab实现鲸鱼优化算法(WOA)求解路径规划问题

目录 1.内容介绍 2.部分代码 3.实验结果 4.内容获取 1内容介绍 鲸鱼优化算法&#xff08;WOA&#xff09;是一种受自然界座头鲸捕食行为启发的优化算法&#xff0c;它通过模拟座头鲸的环绕猎物、螺旋游动和搜索猎物三种主要行为来探索和优化问题的解。WOA因其强大的全局搜索能…