【数据结构】线性表(一):顺序列表

  线性表(linear_list)是最常用且最简单的一种数据结构,简言之,一个线性表是n个数据元素的有序序列。

 

例如:(a1 , ... , ai-1 , ai , ai+1 , ... , an):ai-1 是 ai 的直接前驱,ai+1 是 ai 的直接后驱。

并且,当 i = 1,2,... ,n-1:ai 有且只有一个直接后驱。当 i = 2,3,... ,n:ai 有且只有一个直接前驱

  

例子:已知线性表LA,LB中的数据元素按值非递减有序排列,现将LA,LB按值非递减排列合并为LC。其伪代码为:

 1 void MergeList(List La, List Lb, List &Lc) {
 2         initList(Lc);  //初始化Lc
 3         i = j = 1;
 4         k = 0;
 5         La_len = ListLength(La);
 6         Lb_len = ListLength(Lb);
 7         while((i <= La_len) && (j <= Lb_len){//La与Lb均非空
 8             GetElem(La, i, ai);
 9             GetElem(Lb, j, bj);
10             if(ai <= bj) {ListInsert(Lc, ++k, ai); ++i; }
11             else{ListInsert(Lc, ++k, bi); ++j;}
12         }
13         while(i <= La_len){       //La中剩余值插入Lc
14             GetElem(La, i++, ai);
15             ListInsert(Lc, ++k, ai);
16         }
17         while(j <= Lb_len){       //Lb中剩余值插入Lc
18             GetElem(Lb, j++, bj);
19             ListInsert(Lc, ++k, bj);
20         }
21 }

  分析:GetElem 和 ListInsert 这两个操作的执行时间和表长无关,所以该算法的事件复杂度为O(ListLength(LA) + ListLength(LB))。

 

线性表的顺序表示和实现:顺序表示指的是用一组地址连续的存储单元依次存储线性表的数据元素。

所以可以用 LOC(ai) = LOC(a1) + (i-1) * l  来表示ai的地址(其中:l 表示每个元素所占存储单元)

优点:轻松实现随机存储。

顺序表的初始化、插入、删除、合并的伪代码如下:

 

 1 // -------------线性表的动态分配顺序存储结构--------------
 2 #define LIST_INIT_SIZE 100   //线性表存储空间的初始化分配量
 3 #define LISTINCREMENT 10     //线性表存储空间的分配增量
 4 typedef struct{
 5     ElemType *elem;     //存储空间基址
 6     int length;         //当前长度
 7     int listsize;       //当前分配的存储容量(以sizeof(ElemType)为单位)
 8 }SqList;
 9 
10 //-----------初始化顺序表--------------
11 Status InitList_Sq(SqList &L){
12     //构造一个空的线性表
13     L.elem = (ElemType *)malloc(LIST_INIT_SIZE * sizeof(ElemType));
14     if(! L.elem)  eixt(OVERFLOW);       //存储分配失败
15     L.length = 0;                       //空表长度为0
16     L.listsize = LIST_INIT_SIZE;        //初始存储容量
17     return OK;
18 }
19 
20 //------------顺序表的插入算法-------------时间复杂度为O(n)
21 Status ListInsert_Sq(SqList &L, int i, ElemType e) {
22     //在顺序线性表L中第i个位置之前插入新的元素e
23     //i的合法值为1<=i<=ListLength_Sq(L)+1
24     if((i<1) || (i>L.length+1))  return ERROR;        //i值不合法
25     if(L.length >= L.listsize){           //当前存储空间已满,增加分配
26         newbase = (ElemType *)realloc(L.elem, (L.listsize + LISTINCREMENT)*sizeof(ElemTpye));
27         if(!newbase) exit(OVERFLOW);        //存储分配失败
28         L.elem = newbase;               //新基址
29         L.listsize += LISTINCREMENT;        //增加存储容量
30     }
31     q = &(L.elem[i-1]);         //q为插入位置
32     for(p = &(L.elem[L.length-1]); p >= q; --p) 
33         *(p+1) = *p;            //插入位置及之后的元素右移
34         *q = e;             //插入e
35         ++L.length;         //表长增一
36         return OK;
37 }
38 
39 //-------------顺序表的删除操作--------------时间复杂度为O(n)
40 Status ListDelete_Sq(SqList &L, int i, ElemType &e){
41     //在顺序线性表L中删除第i个元素,并用e返回其值。
42     if((i<1) || (i>L.length)) return ERROR;     //i值不合法
43     p = &(L.elem[i-1]);             //p为被删除元素的位置
44     e = *p;         //被删除元素的值赋给e
45     q = L.elem + L.length-1;        //表尾元素的位置
46     for(++p; p <= q; ++p){
47         *(p-1) = *p;    //被删除元素之后的元素左移
48     }
49     --L.length;         //表长减1
50     return OK;
51 }
52 
53 //-----------查找制定元素的位序---------------事件复杂度为O(n)
54 int LocateElem_Sq(SqList L, ElemType e, Status (* compare)(ElemType, ElemType)){
55     //在顺序线性表L中查找第1个值与e满足compare()的元素的位序
56     //若找到,则返回其在L中的位序,否则返回0
57     i = 1;              //i的初值为第一个元素的位序
58     p = L.elem;         //p的初值为第一个元素的存储位置
59     while(i <= L.length && !(* compare)(*p++, e)) ++i;
60     if(i<L.length) return i;
61     else return 0;
62 }
63 
64 //------------对于顺序表的合并------------------时间复杂度为O(La.length + Lb.length)
65 void MergeList_Sq(SqList La, SqList Lb, SqList &Lc) {
66     pa = La.elem;
67     pb = Lb.elem;
68     Lc.listsize = Lc.length = La.length + Lb.length;
69     pc = Lc.elem = (ElemType *) malloc (Lc.listsize*sizeof(ElemType));
70     if(!Lc.elem) exit(OVERFLOW);        //存储分配失败
71     pa_last = La.elem + La.length -1;
72     pb_last = Lb.elem + Lb.length -1;
73     while(pa <= pa_last && pb <= pb_last){  //归并
74         if(*pa <= *pb) *pc++ = *pa++;
75         else *pc++ = *pb++;
76     }
77     while(pa <= pa_last) *pc++ = *pa++;
78     while(pb <= pb_last) *pc++ = *pa++;
79

 

  C语言代码实现:

  1 #include <stdio.h>
  2 #include <stdlib.h>
  3 #include <string.h>
  4 
  5 #define LIST_INIT_SIZE 100      //线性表存储空间初始分配量
  6 #define LISTINCREMENT 10        //线性表存储空间的分配增量
  7 
  8 typedef struct {
  9     int *elem;      //存储空间基址
 10     int length;     //当前长度
 11     int listsize;   //当前分配的存储空间容量(以sizeof(int)为单位
 12 }SqList;
 13 
 14 //初始化一个新表
 15 void InitList_Sq(SqList *L){
 16     //构造一个空的线性表L
 17     (*L).elem = (int *) malloc(LIST_INIT_SIZE*sizeof(int));
 18     if(!(*L).elem) exit(-1);     //存储分配失败
 19     (*L).length = 0;
 20     (*L).listsize = LIST_INIT_SIZE;
 21 }
 22 
 23 //向表中第i个位置出插入一个数e
 24 void ListInsert_Sq(SqList *L, int i, int e){
 25     if(i<1 || i>(*L).length+1) {
 26         printf("插入时i值不合法...\n");
 27         exit(-1);
 28     }
 29     if((*L).length > (*L).listsize) {     //当前存储空间已满,增加分配
 30         int *newbase = (int *)realloc((*L).elem, ((*L).length+LISTINCREMENT)*sizeof(int));
 31         if(!newbase) exit(-1);
 32         (*L).elem = newbase;
 33         (*L).listsize += LISTINCREMENT;
 34     }
 35     int *q = &(*L).elem[i-1];
 36     for(int *p=&(*L).elem[(*L).length -1];p>=q;--p){
 37         *(p+1) = *p;
 38     }
 39     *q = e;
 40     ++(*L).length;
 41 }
 42 
 43 //删除表中第i个位置的数并返回到e中
 44 void ListDelete_Sq(SqList *L, int i, int *e){
 45     if(i<1 || i>(*L).length){
 46         printf("删除时i值不合法...\n");
 47         exit(-1);
 48     }
 49     int *p = &(*L).elem[i-1];       //要删除的元素
 50     *e = *p;                        //被删除的元素赋值给e
 51     int *q = &(*L).elem[(*L).length-1];     //表尾元素
 52     for(++p;p<=q;p++){          //被删除元素之后的元素左移
 53         *(p-1) = *p;            
 54     }
 55     --(*L).length;
 56 }
 57 
 58 void MergeList_Sq(SqList La, SqList Lb, SqList *Lc){
 59     //已知顺序表La,Lb的元素按值非递减排列
 60     //归并La,Lb得到新的顺序列表Lc,Lc的元素也按值非递减排列
 61     int *pa = La.elem;
 62     int *pb = Lb.elem;
 63     (*Lc).listsize = (*Lc).length = La.length + Lb.length;
 64     int *pc = (*Lc).elem = (int *)malloc((*Lc).listsize * sizeof(int));
 65     if(!(*Lc).elem){
 66         printf("内存分配失败...\n");
 67         exit(-1);
 68     }
 69     int *pa_last = La.elem + La.length -1;      //La的表尾元素地址
 70     int *pb_last = Lb.elem + Lb.length -1;      //Lb的表尾元素地址
 71     while(pa<=pa_last && pb<=pb_last){
 72         if(*pa <= *pb){
 73             *pc++ = *pa++;
 74         }else{
 75             *pc++ = *pb++;
 76         }
 77     }
 78     while(pa<=pa_last){         //插入La剩余的元素
 79         *pc++ = *pa++;          //插入Lb剩余的元素
 80     }
 81     while(pb <= pb_last){
 82         *pc++ = *pb++;
 83     }
 84 }
 85 
 86 //查看顺序表中第一个的满足compare()的元素的位序
 87 int LocateElem_Sq(SqList L, int e, int (* compare)(int x, int y)){
 88     int i = 1;
 89     int *p = L.elem;
 90     while(i <= L.length && !(*compare)(*p++, e)){
 91         ++i;
 92     }
 93     if(i <= L.length){
 94         return i;
 95     }else{
 96         return 0;
 97     }
 98 }
 99 
100 int cmp(int x, int y){
101     if(x == y){
102         //printf("x=y\n");
103         return 1;
104     }else{
105         //printf("x!=y\n");
106         return 0;
107     }
108 }
109 int main(){
110     SqList La;
111     SqList Lb;
112     SqList Lc;
113     int N,e;
114     InitList_Sq(&La);
115     printf("输入La元素的个数N:");
116     scanf("%d",&N);
117     printf("输入La的元素:");
118     while(N--){
119         scanf("%d",&e);
120         int i =La.length+1;
121         ListInsert_Sq(&La,i,e);
122     }
123     InitList_Sq(&Lb);
124     printf("输入Lb元素的个数N:");
125     scanf("%d",&N);
126     printf("输入Lb的元素:");
127     while(N--){
128         scanf("%d",&e);
129         int i =Lb.length+1;
130         ListInsert_Sq(&Lb,i,e);
131     }
132     printf("此时La中的元素顺序表的值为:");
133     for(int i=0;i<La.length;i++){
134         printf("%d ",La.elem[i]);
135     }
136     printf("\nLa.length = %d\tLa.listsize = %d\n",La.length,La.listsize);
137     printf("此时Lb中的元素顺序表的值为:");
138     for(int i=0;i<Lb.length;i++){
139         printf("%d ",Lb.elem[i]);
140     }
141     printf("\nLb.length = %d\tLb.listsize = %d\n",Lb.length,Lb.listsize);
142     ListDelete_Sq(&La,4,&e);
143     printf("删除第四个元素后La的元素顺序表的值为:");
144     for(int i=0;i<La.length;i++){
145         printf("%d ",La.elem[i]);
146     }
147     printf("\nLa.length = %d\tLa.listsize = %d\te = %d\n",La.length,La.listsize,e);
148     MergeList_Sq(La, Lb, &Lc);
149     printf("La和Lb合并后得到Lc,其Lc的信息为:\n");
150     printf("Lc的元素顺序表为:");
151     for(int i=0;i<Lc.length;i++){
152         printf("%d ",Lc.elem[i]);
153     }
154     printf("\nLc.length = %d\tLc.listsize = %d\n",Lc.length,Lc.listsize);
155     printf("输入你要在Lc中查找的元素e:");
156     scanf("%d",&e);
157     printf("该元素e在Lc的位序位:%d\n",LocateElem_Sq(Lc, e, cmp));
158   free(La.elem);free(Lb.elem);free(Lc.elem);return 0;
159 }

  结果为:

  

 

转载于:https://www.cnblogs.com/TsnTse/p/9070013.html

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

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

相关文章

Python_XlrdXlwt

1 import xlrd 2 # \U 开始的字符被编译器认为是八进制 解决方法 r 3 objWB xlrd.open_workbook(rC:\Users\IBM\Desktop\S1\7月下旬入库表.xlsx) 4 # 索引号 objTable objWB.sheet_by_index(0) 5 objTable objWB.sheet_by_name(7月下旬入库表) 6 # 单元格3种读取方式 7 print…

校招需要看的书 巩固的知识

前言 感谢教练&#xff0c;学长们&#xff0c;队友&#xff0c;lollipop&#xff0c;猫哥&#xff0c;李哥&#xff0c;表哥&#xff0c;鸡哥&#xff0c;样样&#xff0c;咸糖&#xff0c;茗记&#xff0c;明沙&#xff0c;嘻&#xff0c;树佬(排名不分先后)等等太多太多的人的…

新的Teams API权限控制

这篇继续介绍BUILD大会里的内容&#xff1a;新的Teams API权限。这些新的权限让开发者可以更加细粒度的设置权限。 之前有些开发人员有问过我&#xff0c;为什么Graph API的权限这么多&#xff0c;为什么不针对Teams弄一个总的权限&#xff0c;这样不是更加简单吗&#xff1f;…

物料主数据(MM03)跳转函数

CP_08_MATERIAL_SHOW 使用感觉能使自己的代码显得更改高端些。 其中参数MTSTA_IMP的选值参照表T132。转载于:https://www.cnblogs.com/tangcy1110/p/9081380.html

二叉树的蛇形遍历 leetcode 103

给定一个二叉树&#xff0c;返回其节点值的锯齿形层次遍历。&#xff08;即先从左往右&#xff0c;再从右往左进行下一层遍历&#xff0c;以此类推&#xff0c;层与层之间交替进行&#xff09;。 例如&#xff1a;给定二叉树 [3,9,20,null,null,15,7], 3/ \9 20/ \15 7返回…

Teams Tab的Single Sign-On

在我写这篇文章的时候&#xff0c;这个SSO机制还是在 Developer Preview 阶段&#xff0c;可能在发布前还会有一些改进。不过我觉得这个功能很好&#xff0c;所以先和大家分享一下。 如果大家之前已经开发过Teams的tab应用&#xff0c;可能会发现如果你需要一个当前用户的toke…

vim编辑器的使用--转自MJ学长

一、引言 1. vim是一款功能强大的文本编辑器&#xff0c;如果使用熟练&#xff0c;将会有效帮助我们提高编辑文本、程序的效率。vim编辑器的上手使用门槛比较高&#xff0c;很多人怯于要记很多命令&#xff0c;往往在学习的初期阶段就望而却步。 2. vim的学习需要不断的练习、使…

算法引入

算法的概念&#xff1a; 解决问题的思路。 时间复杂度&#xff1a; 定义&#xff1a; 基本运算的执行数量。是算法效率的衡量的量。 计算准则&#xff1a; 基本操作&#xff1a;即只有常数项。复杂度认为1顺序&#xff0c;按照加法计算循环&#xff0c;按照乘法计算条件。按照最…

如何开发Teams Bot

很多朋友问我如何开发一个成功的Teams Bot&#xff0c;他们说Bot Framework SDK看起来简单&#xff0c;但是真要的去开发一款成熟的bot&#xff0c;很多地方还是不知道如何使用。我从最早的bot framework还在beta的时候开始用&#xff0c;后来framework经历了多次大的改动&…

[CF903G]Yet Another Maxflow Problem

[CF903G]Yet Another Maxflow Problem 题目大意&#xff1a; 有\(A\)类点和\(B\)类点各\(n(n\le2\times10^5)\)个&#xff0c;所有\(A_i\)到\(A_{i1}\)有一条权值为\(a_i\)的有向边&#xff0c;所有\(B_i\)到\(B_{i1}\)有一条权值为\(b_i\)的有向边&#xff0c;另有\(m(m\le2\t…

P1579哥德巴赫猜想

写来自己学习用~ 题目内容&#xff1a; 1742年6月7日哥德巴赫写信给当时的大数学家欧拉&#xff0c;正式提出了以下的猜想&#xff1a;任何一个大于9的奇数都可以表示成3个质数之和。质数是指除了1和本身之外没有其他约数的数&#xff0c;如2和11都是质数&#xff0c;而6不是质…

在VSCode Remote环境下开发Teams Bot

我使用VS Code开发已经有蛮长一段时间了&#xff0c;时间长了&#xff0c;越来越喜欢VS Code&#xff0c;虽然有些时候会没有传统的VS方便&#xff0c;比如开发Azure Function时你需要编写一下launch.json&#xff0c;而且你需要手动启动StorageEmulator&#xff0c;但是也正是…

查看安卓APK源码破解

原文:查看安卓APK源码破解工具准备&#xff1a; <1>.android4me的AXMLPrinter2工具 <2>dex2jar <3>jd-gui 工具下载&#xff1a;http://download.csdn.net/detail/catshitone/8491347 开始&#xff1a; 第一步&#xff1a; 首先用解压软件&#xff08;如好…

实验六:类的封装

一、实验代码如下&#xff1a; 1 package 实验6;2 3 import java.util.Scanner;4 5 6 public class Account {7 8 public int id;9 public String name;10 public long number;11 public long time;12 public int money;13 14 //方法Account()…

Teams Bot开发系列:初识Bot

上次我们讲了Teams Bot开发的概述&#xff0c;讲了Azure Bot Service&#xff0c;Bot Framework SDK和我们自己的bot服务的概念&#xff0c;这篇文章就带大家看看Azure Bot Service和我们的bot是如何发生关系的。 我们自己开发的bot服务实际上就是一个api service&#xff0c;…

[环境搭建]SDN网络感知服务与最短路径应用

1.安装python模块networkxpip install networkx2.给Network_Awareness.py加修改权限chmod 777 Network_Awareness.py3.下载安装ryugit clone git://github.com/osrg/ryu.gitcd ryu sudo python ./setup.py install#若已安装ryu,删了再装&#xff0c; pip uninstall ryu4.修改“…

我需要别人承认才快乐吗?

关于生命的感悟两个故事第一个故事&#xff0c;一个尖子生考上了麻省理工学院&#xff0c;在那里所有同学都很优秀&#xff0c;竞争非常强烈&#xff0c;她发现再也不能出类拔萃&#xff0c;在各方面赢过别人&#xff0c;于是觉得生活看不到希望&#xff0c;郁郁寡欢&#xff0…

Teams Bot开发系列:Activity和Turn

这篇文章我们来说一下Activity和Turn这两个bot framework中最重要的两个概念&#xff0c;同时也介绍一下TurnContext和BotAdapter Activity 一个activity是聊天双方的一个信息载体&#xff0c;它可以是一条消息&#xff0c;也可以是一个动作。比如用户给bot发送一条文字消息&…

ubuntu16.04下安装opencv出现libgtk2.0-dev配置失败问题解决方法

第一次在ubuntu下安装opencv&#xff0c;遇到很多问题&#xff0c;特别是libgtk2.0-dev总是配置失败的问题&#xff0c;在网上也看到一些解决方法&#xff0c;自己也遇到一些比较奇葩的问题&#xff0c;故整理于此。 网上大部分的解决方案就是更改下载源&#xff0c;我看到一些…

03|模型I/O:输入提示、调用模型、解析输出

03&#xff5c;模型I/O&#xff1a;输入提示、调用模型、解析输出 从这节课开始&#xff0c;我们将对 LangChain 中的六大核心组件一一进行详细的剖析。 模型&#xff0c;位于 LangChain 框架的最底层&#xff0c;它是基于语言模型构建的应用的核心元素&#xff0c;因为所谓 …