Bootstrap

数据结构线性表——顺序表(C/C++(类模板))含代码

总结

就是我个人的学习总结。

  • 线性结构分为线性表(一般),栈、队列、串(特殊),数组(推广)
  • 同一线性表中的数据元素必定是具有相同的特性的,即属于同一数据对象,相邻元素间存在序偶关系。
  • 数据表其实就是一个数据对象,由n(n>=0)个数据元素构成的有限序列
  • 线性表可以分为顺序表和链表
  • 顺序表的特点:可随机存取(类似于数组)

C

//动态分配一维数组表示线性表
#include<stdio.h>
#include<stdlib.h>
#define MAX 100     //线性表的存储空间初始分配量
#define IN 10     //存储空间分配增量
typedef struct{
    int *elem;  //基址
    int length; // 表长
    int listsize;   //分配的存储容量,以sizeof(ElemType)为单位
}SqList;

线性表的基本操作实现:
1.初始化(创建)

//创建并初始化线性表
int InitList_Sq(SqList *L){
    (*L).elem=(int*)malloc(IN*sizeof(int));
    if(!(*L).elem)printf("error\n");    //空间分配失败
    (*L).length=0;                     //表长为0
    (*L).listsize=MAX;                  //初始存储容量
    return 0;
}

2.取值
跟据指定的位置序号i,获取表中第i个数据元素的值(就像数组一样通过下标定位获取元素的值)

//取位置i的元素
int GetElem_Sq(SqList L,int i,int *e){
    if(i<1||i>L.length)return 0;
    *e=L.elem[i-1];
    return 1;//返回1代表已经找到了元素,若想要输出元素,需要在加一个输出语句
}

3.查找:
根据指定元素e,查找表中第一个与e相等的元素elem[i],成功后返回i+1.

int LocateElem_Sq(SqList L,int e){
     //在表中查找第一个值与ecompare()的元素的位序
    //如若找到返回其位序,否则返回0
    int i=0;
    int j;
    int *p=L.elem;
    for(;i<L.length;i++){
     if(L.elem[i]==e)
         j=i+1;
 }
    if(i<L.length){
        return j;
    }
    return 0;
}

4.插入:

int ListInsert_Sq(SqList *L,int i,int e){
    int *newbase;
    int *p,*q;
    if(i<1||i>(*L).length)return 0;    //i值不合法
    if((*L).length>=(*L).listsize){//存储空间已满,增加分配
        newbase=(int*)realloc((*L).elem,((*L).listsize+10)*sizeof(int));
        (*L).elem=newbase;//新基址
        (*L).listsize+=10;         //增加存储容量
    }
    q=&(*L).elem[i-1];//插入元素的位置
    p=&(*L).elem[(*L).length-1];
    for(;q<=p;p--){
     *(p+1)=*p;
 }
    *q=e;              //插入
    (*L).length++;        //表长+1
    return 1;
}

5.删除:

int ListDelete_Sq(SqList *L,int i,int *e){
    //删除表中第i个元素,并用e返回其值
    int j;
    int *p,*q;
    if(i<1||i>(*L).length)return 0;//i值不合法
    p=&(*L).elem[i-1];       //p为被删除的位置
    *e=*p;                   //赋值给e
    q=(*L).elem+(*L).length-1;//表尾部元素位置
    while(p<=q){           //元素后移
        *p=*(p+1);
        ++p;
    }
    (*L).length--;   //表长-1
    return 1;
}

6.创建一个线性表(初始化)

void CreateList_Sq(SqList *L,int n){
    printf("输入%d个元素:",n);
    int i=0;
    for(;i<n;i++){
        scanf("%d",&(*L).elem[i]);
    }
    printf("\n");
    (*L).length=n;
}

7.合并两个线性表为一个:

//合并两个线性表为一个线性表
void MargsList_Sq(SqList La,SqList Lb,SqList *Lc){
    //已经知道线性表元素按值非递减顺序排列
    //归并La和Lb得到新的顺序表Lc的元素也是按值非递减排列的
    int *pa;int *pb;int *pc;int *pa_last;int *pb_last;
    pa=La.elem;pb=Lb.elem;
    (*Lc).listsize=(*Lc).length=La.length+Lb.length;
    pc=(*Lc).elem=(int *)malloc((*Lc).listsize*sizeof(int));
    if(!(*Lc).elem)exit;
    pa_last=La.elem+La.length-1;
    pb_last=Lb.elem+Lb.length-1;
    int i=0;
    for(;pa<=pa_last&&pb<=pb_last;pa++,pb++,pc++){
     if(*pa<=*pb){
      *pc=*pa;
      *(++pc)=*pb;
  }
  else{
   *pc=*pb;
   *(++pc)=*pa;
  }           
 } 
}

8.一些简单函数

//判断空表
int ListEmpty_Sq(SqList L){
    return (L.length==0);
}
//表长
int ListLength_Sq(SqList L){
    return L.length;
}
//遍历线性表
int ListTraverse_Sq(SqList L){
    int i;
    for(i=0;i<L.length-1;i++){
        printf("%d  ",L.elem[i]);
    }
    printf("\n");
}

