数据结构——线性表之单链表的完整详细操作(实验报告)

单链表的基本操作

  • 首先,创建一个项目,就叫LinkList

  • 再新建三个文件
    LinkList.h是头文件、函数声明和配置数据
    LinkList.cpp是函数的具体实现
    main.cpp是调用函数和用户交互的主函数

  • LinkList.h文件

#ifndef LINKLIST_H_INCLUDED
#define LINKLIST_H_INCLUDED
#include<bits/stdc++.h>
#define TRUE 1
#define FALSE 0
#define OK 1
#define ERROR 0
using namespace std;

typedef int status;
typedef int ELemType;

typedef struct LNode{
    ELemType data;
    struct LNode * next;
}LNode,*LinkList;

//初始化
status InitList(LinkList& L);
//创建
status Create(LinkList& L);
//头插尾插
void HeadInsert(LinkList& L,int n);
void TailInsert(LinkList& L,int n);
//输出
void OutputList(LinkList L);
//求长度
int ListLength(LinkList L);
//查找
int LocateElem(LinkList L,ELemType e);
status GetElem(LinkList L,int i,ELemType& e);
//插入
status ListInsert(LinkList& L,int i,ELemType e);
//删除
status ListDelete(LinkList& L,int i,ELemType& e);
//判空
status ListEmpty(LinkList L);
//有序表的合并
status MergeSqList(LinkList La,LinkList Lb,LinkList& Lc);
void Merge();
//清空表
status clearList(LinkList& L);
//销毁表
status DestroyList(LinkList& L);
//逆序
status Reverse(LinkList& L);
//菜单
void menu();


#endif // LINKLIST_H_INCLUDED

  • LinkList.cpp文件
#include"LinkList.h"

//初始化
status InitList(LinkList& L)
{
    //创建头指针和头结点
    L=(LinkList)malloc(sizeof(LNode));
    if(!L)exit(OVERFLOW);
    L->next=NULL;
    L->data=0;//存储链表长度
    return OK;
}
//创建
status Create(LinkList& L){
    cout<<"请输入结点数:";
    int n;
    cin>>n;
    int flag=1;
    while(1){
        cout<<"请选择插入方法:\n"
            "1: 头插法(逆序)\n"
            "2: 尾插法(正序)\n";
        int select;
        cin>>select;
        cout<<"请输入数据:";
        switch(select)
        {
        case 1:
            HeadInsert(L,n);
            flag=0;
            break;
        case 2:
            TailInsert(L,n);
            flag=0;
            break;
        default:
            cout<<"输入选项错误,请重新输入!"<<endl;;
            break;
        }
        if(!flag)break;
    }
    return OK;
}
//头插
void HeadInsert(LinkList& L,int n){
    LinkList p;
    for(int i=0; i<n; i++)
    {
        p = (LinkList)malloc(sizeof(LNode));
        if(!p)exit(OVERFLOW);
        cin>>p->data;
        p->next = L->next;
        L->next = p;
        L->data++;
    }
}
//尾插
void TailInsert(LinkList& L,int n){
    LinkList tail=L;
    LinkList p;
    for(int i=0; i<n; i++)
    {
        p = (LinkList)malloc(sizeof(LNode));
        if(!p)exit(OVERFLOW);
        cin>>p->data;
        tail->next = p;
        tail = p;
        L->data++;
    }
    tail->next = NULL;
}
//输出
void OutputList(LinkList L)
{

    while(cin.get()!='\n');
    cout<<"输出ing...\n输入你想要的间隔符号(半角):";
    char ch=getchar();
    cout<<"当前的链表为:";
    for(LinkList p=L->next; p; p=p->next)
    {
        cout<<p->data<<ch;
    }
    cout<<endl;
}

//求长度
int ListLength(LinkList L)
{
    return L->data;
}

//查找
//按值查找
int LocateElem(LinkList L,ELemType e)
{
    int i=0;
    for(LinkList p=L->next; p; p=p->next){
        i++;
        if(p->data==e)return i;
    }
    return 0;
}
//按位置查找
status GetElem(LinkList L,int i,ELemType& e)
{
    LinkList p;
    int j;
    for(p=L->next,j=1; j<i&&p; p=p->next)j++;
    if(!p||j>i)return ERROR;//p为空,则i的长度大于链表长度; j>i,则i<1 过小
    e=p->data;
    return OK;
}

//插入
status ListInsert(LinkList& L,int i,ELemType e)
{
    //插入要找插入点的前一个结点
    LinkList p;
    int j=0;
    for(p=L; p&&j<i-1; p=p->next)j++;
    if(!p||j>i-1)return ERROR;//如果p为空,就不能再后面插入;但是p->next为空,还能接到后面
    LinkList s=(LinkList)malloc(sizeof(LNode));
    s->data = e;
    s->next = p->next;
    p->next = s;
    return OK;
}

//删除
status ListDelete(LinkList& L,int i,ELemType& e)
{
    //删除也是找的前一个结点
    LinkList p=L;
    int j;
    for(j=0; p->next&&j<i-1; p=p->next)j++;
    if(!(p->next)||j>i-1)return ERROR;
    LinkList q=p->next;
    p->next = q->next;
    free(q);
    return OK;
}

