线性表的顺序表示和实现

目录

注:部分素材来源于我的大学老师

1、线性表的抽象数据类型的定义

2、线性表的顺序表示和实现

 2.1、例1

3、顺序表存储结构的实现

3.1、图书表的顺序存储结构类型定义

3.1、补充:C语言的动态分配函数( )

3.2、补充:C++的动态存储分配

3.3、补充:C++中的参数传递

3.3.1、传值方式

3.3.2、传地址方式

4、线性表的重要基本操作

4.1、补充:操作算法中用到的预定义常量和类型

4.2、初始化线性表L (参数用引用)

4.3、初始化线性表L (参数用指针)

4.4、销毁线性表L

4.5、清空线性表L

4.6、求线性表L的长度

4.7、判断线性表L是否为空

4.8、取值(根据位置i获取相应位置数据元素的内容)

4.9、查找(根据指定数据获取数据所在的位置)

4.10、插入

4.11、删除

5、顺序表的优缺点


注:部分素材来源于我的大学老师

1、线性表的抽象数据类型的定义

ADT List{

数据对象:D={ai|ai∈Elemset,i=1,2,…,n,n≥0}

数据关系:R={<ai-1,ai>|ai-1,ai∈D,i=2,…,n}

基本操作

 (1) 初始化:InitList(&L)。

      操作前提:L为未初始化线性表。

      操作结果:将L初始化为空表。

(2) 销毁表:DestroyList(&L)。

      操作前提:线性表L已存在。

      操作结果:释放L空间,将L销毁

(3) 清空表:ClearList(&L)。

      操作前提:线性表L已存在。

      操作结果:将表L置为空表。

(4) 判断表是否为空:ListEmpty (L)。

      操作前提:线性表L已存在。

      操作结果:如果L为空表则返回真,否则返回假。

(5) 求长度:ListLength(L)。

      操作前提:线性表L已存在。

      操作结果:如果L为空表则返回0,否则返回表中的元素个数。

  (6) 查找元素:LocateElem(L,e)。

      操作前提:表L已存在,e为合法元素值。  

      操作结果:如果L中存在元素e,则将“当前指针”指向首次找到的其值为e的元素所在位置并返回真,否则返回假。

(7) 取元素:GetElem(L,i,&e)。

      操作前提:表L存在,且i值合法,即1≤i≤ListLength(L)。

      操作结果:用e返回线性表L中第i个元素的值。

  (8) 插入元素:InsertList(&L,i,e)。

      操作前提:表L已存在,e为合法元素值,且 1≤i≤ListLength(L) + 1。

      操作结果:在L中第i个位置插入新的数据元素e,L的长度加1。

(9) 删除元素:DeleteList(&L,i,&e)。

      操作前提:表L已存在且非空,1≤i≤ListLength(L)。

      操作结果:删除L的第i个数据元素,并用e返回其值,L的长度减1。

(10) 显示线性表:Display(L)。

      操作前提:线性表L已存在且不为空。

      操作结果:依次显示表中所有元素的值。

(11)PriorElem(L,cur_e,&pre_e)
      初始条件:线性表L已经存在。
      操作结果:若cur_e是L的数据元素,且不是第一个,则用pre_e返回它的前驱,否则操作失败;pre_e无意义。 

(12)NextElem(L,cur_e,&next_e)
      初始条件:线性表L已经存在。

      操作结果:若cur_e是L的数据元素,且不是最后个,则用next_e返回它的后继,否则操作失败,next_e无意义。

(13)ListTraverse(&L,visited())
      初始条件:线性表L已经存在。

      操作结果:依次对线性表中每个元素调用visited()。

}ADT List

        上述是线性表抽象数据类型的定义,其中只是一些基本操作,另外可以更复杂。 如:将两个线性表合并等。 复杂的操作可用基本操作实现。

2、线性表的顺序表示和实现

线性表的顺序表示又称为顺序存储结构或顺序映像

>>顺序存储定义:把逻辑上相邻的数据元素存储在物理上相邻的存储单元中的存储结构。

>>简言之,逻辑上相邻,物理上也相邻

>>顺序存储方法:用一组地址连续的存储单元依次存储线性表的元素,可通过数组V[n]来实现。

>>任一元素均可随机存取(优点)。

 2.1、例1

3、顺序表存储结构的实现

#define  MAXSIZE 100     //最大长度
typedef  struct {
  ElemType  *elem; //指向数据元素的基地址
  int  length;  //线性表的当前长度                                                      
 }SqList;
元素类型:
例 typedef int ElemType;
例 typedef  struct  card
{   int num;
     char name[20];
     char author[10];
     char  pub[30];
     float  price;
}ElemType; 