对一些基本操作可以如下主函数检测

int main(){
    SqList L;
    int e;
    int i,a;
    if(InitList_Sq(&L)==0)printf("初始化成功!!!\n");
    if(ListEmpty_Sq(L))
        printf("顺序表为空!\n");
    for(i=0;i<10;i++){
        ListInsert_Sq(&L,i,2*i);
    }
    printf("输出所有元素:");
    ListTraverse_Sq(L);
    printf("输出表长:%d\n",ListLength_Sq(L));
    printf("输出表长是否为空:");
    if(ListEmpty_Sq(L)==1)
        printf("空表!\n");
    else
        printf("非空!\n");
    printf("输出顺序表第3个元素到e:");
    GetElem_Sq(L,3,&e);
    printf("%d\n",e);
    printf("查找元素a");
    scanf("%d",&a);
    printf("元素%d在表中的位置是:%d\n",a,LocateElem_Sq(L,a));
    printf("在表中第4个位置出入%d\n",a+1);
    ListInsert_Sq(&L,4,a+1);
    printf("输出操作后的表:\n");
     ListTraverse_Sq(L);
    printf("删除表中第3个位置的元素:\n");
    ListDelete_Sq(&L,3,&e);
    printf("输出删除的表:\n");
    ListTraverse_Sq(L);
    return 0;
}
        



或者是:

int main(){
    SqList L;
    int e;
    int i,a;
    if(InitList_Sq(&L)==0)printf("初始化成功!!!\n");
    CreateList_Sq(&L,10);
    printf("输出所有元素:");
    ListTraverse_Sq(L);
    printf("输出表长:%d\n",L.length);
    printf("查找元素a");
    scanf("%d",&a);
    printf("元素%d在表中的位置是:%d\n",a,LocateElem_Sq(L,a));
    printf("在表中第4个位置出入%d\n",a+1);
    ListInsert_Sq(&L,4,a+1);
    printf("输出操作后的表:\n");
    ListTraverse_Sq(L);
    printf("删除表中第3个位置的元素:\n");
    ListDelete_Sq(&L,3,&e);
    printf("输出删除的表:\n");
    ListTraverse_Sq(L);
    return 0;
}


合并俩个线性表为一个:

int main(){
    SqList La,Lb,Lc;
    int e;
    int i,a;
    InitList_Sq(&La);
    InitList_Sq(&Lb);
    InitList_Sq(&Lc);
    CreateList_Sq(&La,5);
    CreateList_Sq(&Lb,5);
    printf("输出所有元素:");
    ListTraverse_Sq(La);
    ListTraverse_Sq(Lb);
    printf("输出表长:La=%d\tLb=%d\tLc=%d\n",La.length,Lb.length,Lc.length);
    MargsList_Sq(La,Lb,&Lc);
    printf("输出表长:La=%d\tLb=%d\tLc=%d\n",La.length,Lb.length,Lc.length);
    printf("输出操作后的表:\n");
    ListTraverse_Sq(Lc);
    return 0;
}

C++类(不用模板)

代码:

typedef int DataType;
const int MaxSize=10;          //默认最大表长 
class SqList{
private: 
 DataType *elem;//首地址
 int length;   //实际表长 
 int size;      //最大表长 
public:
 //构造函数(初始化) 
 SqList(){
  size=MaxSize;
  length=0;
  elem=new DataType[size];//分配内存空间
  for(int i=0;i<size;i++){
   elem[i]=NULL;//也可以elem[i]=NULL; 
  } 
 } 
 //析构函数(销毁线性表)
 ~SqList(){
  delete[] elem; //回收内存空间 
 } 
 //基本函数声明
 bool insertelem(DataType Data); //在表尾插入元素 
 bool deleteelem(int location);//删除指定位置处的元素 
 bool changeelem(int location,DataType newData);//修改指定位置处的元素值
 DataType getelem(int location);//返回指定位置处的元素值 
 bool traverseList();//遍历 
 bool createList();//创建 
 int locateElem(DataType Data);//查找元素位置  
 //获取表长
 int lengthlist(){
  return length;
 }    
};
//函数的定义
//在表尾插入新元素
bool SqList::insertelem(DataType Data){
 if(length>=MaxSize){
  cout<<"线性表已满!"<<endl;
  return false;          
 }
 else{
  elem[length]=Data;
  length++;
  return true;
 }
} 
//删除指定位置处的元素
bool SqList::deleteelem(int location){
 if(location<0||location>length){
  cout<<"参数错误!"<<endl;
  return false;
 }
 else{
  for(int i=location-1;i<length;i++){
   elem[i]=elem[i+1];
  }
  length--;
  return true;
 }
} 
//修改指定位置处的元素值
bool SqList::changeelem(int location,DataType newData){
 if(location<0||location>length){
  cout<<"参数错误!"<<endl;
  return false;
 }
 elem[location-1]=newData;
 return true;
} 
//返回指定位置处的元素值 
DataType SqList::getelem(int location){
 if(location>length||location<0){
  cout<<"参数有误!"<<endl;
  return 0; 
 } 
 else{
  return elem[location-1];
 }
}
//遍历
bool SqList::traverseList(){
 for(int i=0;i<length;i++){
  cout<<elem[i]<<'\t';
 }
 cout<<endl;
 return true;
} 
//创建 
bool SqList::createList(){
 int k,n;
 while(1){
  cout<<"输入要输入的元素个数:"<<endl;
     cin>>n;
     if(n<=size){
      break;
  }
     else {
      cout<<"输入的元素个数太大!"<<endl;
      return false;
     }
 } 
 cout<<"输入元素:"<<endl; 
 for(int i=0;i<n;i++){
  cin>>k;
  insertelem(k);
 }
 return true;
} 
//查找元素位置
int SqList::locateElem(DataType Data){
 int k=0;
 for(int i=0;i<length;i++){
  if(elem[i]==Data){
   k=i+1;
   return k;
     }
 }   
    if(k==0)return 0;
}
//将两个非递减的线性表合并为一个线性表
bool MargsList(SqList L1,SqList L2,SqList &L3){
 cout<<"合并两个表为一个表!"<<endl;
 DataType p,q;
 int k=1;
 for(int i=1;i<=L1.lengthlist();i++){
  p=L1.getelem(i);
  for(int j=k;j<=L2.lengthlist();j++){
   q=L2.getelem(j);
   if(p>q){
    k=j+1;
          L3.insertelem(q);
   }
   else {
    k=j;break;
   } 
  }
  L3.insertelem(p);
 }
 return true;
}
//主函数
int main(){
 SqList L1,L2,L3;
 //初始化 
 L1.createList();
 L2.createList();  
 //输出初始化后的顺序表
 L1.traverseList();
 L2.traverseList();  
 //将顺序表索引为5的元素值改为44
 L1.changeelem(5,44);
 L2.changeelem(5,44); 
 //输出修改后的顺序表
 L1.traverseList();
 L2.traverseList(); 
 //将表L1,L2合并得到L3 
 MargsList(L1,L2,L3);
 L3.traverseList();
 return 0;
} 

