IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 数据结构精解(2):线性表(顺序表和链表) -> 正文阅读

[数据结构与算法]数据结构精解(2):线性表(顺序表和链表)

文章结构:
第一部分是描述相关信息,第二部分是相关操作(初始化、增删改查),第三部分是解答相关习题(《数据结构–严蔚敏》),第四部分是leetcode题目(一道easy+一道middle)

青春的幻想既狂热又可爱。一约肖特豪斯

线性表

线性表是由相同数据类型的 nn 个数据元素 a0,a1,an?1组成的有限序列。一个数据元素可以由若干个数据项组成。若用 LL 命名线性表,则其一般表示如下:

L=(a0,a1……an?1)

顺序表

信息描述

顺序表是在计算机内存中以数组形式保存的线性表,是指用一组地址连续的存储单元依次存储数据元素的线性结构。

设顺序表的第一个元素a0,的存储地址为 Loc(a0),每个元素占 d 个存储空间,则第 i 个元素的地址为

Loc(a i?1)=Loc(a0)+(i?1)×d

顺序表在程序中通常用一维数组实现,一维数组可以是静态分配的,也可以是动态分配的。

  • 在静态分配时,由于数组的大小和空间是固定的,一旦空间占满,就无法再新增数据,否则会导致数据溢出。
  • 而在动态分配时,存储数组的空间在程序执行过程中会动态调整大小,当空间占满时,可以另行开辟更大的存储空间来储存数据。

优点和缺点:

  • 顺序表最主要的特点是可以进行 随机访问,即可以通过表头元素的地址和元素的编号(下标),在 O(1) 的时间复杂度内找到指定的元素。
  • 顺序表的不足之处是插入和删除操作需要移动大量的元素,从而保持逻辑上和物理上的连续性。

为什么不用数组?在函数中定义的数组是在栈段中的,但是栈段在内存中被限制了大小

初始化、增删改查

顺序表的动态存储分配

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

typedef struct Vector {
	int* data;
	int len;
	int size;
}Vec;
Vec* init(int n) {
	Vec* v = (Vec*)malloc(sizeof(Vec));
	v->data = (int*)malloc(sizeof(int) * n);
	v->size = n;  // 最大容量
	v->len = 0;  // 当前元素个数
	return v;
}
void freeVec(Vec* v) {
	if (v) {
		free(v->data);
		free(v);
	}
	return;
}
int insert(Vec* v, int idx, int val) {
	if (v == NULL) {
		return 0;
	}
	if (idx < 0 || idx > v->len) {
		return 0;
	}
	if (v->len == v->size) {
		return 0;
	}
	memcpy(v->data + idx + 1, v->data + idx, sizeof(v->len - idx));
	v->data[idx] = val;
	v->len++;
	return 1;
}
void delete(Vec* v, int idx) {
	if (v == NULL) {
		return 0;
	}
	if (idx < 0 ||| idx >= v->len) {
		return 0;
	}
	memcpy(v->data + idx, v->data + idx + 1, sizeof(v->len - idx - 1));
	v->len--;
	return 1;
}
int expand(Vec* v) {
	if (v == NULL) {
		return 0;
	}
	int expsize = v->size;
	while (expsize) {
		// realloc返回一个指向新空间的指针,原来的数据会被复制过去
		int tmp* = realloc(v->data, sizeof(int) * (v->size + expsize));
		if (tmp) {
			break;
		}
		expsize >>= 2;
	}
	v->data = tmp;
	v->size += expsize;
	return 1;
}

链表

信息描述

链表想象成火车,火车头便是链表的表头,每节车厢就是链表的元素,车厢里载的人和物就是元素的数据域,连接车厢的部件就是元素的指针。由此,我们可以得出链表的一个特点:元素之间前后依赖,串联而成。

为什么不用顺序表?顺序表的插入和删除需要移动大量元素,浪费资源

链表是种物理存储单元上非连续、非顺序的存储结构,数据元素的逻辑顺序是
通过链表中的指针链接次序实现的数据结构

我个人觉得链表把头结点,头指针,第一个结点和最后一个结点的位置弄懂了就能懂了,我个人的理解是:头指针只是一个指针,初始化一个链表类型的指针后她就是一个空链表,而这个头指针在传入函数的时候该头指针代表是一个头结点,但是这个头结点可以为NULL

初始化、增删改查

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

typedef struct Node{
    int data;
    struct Node *next;
}Node, *LinkedList;

