龙空技术网

考研《数据结构》知识点总结(2)——线性表

技术闲聊DD 114

前言:

现在兄弟们对“内存分配失败是什么意思”大约比较讲究,同学们都想要学习一些“内存分配失败是什么意思”的相关知识。那么小编在网络上汇集了一些有关“内存分配失败是什么意思””的相关知识,希望我们能喜欢,大家一起来了解一下吧!

2.1 线性表概述

线性表是具有相同数据类型的n个数据元素的有限序列。除第一个元素外,每个元素有且仅有一个直接前驱;除最后一个元素外,每个元素有且仅有一个直接后继。

注意:线性表是一种逻辑结构,表示元素之间一对一的相邻关系。顺序表和链表是指存储结构。两者属于不同层面的概念,因此不要混淆。

2.2 线性表的顺序表示和实现

概念

线性表的顺序存储又称为顺序表。它是用一组地址连续的存储单元,依次存储线性表中的数据元素,从而使得逻辑上相邻的两个元素在物理位置上也相邻。

顺序表的特点是表中元素的逻辑顺序与其物理顺序相同。

顺序表最主要的特点是随机访问(随机存取),即通过首地址和元素序号可以在O(1)的时间内找到指定的元素。但插入和删除操作需要移动大量元素。

顺序表的存储密度高,每个结点只存储数据元素。

顺序表插入操作

最好情况:在表尾插入,元素后移语句不执行,时间复杂度为O(1)

最坏情况:在表头插入,元素后移语句将执行n次,时间复杂度为O(n)

平均情况:平均时间复杂度为O(n)

顺序表删除操作

最好情况:删除表尾元素,无需移动元素,时间复杂度为O(1)

最坏情况:删除表头元素,需要移动除第一个元素外的所有元素,时间复杂度为O(n)

平均情况:平均时间复杂度为O(n)

顺序表按值查找(顺序查找)

最好情况:查找的元素就在表头,仅需比较一次,时间复杂度为O(1)

最坏情况:查找的元素在表尾或不存在时,需要比较n次,时间复杂度为O(n)

平均情况:平均时间复杂度为O(n)

2.3 线性表的链式表示和实现

概念

链式存储线性表时,不需要使用地址连续的存储单元,即它不要求逻辑上相邻的两个元素在物理位置上也相邻,对线性表的插入和删除不需要移动元素,只需要修改指针。

单链表是非随机存储的存储结构,即不能直接找到表中某个特定的结点。

头结点和头指针的区分:不管带不带头结点,头指针始终指向链表的第一个结点,而头结点是带头结点链表中的第一个结点,结点内通常不存储信息。

采用头插法建立单链表

从一个空表开始,生成新结点,并将读取到的数据存放在新结点的数据域中,然后将新结点插入到当前链表的表头,即头结点之后。读入数据的顺序与生成的链表中元素的顺序是相反的。每个结点插入的时间为O(1),设单链表长为n,则总的时间复杂度为O(n)。

采用尾插法建立单链表

将新结点插入到当前链表的表尾上,为此必须增加一个尾指针r,使其始终指向当前链表的尾结点。时间复杂度为O(n)。

按序号查找结点值

在单链表中从第一个结点出发,顺指针next域逐个往下搜索,直到找到第i个结点为止,否则返回最后一个结点指针域NULL。

时间复杂度为O(n)。

按值查找表结点

从单链表第一个结点出发,由前往后依次比较表中各结点数据域的值,若某结点数据域的值等于给定值e,则返回该结点的指针,否则返回NULL。

时间复杂度为O(n)。

插入结点操作

先检查插入位置的合法性,然后找到待插入位置的前驱结点,即第i-1个结点,再在其后插入新结点。

主要的时间开销在于查找第i-1个元素,时间复杂度为O(n)。若是在给定的结点后面插入新结点,则时间复杂度仅为O(1)。

删除结点操作

先检查删除位置的合法性,然后找到待删除位置的前驱结点,即第i-1个结点,再将其删除。时间复杂度为O(n)。

双链表