C++(类模板)

可以看这篇文章了解类模板的基础知识:
https://blog.csdn.net/m0_47484802/article/details/109142448

1.类模板,基本数据结构

#include<iostream>
using namespace std;
/*******线性表的数据结构********/ 
const int defaultSize=10; //设置默认顺序表大小 
template<typename DataType>
class SeqList{
private:
    DataType *elements;      //首地址 
 int maxSize;         //顺序表最大大小 
 int length;          //顺序表的有效长度 
public:
 //构造函数(初始化) 
 SeqList(int size=defaultSize){
  if(size>0){
   maxSize=size;
   length=0; 
   elements=new DataType[maxSize];//分配内存大小 
   for(int i=0;i<maxSize;i++){
    elements[i]=NULL;
   }
  }
 }
 //析构函数
 ~SeqList(){
  delete[] elements; //回收内存空间 
 }  
 //基本操作
 bool insertElement(DataType data);  //在表尾插入新元素
 bool deleteElement(int location);    //删除指定位置的元素 
 DataType getElement(int location);    //获取指定位置的元素 
 bool changeElement(int location,DataType newData);//修改指定位置处的元素值  
 int getLength(){//获取表长 
  return length;
 };
} ;

2.基本操作

//在表尾插入新元素
template<typename DataType> bool SeqList<DataType>::insertElement(DataType data){
 int currentIndex=length;//记录新元素的插入位置
 if(length>maxSize){
  return false;//表满 
 } 
 else{
  elements[currentIndex]=data;// 将新元素插入表尾
  length++;//表长加1
  return true; 
 } 
} 
 //删除指定位置的元素
template<typename DataType> bool SeqList<DataType>::deleteElement(int location){
 if(location>=length||location<0){
  return false; //判断位置是否合法 
 }
 for(int i=location;i<length;i++){
  elements[i]=elements[i+1];//从location位置开始,后一个依次覆盖前一个 
 } 
 length--;//表长减1
 return true; 
}
//获取指定位置的元素值
template<typename DataType> DataType SeqList<DataType>::getElement(int location){
 if(location>=length||location<0){
  cout<<"参数无效!\n" ;
  return false;//判断位置是否合法 
 }
 return elements[location]; //返回指定元素 
} 
//修改指定位置的元素值
template<typename DataType>bool SeqList<DataType>::changeElement(int location,DataType newData){
 if(location<0||location>=length){
  cout<<"参数无效!"<<endl;  //判断位置是否合法 
  return false; 
 }
 elements[location]=newData; //赋值 
 return true;
}

3.主函数实现

//主函数 
int main(){
 SeqList<int> list(10); //建立顺序表
 for(int i=0;i<10;i++){
  list.insertElement(i*10); //插入数据实现初始化 
 } 
 //输出初始化后的顺序表
 for(int i=0;i<list.getLength();i++){
  cout<<list.getElement(i)<<'\t';
 }
 cout<<endl;
 //将顺序表索引为5的元素值改为44
 list.changeElement(5,44);
 //输出修改后的顺序表
 for(int i=0;i<list.getLength();i++){
  cout<<list.getElement(i);
  cout<<'\t';
 } 
 cout<<endl;
 return 0;
}
;