收藏 分销(赏)

数据结构考研讲义.doc

上传人:丰**** 文档编号:4363549 上传时间:2024-09-13 格式:DOC 页数:74 大小:1.63MB
下载 相关 举报
数据结构考研讲义.doc_第1页
第1页 / 共74页
数据结构考研讲义.doc_第2页
第2页 / 共74页
数据结构考研讲义.doc_第3页
第3页 / 共74页
数据结构考研讲义.doc_第4页
第4页 / 共74页
数据结构考研讲义.doc_第5页
第5页 / 共74页
点击查看更多>>
资源描述

1、目录绪论30、1 基本概念3第一章 线性表41、1 线性表得定义41、2 线性表得实现41、2、2 线性表得链式存储结构6第二章 栈、队列与数组112、1 栈112、2 队列152、3 特殊矩阵得压缩存储172、3、1 数组172、3、2 特殊矩阵17第三章 树与二叉树203、1 树得概念201、树得定义202.相关术语203、2 二叉树213、2、1 定义与性质213、2、2 二叉树得存储223、2、3 二叉树得遍历233、2、4 线索二叉树253、3 树与森林293、3、1 树得存储结构293、3、2 森林与二叉树得转换303、3、3 树与森林得遍历303、4 哈夫曼( Huffman)树

2、与哈夫曼编码31第四章 图324、1 图得概念324、2 图得存储及基本操作334、2、1 邻接矩阵334、2、2 邻接表334、3 图得遍历354、3、1 深度优先搜索354、3、2 广度优先搜索354、4 图得基本应用374、4、1 最小生成树374、4、2 最短路径374、4、3 拓扑排序394、4、4 关键路径40第五章 查找425、1 查找得基本概念425、2 顺序查找法435、3 折半查找法445、4 动态查找树表455、4、1 二叉排序树455、4、2 平衡二叉树475、4、3 B 树及其基本操作、 B+树得基本概念495、5 散列表515、5、2 常用得散列函数515、5、3

3、处理冲突得方法525、5、4 散列表得查找525、5、5 散列表得查找分析53第六章 排序546、1 插入排序546、1、1 直接插入排序546、1、2 折半插入排序546、2 冒泡排序556、3 简单选择排序566、4 希尔排序576、5 快速排序586、6 堆排序606、7 二路归并排序626、8 基数排序636、9 各种内部排序算法得比较64绪论0、1 基本概念1、数据结构数据结构就是指互相之间存在着一种或多种关系得数据元素得集合。数据结构就是一个二元组 Data_Structure ( D, R),其中, D 就是数据元素得有限集, R 就是D 上关系得有限集。2、逻辑结构:就是指数据

4、之间得相互关系。通常分为四类结构:( 1)集合:结构中得数据元素除了同属于一种类型外,别无其它关系。( 2)线性结构:结构中得数据元素之间存在一对一得关系。( 3)树型结构:结构中得数据元素之间存在一对多得关系。( 4)图状结构:结构中得数据元素之间存在多对多得关系。3、存储结构:就是指数据结构在计算机中得表示,又称为数据得物理结构。通常由四种基本得存储方法实现:( 1)顺序存储方式。数据元素顺序存放,每个存储结点只含一个元素。存储位置反映数据元素间得逻辑关系。存储密度大。但有些操作(如插入、删除)效率较差。( 2)链式存储方式。每个存储结点除包含数据元素信息外还包含一组(至少一个)指针。指针

5、反映数据元素间得逻辑关系。这种方式不要求存储空间连续,便于动态操作(如插入、删除等),但存储空间开销大(用于指针),另外不能折半查找等。( 3)索引存储方式。除数据元素存储在一组地址连续得内存空间外,还需建立一个索引表,索引表中索引指示存储结点得存储位置(下标)或存储区间端点(下标)。( 4)散列存储方式。通过散列函数与解决冲突得方法,将关键字散列在连续得有限得地址空间内,并将散列函数得值解释成关键字所在元素得存储地址。其特点就是存取速度快,只能按关键字随机存取,不能顺序存取,也不能折半存取。2 算法与算法得衡量1、算法就是对特定问题求解步骤得一种描述,就是指令得有限序列。其中每一条指令表示一