双链表结点中有两个指针prior和next,分别指向其前驱结点和后继结点。双链表中执行按值查找和按位查找的操作和单链表相同,但双链表插入、删除操作的时间复杂度仅为O(1)。

循环单链表

循环单链表和单链表的区别在于,表中最后一个结点的指针不是NULL,而改为指向头结点,从而整个链表形成一个环。故表中没有指针域为NULL的结点。

循环双链表 L中,某结点*p为尾结点时,p->next=L;当循环双链表为空表时,其头结点的prior域和next域都等于L。

2.4 静态链表

静态链表是借助数组来描述线性表的链式存储结构,结点也有数据域data和指针域next,这里的指针是结点的相对地址(数组下标),又称为游标。

2.5 顺序表和链表的比较

1. 存取方式:顺序表可以顺序存取,也可以随机存取,链表只能从表头顺序存取元素。

2. 逻辑结构与物理结构:采用顺序存储时,逻辑上相邻的元素,其对应的物理存储位置也相邻。而采用链式存储时,逻辑上相邻的元素,其物理存储位置则不一定相邻。

3. 查找、插入和删除操作:对于按值查找,当顺序表在无序的情况下,两者的时间复杂度均为O(n);而当顺序表有序时,可采用折半查找,时间复杂度为O(logn)。对于按序号查找,顺序表支持随机访问,时间复杂度仅为O(1),而链表的平均时间复杂度为O(n)。链表插入、删除较优,只需修改相关结点的指针域即可。

4. 空间分配:链式存储的结点空间只在需要的时候申请分配。

通常较稳定的线性表选择顺序存储,而频繁做插入、删除操作的线性表(即动态性较强)宜选择链式存储。

2.6 线性表的基本运算

顺序表的运算

#include <stdio.h>#include <stdlib.h>#define Size 5typedef struct {    int* head; //定义一个名为head的长度不确定的数组,也叫“动态数组”    int length; //记录当前顺序表的长度    int size; //记录顺序表的存储容量}Table;//创建顺序表void initTable(Table* t) {    //构造一个空的顺序表,动态申请存储空间    t->head = (int*)malloc(Size * sizeof(int));    //如果申请失败,作出提示并直接退出程序    if (!t->head)    {        printf("初始化失败");        exit(0);    }    //空表的长度初始化为0    t->length = 0;    //空表的初始存储空间为Size    t->size = Size;}//插入函数,其中,elem为插入的元素,add为插入到顺序表的位置void insertTable(Table* t, int elem, int add){    int i;    //如果插入元素位置比整张表的长度+1还大(如果相等,是尾随的情况),或者插入的位置本身不存在,程序作为提示并自动退出    if (add > t->length + 1 || add < 1) {        printf("插入位置有问题\n");        return;    }    //做插入操作时,首先需要看顺序表是否有多余的存储空间提供给插入的元素,如果没有,需要申请    if (t->length == t->size) {        t->head = (int*)realloc(t->head, (t->size + 1) * sizeof(int));        if (!t->head) {            printf("存储分配失败\n");            return;        }        t->size += 1;    }    //插入操作,需要将自插入位置之后的所有元素全部后移一位    for (i = t->length - 1; i >= add - 1; i--) {        t->head[i + 1] = t->head[i];    }    //后移完成后,直接插入元素    t->head[add - 1] = elem;    t->length++;}//删除函数void delTable(Table* t, int add) {    int i;    if (add > t->length || add < 1) {        printf("被删除元素的位置有误\n");        return;    }    //删除操作    for (i = add; i < t->length; i++) {        t->head[i - 1] = t->head[i];    }    t->length--;}//查找函数int selectTable(Table t, int elem) {    int i;    for (i = 0; i < t.length; i++) {        if (t.head[i] == elem) {            return i + 1;        }    }    return -1;}//更改函数void amendTable(Table* t, int elem, int newElem) {    int add = selectTable(*t, elem);    if (add == -1) {        printf("顺序表中没有找到目标元素\n");        return;    }    t->head[add - 1] = newElem;}//输出顺序表中的元素void displayTable(Table t) {    int i;    for (i = 0; i < t.length; i++) {        printf("%d ", t.head[i]);    }    printf("\n");}int main() {    int i,add;    Table t = { NULL,0,0 };    initTable(&t);    for (i = 1; i <= Size; i++) {        t.head[i - 1] = i;        t.length++;    }    printf("原顺序表:\n");    displayTable(t);    printf("删除元素1:\n");    delTable(&t, 1);    displayTable(t);    printf("在第2的位置插入元素5:\n");    insertTable(&t, 5, 2);    displayTable(t);    printf("查找元素3的位置:\n");    add = selectTable(t, 3);    printf("%d\n", add);    printf("将元素3改为6:\n");    amendTable(&t, 3, 6);    displayTable(t);    return 0;}