//判空
status ListEmpty(LinkList L)
{
    if(!(L->next))return TRUE;
    else return FALSE;
}

//顺序表的合并
status MergeSqList(LinkList La,LinkList Lb,LinkList& Lc)
{
    LinkList tail=Lc;
    LinkList p=La->next,q=Lb->next;
    while(p&&q){
        if(p->data<=q->data){
            tail->next=p;
            tail=p;
            p=p->next;
        }else{
            tail->next=q;
            tail=q;
            q=q->next;
        }
        Lc->data++;
    }
    while(p){
        Lc->data++;
        tail->next=p;
        tail=p;
        p=p->next;
    }
    while(q){
        Lc->data++;
        tail->next=q;
        tail=q;
        q=q->next;
    }
    return OK;
}
void Merge(){
    cout<<"请创建两个有序表!"<<endl;
    LinkList La,Lb,Lc;
    InitList(La);
    InitList(Lb);
    InitList(Lc);
    cout<<"创建La:"<<endl;
    Create(La);
    cout<<"创建Lb:"<<endl;
    Create(Lb);
    if(MergeSqList(La,Lb,Lc)){
        cout<<"合并成功!"<<endl;
        OutputList(Lc);
    }else{
        cout<<"合并失败!"<<endl;
    }
}
//清空表
status clearList(LinkList& L)
{
    LinkList p=L->next,q;
    while(p)
    {
        q=p->next;
        free(p);
        p=q;
    }
    L->next=NULL;
    return OK;
}

//销毁表
status DestroyList(LinkList& L)
{
    LinkList p=L,q;
    while(p)
    {
        q=p->next;
        free(p);
        p=q;
    }
    return OK;
}
//逆序
status Reverse(LinkList& L){
    if(L==NULL||L->next==NULL)return OK;
    LinkList pre=NULL,cur=L->next,next=NULL;
    while(cur){
        next=cur->next;
        cur->next=pre;
        pre=cur;
        cur=next;
    }
    L->next=pre;
    return OK;
}
void menu(){
    cout<<"\t\tHello,this is DuLinkList\n"
        "\t======================================\n"
        "\tplease select the operation you want:\n"
        "\t1. 创建\n"
        "\t2. 输出\n"
        "\t3. 求长度\n"
        "\t4. 查找指定元素的位置\n"
        "\t5. 查找指定位置的元素\n"
        "\t6. 插入\n"
        "\t7. 删除\n"
        "\t8. 判空\n"
        "\t9. 有序表的合并\n"
        "\t10.清空表\n"
        "\t11.销毁表\n"
        "\t12.逆序\n"
        "\t0. 退出\n"
        <<endl;
}
  • main.h文件
#include "LinkList.h"
using namespace std;

int main()
{
    int i,e;
    LinkList L;
    InitList(L);
    menu();
    int select;
    while(1){
        cout<<"\n<<<请输入你的操作:";
        cin>>select;
        switch(select)
        {
        case 1:
            if(Create(L))cout<<"创建成功!"<<endl;
            else cout<<"创建失败!"<<endl;
            break;
        case 2:
            OutputList(L);
            break;
        case 3:
            cout<<"当前单链表的长度是:"<<ListLength(L)<<endl;
            break;
        case 4:
            cout<<"请输入查找的元素:";
            cin>>e;
            if((i=LocateElem(L,e)))cout<<"该元素的位置是:"<<i<<endl;
            else cout<<"没有该元素!"<<endl;
            break;
        case 5:
            cout<<"请输入查找的位置:";
            cin>>i;
            if(GetElem(L,i,e))cout<<"该位置的元素是:"<<e<<endl;
            else cout<<"输入位置错误!"<<endl;
            break;
        case 6:
            cout<<"请输入插入的位置和元素,以空格分开:";
            cin>>i>>e;
            if(ListInsert(L,i,e)){
                cout<<"插入成功!"<<endl;
                OutputList(L);
            }
            else cout<<"插入失败!"<<endl;
            break;
        case 7:
            cout<<"请输入删除元素的位置:";
            cin>>i;
            if(ListDelete(L,i,e)){
                cout<<"删除成功!删除的元素是:"<<e<<endl;
                OutputList(L);
            }
            else cout<<"删除失败!"<<endl;
            break;
        case 8:
            if(ListEmpty(L))cout<<"是空!"<<endl;
            else cout<<"非空!"<<endl;
            break;
        case 9:
            Merge();
            break;
        case 10:
            if(clearList(L))cout<<"清空成功"<<endl;
            else cout<<"清空失败"<<endl;
            break;
        case 11:
            if(DestroyList(L))cout<<"销毁成功"<<endl;
            else cout<<"销毁失败"<<endl;
            break;
        case 12:
            if(Reverse(L)){
                cout<<"逆序成功!"<<endl;
                OutputList(L);
            }
            break;
        case 0:
            exit(1);
        default:
            cout<<"输入错误"<<endl;
            break;
        }
    }
    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值