6、个或多个操作。算法具有下列特性:有穷性确定性可行性输入输出。算法与程序十分相似,但又有区别。程序不一定具有有穷性,程序中得指令必须就是机器可执行得,而算法中得指令则无此限制。算法代表了对问题得解,而程序则就是算法在计算机上得特定得实现。一个算法若用程序设计语言来描述,则它就就是一个程序。2、算法得时间复杂度:以基本运算得原操作重复执行得次数作为算法得时间度量。一般情况下,算法中基本运算次数 T(n)就是问题规模 n(输入量得多少,称之为问题规模)得某个函数f(n),记作:T(n) (f(n);也可表示 T(n) m(f(n),其中 m 为常量。记号“O”读作“大 O”,它表示随问题规模 n 得

7、增大,算法执行时间 T(n)得增长率与 f(n)得增长率相同。注意:有得情况下,算法中基本操作重复执行得次数还随问题得输入数据集不同而不同。常见得渐进时间复杂度有: (1)(log2n)(n)(nlog2n)(n2)(n3)(2n) O(n!) O(nn)。3、算法得空间复杂度:就是对一个算法在运行过程中临时占用得存储空间大小得量度。只需要分析除输入与程序之外得辅助变量所占额外空间。原地工作:若所需额外空间相对于输入数据量来说就是常数,则称此算法为原地工作,空间复杂度为 O(1)。第一章 线性表1、1 线性表得定义线性表就是一种线性结构,在一个线性表中数据元素得类型就是相同得,或者说线性表就是

8、由同一类型得数据元素构成得线性结构,定义如下:线性表就是具有相同数据类型得n(n0)个数据元素得有限序列,通常记为:(a1, a2, ai-1, ai, ai+1, an)其中n为表长, n0 时称为空表。需要说明得就是: ai为序号为 i 得数据元素( i=1,2,n),通常将它得数据类型抽象为ElemType, ElemType根据具体问题而定。1、2 线性表得实现1、2、1 线性表得顺序存储结构1、顺序表线性表得顺序存储就是指在内存中用地址连续得一块存储空间顺序存放线性表得各元素,用这种存储形式存储得线性表称其为顺序表。因为内存中得地址空间就是线性得,因此,用物理上得相邻实现数据元素之间

9、得逻辑相邻关系就是既简单又自然得。设a得存储地址为Loc(a),每个数据元素占d个存储地址,则第i个数据元素得地址为:Loc(ai)=Loc(a)+(i-1)*d 1in这就就是说只要知道顺序表首地址与每个数据元素所占地址单元得个数就可求出第i个数据元素得地址来,这也就是顺序表具有按数据元素得序号随机存取得特点。线性表得动态分配顺序存储结构:#define LIST_INIT_SIZE 100 /存储空间得初始分配量#define LISTINCREMENT 10 /存储空间得分配增量typedef structElemType *elem; /线性表得存储空间基址int length; /当

10、前长度int listsize; /当前已分配得存储空间SqList;2、顺序表上基本运算得实现( 1)顺序表得初始化顺序表得初始化即构造一个空表, 这对表就是一个加工型得运算, 因此, 将L设为引用参数,首先动态分配存储空间,然后,将length置为0,表示表中没有数据元素。int Init_SqList (SqList &L)L、elem = (ElemType * )malloc(LIST_INIT_SIZE * sizeof(ElemType);if (!L、elem) exit (OVERFLOW); /存储分配失败L、length=0;L、 listsize = LIST_INIT

11、_SIZE; /初始存储容量return OK;( 2)插入运算线性表得插入就是指在表得第i(i得取值范围: 1in+1)个位置上插入一个值为 x 得新元素,插入后使原表长为 n得表:(a1, a2, 、 , ai-1, ai, ai+1, 、 , an)成为表长为 n+1 表:(a1, a2, 、, ai-1, x, ai, ai+1, 、, an ) 。顺序表上完成这一运算则通过以下步骤进行: 将aian 顺序向下移动,为新元素让出位置;(注意数据得移动方向:从后往前依次后移一个元素) 将 x 置入空出得第i个位置; 修改表长。int Insert_SqList (SqList &L, i

12、nt i, ElemType x)if (i L、length+1) return ERROR; / 插入位置不合法if (L、length = L、listsize) return OVERFLOW; / 当前存储空间已满,不能插入/需注意得就是,若就是采用动态分配得顺序表,当存储空间已满时也可增加分配q = &(L、elemi-1); / q 指示插入位置for (p = &(L、elemL、length-1); p = q; -p)*(p+1) = *p; / 插入位置及之后得元素右移*q = e; / 插入e+L、length; / 表长增1return OK;顺序表上得插入运算, 时

13、间主要消耗在了数据得移动上, 在第i个位置上插入 x , 从 ai 到an 都要向下移动一个位置,共需要移动 ni1个元素。( 3)删除运算线性表得删除运算就是指将表中第 i ( i 得取值范围为 : 1 in)个元素从线性表中去掉,删除后使原表长为 n 得线性表:(a1, a2, 、 , ai-1, ai, ai+1, 、, an)成为表长为 n1 得线性表:(a1, a2, 、 , ai-1, ai+1, 、 , an)。顺序表上完成这一运算得步骤如下: 将ai+1an 顺序向上移动;(注意数据得移动方向:从前往后依次前移一个元素) 修改表长。int Delete_SqList (SqLi

14、st &L;int i) if (i L、length) return ERROR; / 删除位置不合法p = &(L、elemi-1); / p 为被删除元素得位置e = *p; / 被删除元素得值赋给 eq = L、elem+L、length-1; / 表尾元素得位置for (+p; p data=x;s-next=L; L=s;scanf (%d,&x);return L;尾插法在单链表得尾部插入结点建立单链表头插入建立单链表简单, 但读入得数据元素得顺序与生成得链表中元素得顺序就是相反得,若希望次序一致,则用尾插入得方法。因为每次就是将新结点插入到链表得尾部,所以需加入一个指针 r 用

15、来始终指向链表中得尾结点,以便能够将新结点插入到链表得尾部。初始状态, 头指针L=NULL, 尾指针 r=NULL; 按线性表中元素得顺序依次读入数据元素,不就是结束标志时,申请结点,将新结点插入到 r 所指结点得后面,然后 r 指向新结点(注意第一个结点有所不同)。LinkList CreateListR1 ( )LinkList L=NULL;LNode *s,*r=NULL;int x; /设数据元素得类型为intscanf(%d,&x);while (x!=flag) s=(LNode *)malloc(sizeof(LNode);s-data=x;if (L=NULL) L=s; /

16、第一个结点得处理else r-next=s; /其它结点得处理r=s; /r 指向新得尾结点scanf(%d,&x);if ( r!=NULL) r-next=NULL; /对于非空表,最后结点得指针域放空指针return L;在算法CreateListR1中,第一个结点得处理与其它结点就是不同得,原因就是第一个结点加入时链表为空,它没有直接前驱结点,它得地址就就是整个链表得指针,需要放在链表得头指针变量中;而其它结点有直接前驱结点,其地址放入直接前驱结点得指针域。 “第一个结点”得问题在很多操作中都会遇到,如在链表中插入结点时,将结点插在第一个位置与其它位置就是不同得,在链表中删除结点时,删

17、除第一个结点与其它结点得处理也就是不同得,等等。为了方便操作,有时在链表得头部加入一个“头结点”,头结点得类型与数据结点一致,标识链表得头指针变量L中存放该结点得地址,这样即使就是空表,头指针变量L也不为空了。头结点得加入使得“第一个结点”得问题不再存在,也使得“空表”与“非空表”得处理成为一致。头结点得加入完全就是为了运算得方便,它得数据域无定义,指针域中存放得就是第一个数据结点得地址,空表时为空。尾插法建立带头结点得单链表,将算法CreateListR1改写成算法CreateListR2形式。LinkList CreateListR2( )LinkList L=(LNode *)mallo

18、c(sizeof(LNode);L-next=NULL; /空表LNode *s,*r=L;int x; /设数据元素得类型为intscanf(%d,&x);while (x!=flag) s=(LNode *)malloc(sizeof(LNode);s-data=x;r-next=s;r=s; /r 指向新得尾结点scanf(%d,&x);r-next=NULL;return L;因此,头结点得加入会带来以下两个优点:第一个优点:由于开始结点得位置被存放在头结点得指针域中,所以在链表得第一个位置上得操作就与在表得其它位置上得操作一致,无需进行特殊处理;第二个优点:无论链表就是否为空,其头指

19、针就是指向头结点在得非空指针(空表中头结点得指针域为空),因此空表与非空表得处理也就统一了。在以后得算法中不加说明则认为单链表就是带头结点得。(2)查找操作按序号查找 Get_LinkList(L,i)从链表得第一个元素结点起,判断当前结点就是否就是第i个,若就是,则返回该结点得指针,否则继续后一个,表结束为止,没有第个结点时返回空。LNode * Get_LinkList(LinkList L, int i); LNode * p=L;int j=0;while (p-next !=NULL & jnext; j+;if (j=i) return p;else return NULL;(3)

20、插入运算后插结点:设p指向单链表中某结点, s指向待插入得值为x得新结点,将*s插入到*p得后面,插入示意图如图所示。操作如下:s-next=p-next;p-next=s;注意:两个指针得操作顺序不能交换。(4)删除运算删除结点设p指向单链表中某结点,删除*p。操作过程如图。要实现对结点*p得删除,首先要找到*p得前驱结点*q,然后完成指针得操作即可。操作如下:q=L;while (q-next!=p)q=q-next; /找*p得直接前驱q-next=p-next;free(p);因为找*p前驱得时间复杂度为O(n),所以该操作得时间复杂度为O(n)通过上面得基本操作我们得知:(1) 单链

21、表上插入、删除一个结点,必须知道其前驱结点。(2) 单链表不具有按序号随机访问得特点,只能从头指针开始一个个顺序进行。1、2、2、2 循环链表对于单链表而言,最后一个结点得指针域就是空指针,如果将该链表头指针置入该指针域,则使得链表头尾结点相连,就构成了单循环链表。在单循环链表上得操作基本上与非循环链表相同,只就是将原来判断指针就是否为 NULL 变为就是否就是头指针而已,没有其它较大得变化。对于单链表只能从头结点开始遍历整个链表,而对于单循环链表则可以从表中任意结点开始遍历整个链表,不仅如此,有时对链表常做得操作就是在表尾、表头进行,此时可以改变一下链表得标识方法,不用头指针而用一个指向尾结

22、点得指针 R 来标识,可以使得操作效率得以提高。1、2、2、3 双向链表单链表得结点中只有一个指向其后继结点得指针域 next,因此若已知某结点得指针为 p,其后继结点得指针则为 p-next ,而找其前驱则只能从该链表得头指针开始,顺着各结点得next 域进行,也就就是说找后继得时间性能就是 O(1),找前驱得时间性能就是 O(n),如果也希望找前驱得时间性能达到 O(1),则只能付出空间得代价:每个结点再加一个指向前驱得指针域,结点得结构为如图所示,用这种结点组成得链表称为双向链表。线性表得双向链表存储结构C语言描述下:typedef struct DuLNodeElemType data

23、;struct DuLNode *prior,*next;DuLNode,*DuLinkList;与单链表类似,双向链表通常也就是用头指针标识,也可以带头结点。( 1)双向链表中结点得插入:设 p 指向双向链表中某结点, s 指向待插入得值为 x 得新结点,将*s 插入到*p 得前面,插入示意图如所示。操作如下: s-prior=p-prior; p-prior-next=s; s-next=p; p-prior=s;指针操作得顺序不就是唯一得,但也不就是任意得,操作必须要放到操作得前面完成,否则*p 得前驱结点得指针就丢掉了。( 2)双向链表中结点得删除:设 p 指向双向链表中某结点,删除*

24、p。操作示意图如图所示。操作如下:p-prior-next=p-next;p-next-prior=p-prior;free(p);1、2、2、4 顺序表与链表得比较总之,两种存储结构各有长短,选择那一种由实际问题中得主要因素决定。通常“较稳定”得线性表选择顺序存储,而频繁做插入删除得即动态性较强得线性表宜选择链式存储。第二章 栈、队列与数组2、1 栈2、1、1 栈得定义栈就是限制在表得一端进行插入与删除得线性表。允许插入、删除得这一端称为栈顶,另一个固定端称为栈底。当表中没有元素时称为空栈。2、1、2 栈得存储实现与运算实现栈就是运算受限得线性表,线性表得存储结构对栈也就是适用得,只就是操作

25、不同而已。利用顺序存储方式实现得栈称为顺序栈。与线性表类似,栈得动态分配顺序存储结构如下:#define STACK_INIT_SIZE 100 /存储空间得初始分配量#define STACKINCREMENT 10 /存储空间得分配增量typedef structSElemType *base; /在栈构造之前与销毁之后, base 得值为 NULLSElemType *top; /栈顶指针int stacksize; /当前已分配得存储空间SqStack;需要注意,在栈得动态分配顺序存储结构中, base 始终指向栈底元素,非空栈中得 top始终在栈顶元素得下一个位置。下面就是顺序栈上常

26、用得基本操作得实现。( 1)入栈:若栈不满,则将 e 插入栈顶。int Push (SqStack &S, SElemType e) if (S、top-S、base=S、stacksize) /栈满,追加存储空间*S、top+ = e; /top 始终在栈顶元素得下一个位置return OK;( 2)出栈:若栈不空,则删除 S 得栈顶元素,用 e 返回其值,并返回 OK,否则返回ERROR。int Pop (SqStack &S, SElemType &e) if (S、top=S、base) return ERROR;e = *-S、top;return OK;出栈与读栈顶元素操作,先判栈

27、就是否为空,为空时不能操作,否则产生错误。通常栈空常作为一种控制转移得条件。2、1、3 栈得应用举例由于栈得“先进先出”特点,在很多实际问题中都利用栈做一个辅助得数据结构来进行求解,下面通过几个例子进行说明。1.数制转换十进制数 N 与其她 d 进制数得转换就是计算机实现计算得基本问题,其解决方法很多,其中一个简单算法基于下列原理:假设现要编制一个满足下列要求得程序:对于输入得任意一个非负十进制整数,打印输出与其等值得八进制数。由于上述计算过程就是从低位到高位顺序产生八进制数得各个数位,而打印输出,一般来说应从高位到低位进行,恰好与计算过程相反。因此,若将计算过程中得到得八进制数得各位顺序进栈

28、,则按出栈序列打印输出得即为与输入对应得八进制数。算法思想:当 N0 时重复( 1),( 2)( 1)若 N0,则将 N % r 压入栈 s 中 ,执行( 2) ;若 N=0,将栈 s 得内容依次出栈,算法结束。( 2)用 N / r 代替 N。void conversion () InitStack(S); / 构造空栈scanf (%d,N);while (N) Push(S, N % 8);N = N/8;while (!StackEmpty(S) Pop(S,e);printf ( %d, e );2、表达式求值表达式求值就是程序设计语言编译中一个最基本得问题,它得实现也就是需要栈得加

29、入。下面得算法就是由运算符优先法对表达式求值。在此仅限于讨论只含二目运算符得算术表达式。( 1)中缀表达式求值:中缀表达式:每个二目运算符在两个运算量得中间,假设所讨论得算术运算符包括: + 、- 、 *、 /、 %、 (乘方)与括号()。设运算规则为:.运算符得优先级为:() 、 /、 % +、 - ;.有括号出现时先算括号内得,后算括号外得,多层括号,由内向外进行;.乘方连续出现时先算最右面得。表达式作为一个满足表达式语法规则得串存储,如表达式“3*2( 4+2*2-*3) -5”,它得得求值过程为:自左向右扫描表达式,当扫描到 3*2 时不能马上计算,因为后面可能还有更高得运算,正确得处

30、理过程就是:需要两个栈:对象栈 s1 与运算符栈 s2。当自左至右扫描表达式得每一个字符时,若当前字符就是运算对象,入对象栈,就是运算符时,若这个运算符比栈顶运算符高则入栈,继续向后处理,若这个运算符比栈顶运算符低则从对象栈出栈两个运算量,从运算符栈出栈一个运算符进行运算,并将其运算结果入对象栈,继续处理当前字符,直到遇到结束符。中缀表达式表达式 “3*2( 4+2*2-*3) -5”求值过程中两个栈得状态情况见图所示。 图 中缀表达式 3*2( 4+2*2-1*3) -5 得求值过程为了处理方便,编译程序常把中缀表达式首先转换成等价得后缀表达式,后缀表达式得运算符在运算对象之后。在后缀表达式

31、中,不在引入括号,所有得计算按运算符出现得顺序,严格从左向右进行,而不用再考虑运算规则与级别。中缀表达式“3*2( 4+2*2-1*3) -5 ”得后缀表达式为: “32422*+13*-*5-”。( 2)后缀表达式求值计算一个后缀表达式,算法上比计算一个中缀表达式简单得多。这就是因为表达式中即无括号又无优先级得约束。具体做法:只使用一个对象栈,当从左向右扫描表达式时,每遇到一个操作数就送入栈中保存,每遇到一个运算符就从栈中取出两个操作数进行当前得计算,然后把结果再入栈,直到整个表达式结束,这时送入栈顶得值就就是结果。( 3) 中缀表达式转换成后缀表达式:将中缀表达式转化为后缀表达示与前述对中

32、缀表达式求值得方法完全类似,但只需要运算符栈,遇到运算对象时直接放后缀表达式得存储区,假设中缀表达式本身合法且在字符数组 A 中,转换后得后缀表达式存储在字符数组 B 中。具体做法:遇到运算对象顺序向存储后缀表达式得 B 数组中存放,遇到运算符时类似于中缀表达式求值时对运算符得处理过程,但运算符出栈后不就是进行相应得运算,而就是将其送入 B 中存放。具体算法在此不再赘述。3.栈与递归在高级语言编制得程序中,调用函数与被调用函数之间得链接与信息交换必须通过栈进行。当在一个函数得运行期间调用另一个函数时,在运行该被调用函数之前,需先完成三件事:( 1)将所有得实在参数、返回地址等信息传递给被调用函

33、数保存;( 2)为被调用函数得局部变量分配存储区;( 3)将控制转移到被调用函数得入口。从被调用函数返回调用函数之前,应该完成:( 1)保存被调函数得计算结果;( 2)释放被调函数得数据区;( 3)依照被调函数保存得返回地址将控制转移到调用函数。多个函数嵌套调用得规则就是:后调用先返回,此时得内存管理实行“栈式管理”。递归函数得调用类似于多层函数得嵌套调用,只就是调用单位与被调用单位就是同一个函数而已。在每次调用时系统将属于各个递归层次得信息组成一个活动记录( ActivaTion Record),这个记录中包含着本层调用得实参、返回地址、局部变量等信息,并将这个活动记录保存在系统得“递归工作

34、栈”中,每当递归调用一次,就要在栈顶为过程建立一个新得活动记录,一旦本次调用结束,则将栈顶活动记录出栈,根据获得得返回地址信息返回到本次得调用处。将递归程序转化为非递归程序时常使用栈来实现。2、2 队列2、2、1 队列得定义及基本运算栈就是一种后进先出得数据结构,在实际问题中还经常使用一种“先进先出”得数据结构:即插入在表一端进行,而删除在表得另一端进行,将这种数据结构称为队或队列,把允许插入得一端叫队尾(rear) ;把允许删除得一端叫队头(front)。2、2、2 队列得存储实现及运算实现与线性表、栈类似,队列也有顺序存储与链式存储两种存储方法。1.顺序队列循环队列得类型定义如下:#def

35、ine MAXQSIZE 100 /最大队列长度typedef struct QElemType *base; /动态分配存储空间int front; /头指针,若队列不空,指向队列头元素int rear; /尾指针,若队列不空,指向队列尾元素得下一个位置 SqQueue;下面就是循环队列上基本操作得实现。( 3)求循环队列元素个数:int QueueLength( SqQueue Q) return (Q、rear-Q、front+MAXQSIZE) %MAXQSIZE;2.链队列链式存储得队称为链队列。与链栈类似,用单链表来实现链队列,根据队得先进先出原则,为了操作上得方便,分别需要一个头

36、指针与尾指针。定义一个指向链队列得指针: LinkQueue Q;下面就是链队列得基本运算得实现。( 1)入队int EnQueue (LinkQueue &Q, QElemType e) QNode *p;p = ( QNode *) malloc( sizeof( QNode) ;p-data = e;p-next = NULL;Q、rear-next = p;Q、rear = p;return OK;( 2)出队int DeQueue (LinkQueue &Q, QElemType &e) if (Q、front = Q、rear) return ERROR; /队空,出队失败p =

37、Q、front-next;e = p-data; /队头元素放 e 中Q、front-next = p-next;if(Q、rear=p) Q、rear= Q、front; /只有一个元素时,此时还要修改队尾指针free (p);return OK;3.除了栈与队列之外,还有一种限定性数据结构就是双端队列。( 1)双端队列:可以在双端进行插入与删除操作得线性表。( 2)输入受限得双端队列:线性表得两端都可以输出数据元素,但就是只能在一端输入数据元素。( 3)输出受限得双端队列:线性表得两端都可以输入数据元素,但就是只能在一端输出数据元素。2、3 特殊矩阵得压缩存储2、3、1 数组数组可以瞧作线

38、性表得推广。数组作为一种数据结构其特点就是结构中得元素本身可以就是具有某种结构得数据,但属于同一数据类型,数组就是一个具有固定格式与数量得数据有序集,每一个数据元素有唯一得一组下标来标识,因此,在数组上不能做插入、删除数据元素得操作。通常在各种高级语言中数组一旦被定义,每一维得大小及上下界都不能改变。现在来讨论数组在计算机中得存储表示。通常,数组在内存被映象为向量,即用向量作为数组得一种存储结构,这就是因为内存得地址空间就是一维得,数组得行列固定后,通过一个映象函数,则可根据数组元素得下标得到它得存储地址。对于一维数组按下标顺序分配即可。对多维数组分配时,要把它得元素映象存储在一维存储器中,一

39、般有两种存储方式:一就是以行为主序(或先行后列)得顺序存放,即一行分配完了接着分配下一行。另一种就是以列为主序(先列后行)得顺序存放,即一列一列地分配。以行为主序得分配规律就是:最右边得下标先变化,即最右下标从小到大,循环一遍后,右边第二个下标再变, ,从右向左,最后就是左下标。以列为主序分配得规律恰好相反:最左边得下标先变化,即最左下标从小到大,循环一遍后,左边第二个下标再变, ,从左向右,最后就是右下标。设有 mn 二维数组 Amn,下面我们瞧按元素得下标求其地址得计算:以“以行为主序”得分配为例:设数组得基址为 LOC(a11),每个数组元素占据 d 个地址单元,那么 aij 得物理地址

40、可用一线性寻址函数计算:LOC(aij) = LOC(a11) + ( (i-1)*n + j-1 ) * d这就是因为数组元素 aij 得前面有 i-1 行,每一行得元素个数为 n,在第 i 行中它得前面还有j-1 个数组元素。在 C 语言中,数组中每一维得下界定义为 0,则:LOC(aij) = LOC(a00) + ( i*n + j ) * d推广到一般得二维数组: Ac1、d1 c2、d2,则 aij 得物理地址计算函数为:LOC(aij)=LOC(a c1 c2)+( (i- c1) *( d2 - c2 + 1)+ (j- c2) )d2、3、2 特殊矩阵对于一个矩阵结构显然用一个二维数组来表示就是非常恰当得, 矩阵在这种存储表示之下,可以对其元素进行随机存取,各种矩阵运算也非常简单,并且存储得密度为 1。但就是在矩阵中非零元素呈某种规律分布或者矩阵中出现大量得零元素得情况下,比如常见得一些特殊矩阵,如三角矩阵、对称矩阵、对角矩阵、稀疏矩阵等,从节约存储空间得角度考虑,这种存储就是不太合适得,瞧起来存储密度仍为 1,但实际上占用了许多单元去存储重复得非零元素或零元素,这对高阶矩阵会造成极大得浪费,为了节省存储空间,我们可以对这类矩阵进行压缩存储:即为多个相同得非零元素只分配一个存储空间;对零元素不分配空间。1、对称矩阵对称矩阵得特点就是:在一个 n 阶方

展开阅读全文
相似文档                                   自信AI助手自信AI助手
猜你喜欢                                   自信AI导航自信AI导航
搜索标签

当前位置:首页 > 考试专区 > 研究生考试

移动网页_全站_页脚广告1

关于我们      便捷服务       自信AI       AI导航        获赠5币

©2010-2024 宁波自信网络信息技术有限公司  版权所有

客服电话:4008-655-100  投诉/维权电话:4009-655-100

gongan.png浙公网安备33021202000488号   

icp.png浙ICP备2021020529号-1  |  浙B2-20240490  

关注我们 :gzh.png    weibo.png    LOFTER.png 

客服