运行结果如下:

链表的基本运算

#include <stdio.h>#include <stdlib.h>//链表中节点的结构typedef struct link {    int  elem;    struct link* next;}Link;Link* initLink() {    int i;    //1、创建头指针    Link* p = NULL;    //2、创建头结点    Link* temp = (Link*)malloc(sizeof(Link));    temp->elem = 0;    temp->next = NULL;    //头指针指向头结点    p = temp;    //3、每创建一个结点,都令其直接前驱结点的指针指向它    for (i = 1; i < 5; i++) {        //创建一个结点        Link* a = (Link*)malloc(sizeof(Link));        a->elem = i;        a->next = NULL;        //每次 temp 指向的结点就是 a 的直接前驱结点        temp->next = a;        //temp指向下一个结点(也就是a),为下次添加结点做准备        temp = temp->next;    }    return p;}//p为链表,elem为目标元素,add为要插入的位置void insertElem(Link* p, int elem, int add) {    int i;    Link* c = NULL;    Link* temp = p;//创建临时结点temp    //首先找到要插入位置的上一个结点    for (i = 1; i < add; i++) {        temp = temp->next;        if (temp == NULL) {            printf("插入位置无效\n");            return;        }    }    //创建插入结点c    c = (Link*)malloc(sizeof(Link));    c->elem = elem;    //① 将新结点的 next 指针指向插入位置后的结点    c->next = temp->next;    //② 将插入位置前结点的 next 指针指向插入结点;    temp->next = c;}//p为原链表,elem 为要删除的目标元素int delElem(Link* p, int elem) {    Link* del = NULL, *temp = p;    int find = 0;    //1、找到目标元素的直接前驱结点    while (temp->next) {        if (temp->next->elem == elem) {            find = 1;            break;        }        temp = temp->next;    }    if (find == 0) {        return -1;//删除失败    }    else    {        //标记要删除的结点        del = temp->next;        //2、将目标结点从链表上摘除        temp->next = temp->next->next;        //3、释放目标结点        free(del);        return 1;    }}//p为原链表,elem表示被查找元素int selectElem(Link* p, int elem) {    int i = 1;    //带头结点,p 指向首元结点    p = p->next;    while (p) {        if (p->elem == elem) {            return i;        }        p = p->next;        i++;    }    return -1;//返回-1,表示未找到}//p 为有头结点的链表,oldElem 为旧元素,newElem 为新元素int amendElem(Link* p, int oldElem, int newElem) {    p = p->next;    while (p) {        if (p->elem == oldElem) {            p->elem = newElem;            return 1;        }        p = p->next;    }    return -1;}//输出链表中各个结点的元素void display(Link* p) {    p = p->next;    while (p) {        printf("%d ", p->elem);        p = p->next;    }    printf("\n");}//释放链表void Link_free(Link* p) {    Link* fr = NULL;    while (p->next)    {        fr = p->next;        p->next = p->next->next;        free(fr);    }    free(p);}int main() {    Link* p = initLink();    printf("初始化链表为:\n");    display(p);    printf("在第 3 的位置上添加元素 6:\n");    insertElem(p, 6, 3);    display(p);    printf("删除元素4:\n");    delElem(p, 4);    display(p);    printf("查找元素 2:\n");    printf("元素 2 的位置为:%d\n", selectElem(p, 2));    printf("更改元素 1 的值为 6:\n");    amendElem(p, 1, 6);    display(p);    Link_free(p);    return 0;}

标签: #内存分配失败是什么意思