3.1、图书表的顺序存储结构类型定义


#define MAXSIZE 10000	//图书表可能达到的最大长度 
typedef struct			//图书信息定义
{ 
   char no[20];			//图书ISBN
   char name[50];		//图书名字
   float price; 		//图书价格
}Book; 
typedef struct
{ 
   Book *elem;	//存储空间的基地址 
   int length;	//图书表中当前图书个数 
}SqList;		//图书表的顺序存储结构类型为SqList

3.1、补充:C语言的动态分配函数( <stdlib.h> )

malloc(m):开辟m字节长度的地址空间,并返回这段空间的首地址

sizeof(x):计算变量x的长度

free(p):释放指针p所指变量的存储空间,即彻底删除一个变量

SqList L;
L.elem=(ElemType*)malloc(sizeof(ElemType)*MaxSize);

3.2、补充:C++的动态存储分配

new  类型名T(初值列表)

>>功能: 申请用于存放T类型对象的内存空间,并依初值列表赋以初值

>>结果值:

    成功:T类型的指针,指向新分配的内存

    失败:0(NULL)

int *p1= new int;
或 
int *p1 = new int(10);

delete 指针P 

>>功能: 释放指针P所指向的内存。

>>P必须是new操作的返回值

delete p1;

3.3、补充:C++中的参数传递

1.函数调用时传送给形参表的实参必须与形参在类型个数顺序上保持一致

2.参数传递有两种方式

>>传值方式(参数为整型、实型、字符型等)

>>传地址

     参数为指针变量

     参数为引用类型

     参数为数组名

3.3.1、传值方式

1.把实参的值传送给函数局部工作区相应的副本中,函数使用这个副本执行必要的功能。

2.函数修改的是副本的值,实参的值不变

#include <iostream>
void swap(float m,float n)
{float temp;
 temp=m;
 m=n;
 n=temp;
}


int main()
{float a,b;
 cin>>a>>b;
 swap(a,b);
cout<<a<<endl<<b<<endl;
return 0;
}

3.3.2、传地址方式

>>指针变量作参数

形参变化影响实参

#include <iostream>
void swap(float *m,float *n)
{float t;
 t=*m;
 *m=*n;
 *n=t;
}

int main()
{float a,b,*p1,*p2;
 cin>>a>>b;
 p1=&a;   p2=&b; 
  swap(p1, p2);
cout<<a<<endl<<b<<endl;
return 0;
}

形参变化不影响实参??

#include <iostream>
void swap(float *m,float *n)
{float *t;
 t=m;
 m=n;
 n=t;
}


int main()
{float a,b,*p1,*p2;
 cin>>a>>b;
 p1=&a;   p2=&b; 
  swap(p1, p2);
cout<<a<<endl<<b<<endl;
return 0;
}

>>引用类型作参数

引用:它用来给一个对象提供一个替代的名字。

#include<iostream>
int main(){
	int i=5;
	int &j=i;
	i=7;
	cout<<"i="<<i<<" j="<<j;
   return 0;
}

//j是一个引用类型, 代表i的一个替代名,i值改变时,j值也跟着改变,所以会输出i=7 j=7
#include <iostream>
void swap(float& m,float& n)
{float temp;
 temp=m;
 m=n;
 n=temp;
}

int main()
{float a,b;
 cin>>a>>b;
 swap(a,b);
cout<<a<<endl<<b<<endl;
return 0;
}

引用类型作形参的三点说明:

(1)传递引用给函数与传递指针的效果是一样的,形参变化实参也发生变化

(2)引用类型作形参,在内存中并没有产生实参的副本,它直接对实参操作;而一般变量作参数,形参与实参就占用不同的存储单元,所以形参变量的值是实参变量的副本。因此,当参数传递的数据量较大时,用引用比用一般变量传递参数的时间和空间效率都好。

(3)指针参数虽然也能达到与使用引用的效果,但在被调函数中需要重复使用“*指针变量名”的形式进行运算,这很容易产生错误且程序的阅读性较差;另一方面,在主调函数的调用点处,必须用变量的地址作为实参。

>>数组名作参数

1.传递的是数组的首地址

2.对形参数组所做的任何改变都将反映到实参数组中

#include<iostream>
void sub(char);
int main(void )
{   char a[10]=“hello”;
     sub(a);
     cout<<a<<endl;
    return 0;
}
void sub(char b[ ])
{   b[ ]=“world”;}
//用数组作函数的参数,求10个整数的最大数

