1、第二讲第二讲线性表线性表线性结构的线性结构的基本特征基本特征为为:1集合中必存在唯一的一个集合中必存在唯一的一个“第一元素第一元素”;2集合中必存在唯一的一个集合中必存在唯一的一个“最后元素最后元素”;3除最后元素在外,均有除最后元素在外,均有 唯一的后继唯一的后继;4除第一元素之外,均有除第一元素之外,均有 唯一的前驱唯一的前驱。线性结构线性结构是一个数据元素的是一个数据元素的有序有序(次序)集(次序)集线性表线性表是一种最简单的线性结构线性结构2.1 线性表的类型定义线性表的类型定义2.3 线性表类型的实现线性表类型的实现 链式映象链式映象2.4 一元多项式的表示一元多项式的表示2.2 线
2、性表类型的实现线性表类型的实现 顺序映象顺序映象主要内容主要内容2.1线性表的类型定义线性表的类型定义抽象数据类型线性表线性表的定义如下:ADT List 数据对象数据对象:D ai|ai ElemSet,i=1,2,.,n,n0 称 n 为线性表的表长表长;称 n=0 时的线性表为空表空表。数据关系数据关系:R1|ai-1,aiD,i=2,.,n 设线性表为(a1,a2,.,ai,.,an),称 i 为 ai 在线性表中的位序位序。基本操作:基本操作:结构初始化操作结构初始化操作结构销毁操作结构销毁操作 引用型操作引用型操作 加工型操作加工型操作 ADT List InitList(&L)操
3、作结果:操作结果:构造一个空的线性表L。初始化操作初始化操作 结构销毁操作结构销毁操作DestroyList(&L)初始条件:操作结果:线性表 L 已存在。销毁线性表 L。ListEmpty(L)ListLength(L)PriorElem(L,cur_e,&pre_e)NextElem(L,cur_e,&next_e)GetElem(L,i,&e)LocateElem(L,e,compare()ListTraverse(L,visit()引用型操作引用型操作 ListEmpty(L)初始条件:操作结果:线性表L已存在。若L为空表,则返回TRUE,否则FALSE。(线性表判空)ListLeng
4、th(L)初始条件:操作结果:线性表L已存在。返回L中元素个数。(求线性表的长度)PriorElem(L,cur_e,&pre_e)初始条件:操作结果:线性表L已存在。若cur_e是L的元素,但不是第一个,则用pre_e 返回它的前驱,否则操作失败,pre_e无定义。(求数据元素的前驱)NextElem(L,cur_e,&next_e)初始条件:操作结果:线性表L已存在。若cur_e是L的元素,但不是最后一个,则用next_e返回它的后继,否则操作失败,next_e无定义。(求数据元素的后继)GetElem(L,i,&e)初始条件:操作结果:线性表L已存在,且 1iLengthList(L)。
5、用 e 返回L中第 i 个元素的值。(求线性表中某个数据元素)LocateElem(L,e,compare()初始条件:操作结果:线性表L已存在,e为给定值,compare()是元素判定函数。返回L中第中第1个个与e满足满足关系compare()的元素的位序。若这样的元素不存在,则返回值为0。(定位函数)ListTraverse(L,visit()初始条件:操作结果:线性表L已存在,Visit()为某个访问函数。依次依次对L的每个元素调用函数visit()。一旦visit()失败,则操作失败。(遍历线性表)加工型操作加工型操作 ClearList(&L)PutElem(&L,i,&e)List
6、Insert(&L,i,e)ListDelete(&L,i,&e)ClearList(&L)初始条件:操作结果:线性表L已存在。将L重置为空表。(线性表置空)PutElem(&L,i,&e)初始条件:操作结果:线性表L已存在,且 1iLengthList(L)。L中第i个元素赋值同e的值。(改变数据元素的值)ListInsert(&L,i,e)初始条件:操作结果:线性表L已存在,且 1iLengthList(L)+1。在L的第i个元素之前插入插入新的元素e,L的长度增1。(插入数据元素)ListDelete(&L,i,&e)初始条件:操作结果:线性表L已存在且非空,1iLengthList(L
7、)。删除L的第i个元素,并用e返回其值,L的长度减1。(删除数据元素)利用上述定义的线性表线性表 可以实现其它更复杂的操作例例 2例例 3例例 1 假设:有两个集合集合 A 和和 B 分别用两个线性表线性表 LA 和和 LB 表示,即:线性表中的数据元素即为集合中的成员。现要求一个新的集合现要求一个新的集合AAB。例例1 要求对线性表作如下操作:扩大线性表 LA,将存在于线性表存在于线性表LB 中中而不存在于线性表不存在于线性表 LA 中中的数据元素插入到线性表插入到线性表 LA 中中去。上述问题可演绎为:1从线性表LB中依次察看每个数据元素;2依值在线性表LA中进行查访;3若不存在,则插入之
8、。GetElem(LB,i)e LocateElem(LA,e,equal()ListInsert(LA,n+1,e)操作步骤:操作步骤:GetElem(Lb,i,e);/取取Lb中第中第i个数据元素赋给个数据元素赋给e if(!LocateElem(La,e,equal()ListInsert(La,+La_len,e);/La中不存在和中不存在和 e 相同的数据元素,则插入之相同的数据元素,则插入之void union(List&La,List Lb)La_len=ListLength(La);/求线性表的长度求线性表的长度 Lb_len=ListLength(Lb);for(i=1;i=
9、Lb_len;i+)/union 已知已知一个非纯集合非纯集合 B,试构造构造一个纯集合纯集合 A,使使 A中只包含中只包含 B 中所有值各中所有值各不相不相 同的数据元素同的数据元素。仍选用线性表线性表表示集合。例例 2 2集合集合 B集合集合 A从集合 B 取出物件放入集合 A要求集合A中同样物件不能有两件以上同样物件不能有两件以上因此,算法的策略应该和例算法的策略应该和例1相同相同void union(List&La,List Lb)La_len=ListLength(La);Lb_len=ListLength(Lb);/union GetElem(Lb,i,e);/取取Lb中第中第 i
10、 个数据元素赋给个数据元素赋给 e if(!LocateElem(La,e,equal()ListInsert(La,+La_len,e);/La中不存在和中不存在和 e 相同的数据元素,则插入之相同的数据元素,则插入之for(i=1;i=Lb_len;i+)InitList(La);/构造(空的)线性表LA若线性表中的数据元素相互之间可以比比较较,并且数据元素在线性表中依值非递依值非递减或非递增有序减或非递增有序排列,即 aiai-1 或 aiai-1(i=2,3,n),则称该线性表为有序表有序表(Ordered List)(Ordered List)。试改变结构,以有序表有序表表示集合。例
11、如例如:(2,3,3,5,6,6,6,8,12)对集合 B 而言,值相同的数据元素必定相邻;值相同的数据元素必定相邻;对集合 A 而言,数据元素依值从小至大的顺序插入。数据元素依值从小至大的顺序插入。因此,数据结构改变了,数据结构改变了,解决问题的策略也相应要改变。解决问题的策略也相应要改变。void purge(List&La,List Lb)InitList(LA);La_len=ListLength(La);Lb_len=ListLength(Lb);/求线性表的长度求线性表的长度 for(i=1;i=Lb_len;i+)/purge GetElem(Lb,i,e);/取取Lb中第中第i
12、个数据元素赋给个数据元素赋给 eif(ListEmpty(La)|!equal(en,e)ListInsert(La,+La_len,e);en=e;/La中不存在和中不存在和 e 相同的数据元素,则插入之相同的数据元素,则插入之 已知线行表已知线行表 La 和和 Lb非递减有序,非递减有序,现要求一个新的集合现要求一个新的集合Lc La Lb,要要求求Lc非递减有序非递减有序。例例 3 3void MergeList(List La,List Lb,List&Lc)/本算法将非递减的有序表 La 和 Lb 归并为 Lc/merge_listwhile(i=La_len)&(j=Lb_len)
13、/La 和和 Lb 均不空均不空 while(i=La_len)/若 La 不空while(j=Lb_len)/若 Lb 不空InitList(Lc);/构造空的线性表 Lci=j=1;k=0;La_len=ListLength(La);Lb_len=ListLength(Lb);/La 和 Lb 均非空,i=j=1,k=0 GetElem(La,i,ai);GetElem(Lb,j,bj);if(ai=bj)/将 ai 插入到 Lc 中 ListInsert(Lc,+k,ai);+i;else /将 bj 插入到 Lc 中 ListInsert(Lc,+k,bj);+j;while(i=La
14、_len)/当La不空时 GetElem(La,i+,ai);ListInsert(Lc,+k,ai);/插入插入 La 表中剩余元素表中剩余元素 while(j=Lb_len)/当Lb不空时 GetElem(Lb,j+,bj);ListInsert(Lc,+k,bj);/插入插入 Lb 表中剩余元素表中剩余元素最简单的一种顺序映象方法是:最简单的一种顺序映象方法是:令令 y y 的存储位置和的存储位置和 x x 的存储位置的存储位置相邻相邻。顺序映象顺序映象 以以 x 的存储位置和的存储位置和 y 的存储位置的存储位置之间某种关系表示逻辑关系之间某种关系表示逻辑关系。用一组地址连续地址连续的
15、存储单元 依次存放依次存放线性表中的数据元素 a1 a2 ai-1 ai an线性表的线性表的起始地址起始地址称作线性表的基地址基地址以“存储位置相邻存储位置相邻”表示有序对 即:LOC(ai)=LOC(ai-1)+C 一个数据元素所占存储量一个数据元素所占存储量所有数据元素的存储位置均取决于所有数据元素的存储位置均取决于 第一个数据元素的存储位置第一个数据元素的存储位置 LOC(ai)=LOC(a1)+(i-1)C 基地址基地址顺序映像的顺序映像的 C 语言描述语言描述typedef struct SqList;/俗称 顺序表顺序表#define LIST_INIT_SIZE 80 /线性表
16、存储空间的初始分配量#define LISTINCREMENT 10 /线性表存储空间的分配增量ElemType*elem;/存储空间基址int length;/当前长度int listsize;/当前分配的存储容量 /(以sizeof(ElemType)为单位)线性表的基本操作在顺序表中的实现线性表的基本操作在顺序表中的实现InitList(&L)/结构初始化结构初始化LocateElem(L,e,compare()/查找查找ListInsert(&L,i,e)/插入元素插入元素ListDelete(&L,i)/删除元素删除元素Status InitList_Sq(SqList&L)/构造一
17、个空的线性表 /InitList_Sq算法时间复杂度时间复杂度:O(1)L.elem=(ElemType*)malloc(LIST_ INIT_SIZE sizeof(ElemType);if(!L.elem)exit(OVERFLOW);L.length=0;L.listsize=LIST_INIT_SIZEreturn OK;LocateElem(L,e,compare()/查找操作查找操作23 75 41 38 54 62 17L.elemL.lengthL.listsizee=38pppppi 1 2 3 4 1 850p可见,基本操作是:将顺序表中的元素逐个和给定值 e 相比较。in
18、t LocateElem_Sq(SqList L,ElemType e,Status(*compare)(ElemType,ElemType)/在顺序表中查询第一个满足判定条件的数据元素,在顺序表中查询第一个满足判定条件的数据元素,/若存在,则返回它的位序,否则返回若存在,则返回它的位序,否则返回 0 0/LocateElem_Sq O(ListLength(L)算法的算法的时间复杂度时间复杂度为:为:i=1;/i i 的初值为第的初值为第 1 1 元素的位序元素的位序p=L.elem;/p p 的初值为第的初值为第 1 1 元素的存储位置元素的存储位置while(i=L.length&!(*
19、compare)(*p+,e)+i;if(i=L.length)return i;else return 0;(*compare)(*p+,e)线性表操作 ListInsert(&L,i,e)的实现:首先分析首先分析:插入元素时,线性表的逻辑结构逻辑结构发生什么变化发生什么变化?(a1,ai-1,ai,an)改变为 (a1,ai-1,e,ai,an)a1 a2 ai-1 ai ana1 a2 ai-1 ai ean,表的长度增加 Status ListInsert_Sq(SqList&L,int i,ElemType e)/在顺序表L的第 i 个元素之前插入新的元素e,/i 的合法范围为 1i
20、L.length+1/ListInsert_Sq 算法时间复杂度算法时间复杂度为为:O(ListLength(L)q=&(L.elemi-1);/q 指示插入位置for(p=&(L.elemL.length-1);p=q;-p)*(p+1)=*p;/插入位置及之后的元素右移元素右移*q=e;/插入e+L.length;/表长增1return OK;元素右移元素右移if(L.length=L.listsize)/当前存储空间已满,增加分配 newbase=(ElemType*)realloc(L.elem,(L.listsize+LISTINCREMENT)*sizeof(ElemType);i
21、f(!newbase)exit(OVERFLOW);/存储分配失败 L.elem=newbase;/新基址 L.listsize+=LISTINCREMENT;/增加存储容量if(i L.length+1)return ERROR;/插入位置不合法考虑移动元素的平均情况考虑移动元素的平均情况:假设在第 i 个元素之前插入的概率为 ,则在长度为n 的线性表中插入一个元素所需插入一个元素所需移动元素次数的期望值移动元素次数的期望值为:若假定假定在线性表中任何一个位置上进行插入插入的概率的概率都是相等相等的,则移动元素的期望值移动元素的期望值为:21 18 30 75 42 56 8721 18 3
22、0 75例如:ListInsert_Sq(L,5,66)L.length-10pppq87564266q=&(L.elemi-1);/q 指示插入位置for(p=&(L.elemL.length-1);p=q;-p)*(p+1)=*p;p线性表操作 ListDelete(&L,i,&e)的实现:首先分析:删除元素时,线性表的逻辑结构发生什么变化?(a1,ai-1,ai,ai+1,an)改变为 (a1,ai-1,ai+1,an)ai+1 an,表的长度减少a1 a2 ai-1 ai ai+1 ana1 a2 ai-1 Status ListDelete_Sq (SqList&L,int i,El
23、emType&e)/ListDelete_Sqfor(+p;p=q;+p)*(p-1)=*p;/被删除元素之后的元素左移被删除元素之后的元素左移-L.length;/表长减表长减1 1return OK;算法时间复杂度算法时间复杂度为为:O(ListLength(L)p=&(L.elemi-1);/p 为被删除元素的位置为被删除元素的位置e=*p;/被删除元素的值赋给被删除元素的值赋给 eq=L.elem+L.length-1;/表尾元素的位置表尾元素的位置if(i L.length)return ERROR;/删除位置不合法删除位置不合法元素左移元素左移考虑移动元素的平均情况考虑移动元素的平
24、均情况:假设删除第 i 个元素的概率为 ,则在长度为n 的线性表中删除一个元素所需移动元素次数的期望值移动元素次数的期望值为:若假定在线性表中任何一个位置上进行删除的概率都是相等的,则移动元素的期望值移动元素的期望值为:21 18 30 75 42 56 8721 18 30 75L.length-10pppq8756p=&(L.elemi-1);q=L.elem+L.length-1;for(+p;p next;j=1;/p p指向第一个结点,指向第一个结点,j j为计数器为计数器while(p&jnext;+j;/顺指针向后查找,直到顺指针向后查找,直到 p p 指向第指向第 i i 个元
25、素个元素 /或或 p p 为空为空if(!p|ji)return ERROR;/第第 i i 个元素不存在个元素不存在e=p-data;/取得第取得第 i i 个元素个元素return OK;ai-1 线性表的操作 ListInsert(&L,i,e)在单链表中的实现:有序对有序对 改变为改变为 和和 eaiai-1 因此,在单链表中第因此,在单链表中第 i 个结点之前进个结点之前进行插入的基本操作为行插入的基本操作为:找到线性表中第找到线性表中第i-1i-1个结点,然后修改个结点,然后修改其指向后继的指针。其指向后继的指针。可见,在链表中插入结点只需要修改可见,在链表中插入结点只需要修改指针
26、。但同时,若要在第指针。但同时,若要在第 i 个结点之前个结点之前插入元素,修改的是第插入元素,修改的是第 i-1 个结点的指个结点的指针。针。Status ListInsert_L(LinkList L,int i,ElemType e)/L 为带头结点的单链表的头指针,本算法为带头结点的单链表的头指针,本算法 /在链表中第在链表中第i 个结点之前插入新的元素个结点之前插入新的元素 e /LinstInsert_L算法的算法的时间复杂度时间复杂度为:O(ListLength(L)p=L;j=0;while(p&j next;+j;/寻找第寻找第 i-1 个结点个结点if(!p|j i-1)r
27、eturn ERROR;/i 大于表长或者小于大于表长或者小于1 s=(LinkList)malloc(sizeof(LNode);/生成新结点s-data=e;s-next=p-next;p-next=s;/插入return OK;eai-1aiai-1sp线性表的操作ListDelete(&L,i,&e)在链表中的实现:有序对有序对 和和 改变为改变为 ai-1aiai+1ai-1 在单链表中删除第删除第 i i 个结点个结点的基本基本操作操作为:找到线性表中第找到线性表中第i-1i-1个结点,修个结点,修改其指向后继的指针。改其指向后继的指针。ai-1aiai+1ai-1q=p-next
28、;p-next=q-next;e=q-data;free(q);pq Status ListDelete_L(LinkList L,int i,ElemType&e)/删除以 L 为头指针(带头结点)的单链表中第 i 个结点 /ListDelete_L算法的算法的时间复杂度时间复杂度为:O(ListLength(L)p=L;j=0;while(p-next&j next;+j;/寻找第 i 个结点,并令 p 指向其前趋if (!(p-next)|j i-1)return ERROR;/删除位置不合理q=p-next;p-next=q-next;/删除并释放结点e=q-data;free(q);
29、return OK;操作 ClearList(&L)在链表中的实现:void ClearList(&L)/将单链表重新置为一个空表 while(L-next)p=L-next;L-next=p-next;/ClearListfree(p);算法时间复杂度:O(ListLength(L)如何从线性表得到单链表?如何从线性表得到单链表?链表是一个动态的结构,它不需要链表是一个动态的结构,它不需要予分配空间,因此予分配空间,因此生成链表的过程生成链表的过程是一个结点是一个结点“逐个插入逐个插入”的过程。的过程。例如:逆位序输入例如:逆位序输入 n n 个数据元素的值,个数据元素的值,建立带头结点的单
30、链表。建立带头结点的单链表。操作步骤:操作步骤:一、建立一个一、建立一个“空表空表”;二、输入数据元素二、输入数据元素an,建立结点并插入;建立结点并插入;三、输入数据元素三、输入数据元素an-1,建立结点并插入;建立结点并插入;ananan-1四、依次类推,直至输入四、依次类推,直至输入a a1 1为止。为止。void CreateList_L(LinkList&L,int n)/逆序输入 n 个数据元素,建立带头结点的单链表/CreateList_L算法的算法的时间复杂度时间复杂度为:O(Listlength(L)L=(LinkList)malloc(sizeof(LNode);L-nex
31、t=NULL;/先建立一个带头结点的单链表for(i=n;i 0;-i)p=(LinkList)malloc(sizeof(LNode);scanf(&p-data);/输入元素值 p-next=L-next;L-next=p;/插入回顾回顾 2.1 节中三个例子的算法,节中三个例子的算法,看一下当线性表分别以顺序存储看一下当线性表分别以顺序存储结构和链表存储结构实现时,它结构和链表存储结构实现时,它们的时间复杂度为多少?们的时间复杂度为多少?void union(List&La,List Lb)La_len=ListLength(La);Lb_len=ListLength(Lb);for(i
32、=1;i=Lb_len;i+)GetElem(Lb,i,e);if(!LocateElem(La,e,equal()ListInsert(La,+La_len,e);/for/union控制结构:控制结构:基本操作:基本操作:for 循环循环GetElem,LocateElem 和 ListInsert当以顺序映像实现抽象数据类型线性表时为:O(ListLength(La)ListLength(Lb)当以链式映像实现抽象数据类型线性表时为:O(ListLength(La)ListLength(Lb)例例1算法时间复杂度算法时间复杂度void purge(List&La,List Lb)Init
33、List(LA);La_len=ListLength(La);Lb_len=ListLength(Lb);for(i=1;i=Lb_len;i+)GetElem(Lb,i,e);if(ListEmpty(La)|!equal(en,e)ListInsert(La,+La_len,e);en=e;/for/purge控制结构:控制结构:基本操作:基本操作:for 循环GetElem 和 ListInsert当以顺序映像实现抽象数据类型线性表时为:O(ListLength(Lb)当以链式映像实现抽象数据类型线性表时为:O(ListLength2(Lb)例例2算法时间复杂度算法时间复杂度void M
34、ergeList(List La,List Lb,List&Lc)InitList(Lc);i=j=1;k=0;La_len=ListLength(La);Lb_len=ListLength(Lb);while(i=La_len)&(j=Lb_len)GetElem(La,i,ai);GetElem(Lb,j,bj);if(ai next=L.current-next;L.current-next=s;if(L.tail=L.current)L.tail=s;L.current=s;return OK;Status DelAfter(LinkList&L,ElemType&e)/若当前指针及其
35、后继在链表中,则删除线性链表L中当前 /指针所指结点之后的结点,并返回OK;否则返回ERROR。/DelAfterif(!(L.current&L.current-next)return ERROR;q=L.current-next;L.current-next=q-next;if(L.tail=q)L.tail=L.current;e=q-data;FreeNode(q);return OK;1.双向链表双向链表五、其它形式的链表五、其它形式的链表typedef struct DuLNode ElemType data;/数据域 struct DuLNode *prior;/指向前驱的指针域
36、 struct DuLNode *next;/指向后继的指针域 DuLNode,*DuLinkList;双向链表双向链表(Doubly Linked List)双向链表结点结构双向链表结点结构:指向直接前驱指向直接前驱 指向直接指向直接后继后继非空表非空表 空表空表firstfirst 最后一个结点的指针域的指针又指回第一个结点的链表 a1 a2 .an 2.循环链表循环链表 和单链表的差别仅在于,判别链表中最后一个结点的条件不再是“后继是否为空”,而是“后继是否为头结点”。双向循环链表双向循环链表空表空表非空表非空表 a1 a2 .an双向链表的操作特点:双向链表的操作特点:“查询查询”和单
37、链表相同。和单链表相同。“插入插入”和和“删除删除”时需要同时修时需要同时修改两个方向上的指针。改两个方向上的指针。ai-1aies-next=p-next;p-next=s;s-next-prior=s;s-prior=p;psai-1ai插入插入ai-1删除删除aiai+1p-next=p-next-next;p-next-prior=p;pai-1六、顺序表与链表的比较六、顺序表与链表的比较基于时间的比较基于时间的比较n存取方式存取方式u顺序表可以随机存取,也可以顺序存取顺序表可以随机存取,也可以顺序存取u链表是顺序存取的链表是顺序存取的n插入插入/删除时移动元素个数删除时移动元素个数u
38、顺序表平均需要移动近一半元素顺序表平均需要移动近一半元素u链表不需要移动元素,只需要修改指针链表不需要移动元素,只需要修改指针基于空间的比较基于空间的比较n存储分配的方式存储分配的方式u顺序表的存储空间是静态分配的顺序表的存储空间是静态分配的u链表的存储空间是动态分配的链表的存储空间是动态分配的n存储密度存储密度 =结点数据本身所占的存储量结点数据本身所占的存储量/结点结构所占的存储总量结点结构所占的存储总量u顺序表的存储密度顺序表的存储密度 =1=1u链表的存储密度链表的存储密度 1 1顺序表与链表的比较顺序表与链表的比较在计算机中,可以用一个线性表来表示在计算机中,可以用一个线性表来表示:
39、P=(p0,p1,,pn)一元多项式一元多项式但是对于形如但是对于形如 S(x)=1+3x10000 2x20000的多项式,上述表示方法是否合适?的多项式,上述表示方法是否合适?一般情况下的一元稀疏多项式一元稀疏多项式可写成 Pn(x)=p1xe1+p2xe2+pmxem其中其中:pi 是指数为ei 的项的非零系数,0 e1 e2 em=n可以下列线性表表示:(p1,e1),(p2,e2),(pm,em))P999(x)=7x3-2x12-8x999例如例如:可用线性表 (7,3),(-2,12),(-8,999)表示ADT Polynomial 数据对象数据对象:数据关系数据关系:抽象数据
40、类型一元多项式的定义如下:D ai|ai TermSet,i=1,2,.,m,m0 TermSet 中的每个元素包含一个每个元素包含一个 表示系数的实数和表示指数的整数表示系数的实数和表示指数的整数 R1|ai-1,aiD,i=2,.,n 且ai-1中的指数值中的指数值ai中的指数值中的指数值 CreatPolyn(&P,m)DestroyPolyn(&P)PrintPolyn(&P)基本操作基本操作:操作结果操作结果:输入 m 项的系数和指数,建立一元多项式 P。初始条件初始条件:一元多项式 P 已存在。操作结果操作结果:销毁一元多项式 P。初始条件初始条件:一元多项式 P 已存在。操作结果
41、操作结果:打印输出一元多项式 P。PolynLength(P)AddPolyn(&Pa,&Pb)SubtractPolyn(&Pa,&Pb)ADT Polynomial初始条件初始条件:一元多项式 P 已存在。操作结果操作结果:返回一元多项式 P 中的项数。初始条件初始条件:一元多项式 Pa 和 Pb 已存在。操作结果操作结果:完成多项式相加运算,即:Pa=PaPb,并销毁一元多项式 Pb。一元多项式的实现:一元多项式的实现:typedef struct /项项的表示 float coef;/系数系数 int expn;/指数指数 term,ElemType;typedef OrderedLi
42、nkList polynomial;/用带表头结点的有序链表表示多项式用带表头结点的有序链表表示多项式结点的数据元素类型定义为:Status CreatPolyn(polynomail&P,int m)/输入输入m项的系数和指数,建立表示一元多项式的有序链表项的系数和指数,建立表示一元多项式的有序链表P/CreatPolynInitList(P);e.coef=0.0;e.expn=-1;SetCurElem(h,e);/设置头结点的数据元素for(i=1;i=m;+i)/依次输入 m 个非零项return OK;scanf(e.coef,e.expn);if(!LocateElem(P,e,
43、(*cmp)()if(!InsAfter(P,e)return ERROR;注意注意:1.输入次序不限输入次序不限;2.指数相同的项只能输入一次。指数相同的项只能输入一次。多项式相加多项式相加AH=1-10 x6+2x8+7x14BH=-x4+10 x6-3x10+8x14+4x18Status AddPolyn(polynomial&Pc,polynomial&Pa,polynomial&Pb)/利用两个多项式的结点构成“和多项式”Pc=PaPb if(DelAfter(Pa,e1)a=e1.expn else a=MAXE;if(DelAfter(Pb,e2)b=e2.expn else
44、b=MAXE;while(!(a=MAXE&b=MAXE)/AddPolynswitch(*cmp(e1,e2)case-1:/多项式多项式PA中当前结点的指数值中当前结点的指数值小小 break;case 0:/两者的指数值相等两者的指数值相等 e1.coef=a.coef+b.coef;if(a.coef!=0.0)InsAfter(Pc,e1);break;case 1:/多项式多项式PB中当前结点的指数值小中当前结点的指数值小 break;应用举例应用举例约瑟夫问题约瑟夫问题用循环链表求解约瑟夫问题用循环链表求解约瑟夫问题n 个人围成一个圆圈,首先第个人围成一个圆圈,首先第1个人从个人
45、从1开始一个人开始一个人一个人顺时针报数一个人顺时针报数,报到第报到第m个人,令其出列。然个人,令其出列。然后再从下一个人开始,从后再从下一个人开始,从1顺时针报数,报到第顺时针报数,报到第m个个人,再令其出列,人,再令其出列,如此下去,如此下去,直到圆圈中只剩直到圆圈中只剩一个人为止。此人即为优胜者。一个人为止。此人即为优胜者。例如例如 n=8 m=3约瑟夫问题的解法约瑟夫问题的解法#include#include“CircList.h”void Josephus(int n,int m)for(int i=0;in-1;i+)/执行执行n-1次次 for(int j=0;jm-1;j+)N
46、ext();cout “Delete person”getData()endl;Remove();/删去删去 void main()CircList clist;int n,m;cout n m;/形成约瑟夫环形成约瑟夫环 for(int i=1;i=n;i+)clist.insert(i);clist.Josephus(n,m);/解决约瑟夫问题解决约瑟夫问题本章小结本章小结1.1.了了解解线线性性表表的的逻逻辑辑结结构构特特性性是是数数据据元元素素之之间间存存在在着着线线性性关关系系,在在计计算算机机中中表表示示这这种种关关系系的的两两类类不不同同的的存存储储结结构构是是顺顺序序存存储储结
47、结构构和和链链式式存存储储结结构构。用用前前者者表表示示的的线线性性表表简简称称为为顺顺序序表表,用后者表示的线性表简称为链表。用后者表示的线性表简称为链表。2.2.熟练掌握这两类存储结构的描述方法,以及熟练掌握这两类存储结构的描述方法,以及线性表的各种基本操作的实现。线性表的各种基本操作的实现。3.3.能够从时间和空间复杂度的角度综合比较线性能够从时间和空间复杂度的角度综合比较线性表两种存储结构的不同特点及其适用场合。表两种存储结构的不同特点及其适用场合。习习 题题 1.1.简述线性表的两种存储结构顺序表和链表简述线性表的两种存储结构顺序表和链表的优缺点。的优缺点。2.2.已知已知 L L
48、是无表头结点的单链表,且是无表头结点的单链表,且 P P 是是指向表中某个结点的指针,试写出在指向表中某个结点的指针,试写出在 P P 所指所指结点之前插入指针结点之前插入指针 S S 所指结点的语句序列。所指结点的语句序列。3.3.已知已知 P P 是指向双向链表中某个结点的指针,是指向双向链表中某个结点的指针,试写出删除试写出删除 P P 所指结点的前驱结点的语句序所指结点的前驱结点的语句序列。列。4.4.设顺序表设顺序表 a a 中的数据元素递增有序。中的数据元素递增有序。试写一算法,将试写一算法,将 x x 插入到顺序表的适当插入到顺序表的适当位置上,以保持该表的有序性。位置上,以保持该表的有序性。void void InsertOrderListInsertOrderList(SqListSqList&a,&a,ElemTypeElemType x)x)5.5.试写一算法,对单链表实现就地逆置。试写一算法,对单链表实现就地逆置。习习 题题