LinkedList insert(LinkedList head, Node *node, int index) {
    if (head == NULL) {
        if (index != 0) {
            printf("failed\n");
            return head;
        }
        head = node;
        printf("success\n");
        return head;
    }
    if (index == 0) {
        node->next = head;
        head = node;
        printf("success\n");
        return head;
    }
    Node *current_node = head;
    int count = 0;
    while (current_node->next != NULL && count < index - 1) {
        current_node = current_node->next;
        count++;
    }
    if (count == index - 1) {
        node->next = current_node->next;
        current_node->next = node;
        printf("success\n");
        return head;
    }
    printf("failed\n");
    return head;
}

void output(LinkedList head) {
    if (head == NULL) {
        return;
    }
    Node *current_node = head;
    while (current_node != NULL) {
        printf("%d ", current_node->data);
        current_node = current_node->next;
    }
    printf("\n");
}

LinkedList delete_node(LinkedList head, int index) {
    if (head == NULL) {
        printf("failed\n");
        return head;
    }
    LinkedList current_node = head;
    if (index == 0) {
        head = head->next;
        free(current_node);
        printf("success\n");
        return head;
    }
    int count = 0;
    while (current_node->next != NULL && count < index-1) {
        current_node = current_node->next;
        count++;
    }
    if (count == index-1 && current_node->next != NULL) {
        LinkedList node = current_node->next;
        current_node->next = node->next;
        free(node);
        printf("success\n");
        return head;
    }
    else {
        printf("failed\n");
        return head;
    }
}

LinkedList reverse(LinkedList head) {
    if (head == NULL) return head;
    LinkedList current_node, next_node;
    current_node = head->next;
    head->next = NULL;
    while (current_node != NULL) {
        next_node = current_node->next;
        current_node->next = head;
        head = current_node;
        current_node = next_node;
    }
    return head;
}

void clear(LinkedList head) {
    Node *current_node = head;
    while (current_node != NULL) {
        Node *delete_node = current_node;
        current_node = current_node->next;
        free(delete_node);
    }
}

int main() {
    LinkedList linkedlist = NULL;
    int n;
    int a, b;
    scanf("%d", &n);
    for (int i = 0; i < n; ++i) {
        int signal;
        scanf("%d", &signal);
        if (signal == 1) {
            scanf("%d %d", &a, &b);
            LinkedList node = (LinkedList)malloc(sizeof(Node));
            node->data = b;
            node->next = NULL;
            linkedlist = insert(linkedlist, node, a);
        }
       else if (signal == 2) {
            output(linkedlist);
       }
       else if (signal == 3) {
           scanf("%d", &a);
           linkedlist = delete_node(linkedlist,a);
       }
      else if (signal == 4) {
          linkedlist = reverse(linkedlist);
      }
    }
    return 0;
}

特殊链表

循环链表:是另一种形式的链式存储结构,她的特点是表中最后一个结点的指针域指向头结点,整个链表形成一个环。由此,从表中任一结点触发均可找到表中其他结点

LinkedList insert(LinkedList head, Node *node, int index) {
    if (head == NULL) {		// 如果没有头结点(空表)
        if (index != 0) {
            return head;
        }
        head = node;	// 让该结点成为头结点
        head->next = head;	// 头结点指向自己
        return head;
    }
    if (index == 0) {		// 如果有头结点
        node->next = head->next;	// 让node指向头结点的下一个结点
        head->next = node;	// 让头结点指向node
        return head;
    }
    Node *current_node = head->next;
    int count = 0;
    while (current_node != head && count < index - 1) {
        current_node = current_node->next;
        count++;
    }
    if (count == index - 1) {
        node->next = current_node->next;
        current_node->next = node;
    }
    if (node == head->next) {
        head = node;
    }
    return head;
}

双向链表:在单链表中找到下一个结点的时间复杂度为O(1),找上一个结点的时间复杂度为O(n),为了克服这种缺点,可以利用双向链表。

typedef struct DuLNode {
	int data;
	struct DuLNode* next;
	struct DuLNode* prior;
};

静态链表

#define MAXSIZE 100
typedef struct {
	int data;
	int cur;
}Lisk[MAXSIZE];

Leetcode

leetcode上面的题目一般是头结点即第一个结点(即无头结点)

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-07-29 11:53:53  更:2021-07-29 11:54:09 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年5日历 -2024/5/6 6:47:53-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码