#include <iostream>
#define N 10
int max(int a[]);
int main ( ) {
	int a[10];
	int i,m;
	for(i=0;i<N;i++)
		cin>>a[i];
	m=max(a);
	cout<<"the max number is:"<<m;
            return 0;
}

int max(int b[]){
      int i,n;
       n=b[0];
       for(i=1;i<N;i++)
	if(n<b[i]) n=b[i];
        return n;
}

4、线性表的重要基本操作

4.1、补充:操作算法中用到的预定义常量和类型

//函数结果状态代码
#define TURE 1
#define FALSE 0
#define OK 1
#define ERROR 0
#define INFEASIBLE -1
#define OVERFLOW -2

//Status 是函数的类型,其值是函数结果状态代码
typedef int Status;
typedef char ElemType;

4.2、初始化线性表L (参数用引用)

Status InitList_Sq(SqList &L){        //构造一个空的顺序表L
    L.elem=new ElemType[MAXSIZE];     //为顺序表分配空间
    if(!L.elem){
        exit(OVERFLOW);                //存储空间分配失败,elem的值为空值
    }
    L.length=0;                        //空表长度为0
    return OK;
}

4.3、初始化线性表L (参数用指针)

Status InitList_Sq(Sqlist *L){
    L->elem=new ElemType[MAXSIZE];
    if(!L->elem){
        exit(OVERFLOW);
    }
    L->length=0;
    return OK;
}

4.4、销毁线性表L

void DestroyList(SqList &L){
    if(L.elem){
        delete[]L.elem;
    }
}

4.5、清空线性表L

void ClearList(SqList &L){
    L.length=0;                //将线性表的长度置为0
}

4.6、求线性表L的长度

int GetLength(SqList L){
    return (L.length);
}

4.7、判断线性表L是否为空

int IsEmpty(SqList L){
    if(L.Length==0){
        return 1;
    }else{
        return 0;
    }
}    

4.8、取值(根据位置i获取相应位置数据元素的内容)

int Get(SqList L,int i,ElemType &e){
    if(i<1 || i>L.length){
        return ERROR;        //判断i值是否合理,若不合理,返回ERROR
    }
    e=L.elem[i-1];           //第i-1的单元存储着第i个数据,随机存取
}

4.9、查找(根据指定数据获取数据所在的位置)

int LocateElem(Sqlist L,ElemType e){
   for(int i=0;i<L.length;i++){
        if(L.elem[i]==e){
            return i+1;
        }
    }
    return 0;
}

4.10、插入

        线性表的插入运算是指在表的第i(1≤i≤n+1)个位置上,插入一个新结点e,使长度为n的线性表(a1,...,ai-1,ai,...,an)变成长度为n+1的线性(a1,...,ai-1,e,ai,.,.an)

(1)判断插入位置i 是否合法。

(2)判断顺序表的存储空间是否已满。    

(3)将第n至第i 位的元素依次向后移动一个位置,空出第i个位置。

(4)将要插入的新元素e放入第i个位置。

(5)表长加1,插入成功返回OK。

Status ListInsert_Sq(SqList &L,int i,ELemType e){
    if(i<1 || i>L.length+1){
        return ERROR;            //i值不合理
    }
    if(L.length==MAXSIZE){
        return ERROR;            //当前存储空间已满
    }
    for(int j=n-1;j>=i-1;j--){
        L.elem[j+1]=L.elem[j];    //插入位置及之后的元素后移
    }
    L.elem[i-1]=e;                //将新元素e放入第i个位置
    L.length++;                   //表长增1 
    return OK;
}

4.11、删除

        线性表的删除运算是指将表的第i(1≤i≤n)个结点删除使长度为n的线性表(a1,...,ai-1,ai,ai+1,...,an)变成长度为n-1的线性表(a1,...,ai-1,ai+1,...,an)

(1)判断删除位置i 是否合法(合法值为1≤i≤n)。

(2)将欲删除的元素保留在e中。      

(3)将第i+1至第n 位的元素依次向前移动一个位置。

(4)表长减1,删除成功返回OK。 

Status ListDelete_Sq(SqList &L,int i){
    if(i<1 || i>L.length){
        return ERROR;                    //i值不合理
    }
    for(int j=i;j<=L.length-1;j++){
        L.elem[j-1]=L.elem[j];           //被删除元素之后的元素前移
    }
    L.length--;                          //表长减1
    return OK;
}

5、顺序表的优缺点

  • 优点:
    • 存储密度大(结点本身所占存储量/结点结构所占存储量)
    • 可以随机存取表中任一元素
  • 缺点:
    • 在插入、删除某一元素时,需要移动大量元素
    • 浪费存储空间
    • 属于静态存储形式,数据元素的个数不能自由扩充
  • 5
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值