单链表的基本操作代码实现


前言:

        在实现顺序表的基本操作后,觉得自己对单链表基本操作的思路无大问题,因此当时没有对链表基本操作进行实现,在后来的稀疏多项式的运算中需要运用到单链表(顺序表实现会造成大量空间的浪费),而自己并没有实现过这些基本操作,为防止在做稀疏多项式的运算时出现难以解决的问题,打算再把单链表的基本操作实现一遍。并写一篇博客,耗时:70分钟。

        希望能够对您有一定帮助和参考价值。新csdner,有所不足尽管提出。

        如果看完后有点点帮助,可以去CSDN给俺点一个赞嘛?(期待脸q(≧▽≦q))

        单链表的基本操作代码实现(C语言版)_KT pro的博客-CSDN博客

单链表的基本操作

        单链表的基本操作包括单链表的初始化判断空链表销毁清空以及求表长这些较为简单的操作,还有更为重要的单链表的取值、按值查找返回元素所在地址、按值查找返回元素所对应序号、结点插入和删除以及头插法和尾插法建立单链表。目前只实现了这些操作,并进行的测试。

准备工作(头文件、各种宏定义以及结构体定义)

#include <stdio.h>
#include <stdlib.h>
#include<stdbool.h>

#define OK 1
#define ERROR 0        /*用于返回函数执行的状态值*/

typedef int Status;
typedef int ElemType;

typedef struct Node {

    ElemType data;             //存储数据
    struct Node *next;//存储下一个元素的地址

} Node;

typedef struct Node *LinkList; /* 定义LinkList */

一.较简单操作

1.单链表的初始化

        单链表的初始化指生成一个只有头指针的单链表,并将其指向的头结点的指针域置空。需要注意的是,传入的参数为LinkList *L;为什么L本来就是指针类型了还要加   “ * “呢,因为你要知道你是要对链表的本质(内存,指针指向等)进行更改而不是使用链表进行元素的查找取值等,如果只是这些操作,只要告诉计算机你要查找的对象是哪个表,即把链表的头指针传过去就行(这样就不需要添加 “ * “ )。以下是代码实现:

/* 初始化链式线性表 */
Status IniList(LinkList *L) {

    *L=(LinkList)malloc(sizeof(Node)); /* 产生头结点,并使L指向此头结点 */

    (*L)->next=NULL; /* 指针域为空 */

    return OK;
}

2.判断单链表是否为空表

        只需要判断对应单链表头结点的指针域是否为空即可。该实现我使用的是bool类型返回值,实际上c语言中逻辑值就是0,1,因此多此一举了属于是;如果要用bool类型需要添加头文件#include<stdbool.h>。以下是代码实现:

bool Empty(LinkList L) {

    if(L->next==NULL)
        return true;    

    return false;
//main函数中测试bool c=Empty(L); printf("%d",c);  返回 1
}

3.单链表的销毁

        单链表的销毁指将链表中的所有结点包括头结点都释放掉(c语言中使用free(),c++中使用delete())。由于仍然是对单链表的“本质”进行修改,所以参数需加  “ * “ ,思路是利用头指针以及新指针p(用于释放其指向的结点),在释放之前,先将头指针后移,否则找不到下一结点。以下是代码实现:

/* 单链表的销毁*/         

Status DestoryList(LinkList *L) {
    LinkList p=*L;              //p此时指向头结点

    //注意此时不能直接释放p,因为头结点中还存放着下一结点的地址,释放就找不到下一个结点了
    while(*L) {
        *L=(*L)->next;
        free(p);            
        p=*L;    
    }

    return OK;

    //测试int b=DestoryList(&L);  printf("%d",b);ListTraverse(L); 输出1,且并不打印其他的
}

4.单链表的清空

        将除头结点以外的结点释放,并将头结点的指针域置空。与销毁不同,该操作的头指针不能动,所以还需要新定义一个指针对下一结点进行定位。以下是代码实现:

/*清空链表*/
Status ClearList(LinkList L) {

    LinkList p,q;    //p用于指向需要释放的结点,q用于在p释放之前指向下一结点,以使p移动
    p=L->next;       //p指向首元结点

    while(p) {
        q=p->next;   //q指向下一结点
        free(p);
        p=q;
    }

    //最后不要忘了,结点释放后还需要将头结点的指针域更改为NULL,形成空表
    L->next=NULL;

    return OK;

    /*测试CreateList(&L,n); ClearList(L);ListTraverse(L);//不输出创建时输入的元素*/

}

5.求单链表的表长

        以下是代码实现:

/*求链表的表长*/
int ListLength(LinkList L) {

    LinkList p;   //用于指向除头结点以外的结点
    int count=0;
    p=L->next;

    while(p) {
        count++;
        p=p->next;
    }

    return count;    

 //测试  printf("%d",ListLength(L)); 输出正确
}

二.较重要操作

1.单链表的取值

        取出链表中位置 i 对应的元素,i表示的是序号,不按逻辑层面确定。思路:将指针p移动到对应序号逻辑层面存储位置,随后将其指向的元素返回即可。注意循环终止条件,以下是代码实现:

/*获得某一元素*/  //返回链表中第i位置的元素
Status GetElem(LinkList L,int i,ElemType *e) {

    Node *p;
    p=L->next;
    int j = 1;

    for(; p&&j<i ;) {
        p = p->next; /*p所指元素存在且j<1时循环执行*/
        ++j;
    }

    if(!p||j>i) //j大于i的情况:如i传入的为0或负数
        return ERROR;

    *e=p->data;

    return OK;
 //测试:元素返回正确
}

2.单链表元素的查找

        查找分为按值查找返回元素所在地址和返回所在序号,思路:定义一个指针p指向头结点,随后利用循环将指针p进行移动即对链表进行遍历,将每次循环时结点的数据与待查找数据比较,如果找到则返回对应地址(反映为当前指针的指向),以及对应的序号。以下是代码实现:

//查找
/*按值查找数据所在地址并返回*/  //参数为所需查询链表,需要查询的数据,返回p的地址
LinkList searchAD(LinkList L,ElemType e) {

    LinkList p;
    p=L->next;

    for(int i=0; p&&i<ListLength(L); i++) {
        if(p->data==e)
            break;
        p=p->next;
    }

    return p; 

//测试;printf("%p",searchAD(L,5));  5所在元素次序不同,打印的地址不同

}

/*按值查找数据所在链表的序号(第几个元素)*/  //参数:所需查询链表,所需查询数据,返回数据所在序号
int searchNum(LinkList L,ElemType e) {

    LinkList p;
    p=L->next;

    for(int i=0; p&&i<ListLength(L) ; i++) {
        if(p->data==e) {    //遍历链表中p所指项结点的数据,与e比较,找到后返回对应序号即可
            return i+1;
        }

        p=p->next; //当前结点的数据不等于e则p指针后移
    }

    return 0; 

 //测试:printf("%d",searchNum(L,5));  返回5所在位置正确,如果不存在所查询数据,返回0
}

3.单链表的结点插入

        思路:先将待插入位置找到,利用其前一个位置的指针p对其后继进行保留,随后定义一个新结点并用指向它的指针q,进行前驱后继的链接。以下是代码实现:

//链表元素的插入    //参数:需要插入数据的链表,需要插入元素的位置,需要插入的数据,返回插入结束状态
Status ListInsert(LinkList L,int i,ElemType e) {

    LinkList p,q;
    p=L;        //指向头结点

    //先新建并初始化好需要插入的结点,用指针q指向它
    q=(Node *)malloc(sizeof(Node));
    q->data=e;
    q->next=NULL;


    for(int j=0; p&&j<i-1 ; j++) { 

        if(i>ListLength(L)+1)
            return ERROR;   //如果插入的位置大于链表的尾结点后面的一个位置,则返回错误
        p=p->next; 
    }

    //将待插入结点的前后结点与之形成链式关系

    q->next=p->next;    //接后继结点
    p->next=q;            //接前驱结点

    return OK;

    /*        测试:ListInsert(L,4,99);    ListTraverse(L); 在表中元素为5个的情况下        插入位置为1--6均可正确插入,如果插入位置大于等于7,则不插入元素                  */
}

4.单链表的结点删除

        思路:同样是先找到待删除结点,将q指针指向它,并新定义一个指针p对其前驱进行指向,以便后续链接,接好待删除结点的前驱后继后即可释放该结点。以下是实现代码:

//链表元素的删除  //参数需要删除元素的链表,需要删除的结点
Status deleElem(LinkList L,int i) {

    LinkList p,q;   //p指向删除位置的前一结点,q指向需要删除的结点
    p=L;  //p指向头结点


    //首先需要将p移动到删除结点的前一结点
    for(int j=0; p&&j<i-1; j++) {
        p=p->next;
    }

    q=p->next;  //如果所需删除结点存在,则将q指向它,如果不存在返回错误,不执行删除

    if(!q)
        return ERROR;  //判断删除的结点是否存在


    p->next=p->next->next;  //也可以p->next=q->next;但一定要在释放之前将p指向下一结点,否则找不到
    free(q);


    return OK;
    /*    测试:    deleElem(L,1);    ListTraverse(L);假设表中有5个元素,则第1--5个位置的元素正常删除,如果为大于等于6的位置,不删除    */
}

5.单链表的创建

        包括尾插法和头插法,进行单链表的内存分配和结点的添加与赋值。以下是代码实现:

/*头插法创建链表*/ //输入n个元素 ,元素从头开始依次插入

Status CreateList_H(LinkList *L,int n) {
    LinkList p,r;  //p用于指向新结点,r指向头结点

    //建立一个带有头结点的单链表并将指针域置为空
    *L=(LinkList)malloc(sizeof(Node));
    r=*L;
    r->next=NULL;
//结点添加与赋值
    for(int i=n; i>0; i--) {
        p=(Node *)malloc(sizeof(Node));
        printf("请输入该链表的第%d个元素:",i);
        scanf("%d",&p->data);
        p->next=NULL;
        p->next=r->next;
        r->next=p;
    }

    return OK;

    /*测试:printf("请输入头插法创建链表的元素个数:");    scanf("%d",&n);    CreateList_H(&L_H,n);    ListTraverse(L_H);  新建正常    */
}
/*尾插法创建链表*/ //输入n个元素, 元素从尾部依次接入
Status CreateList(LinkList *L,int n) {

    LinkList p,r;//p用于指向新结点(待插入结点),r指向尾部结点

    *L = (LinkList)malloc(sizeof(Node));
    r=*L;  //刚开始时r指向头结点
    r->next=NULL;

    for(int i=0; i<n; i++) {
        p=(Node *)malloc(sizeof(Node));   //注意这里不要忘记将这块内存定义为Node的指针类型
        printf("请输入该链表的第%d个元素:",i+1);
        scanf("%d",&p->data);
        p->next=NULL;
        r->next=p;
        r=r->next;
    }
    r->next=NULL;
    return OK;
}

以下是主函数以及函数声明

Status IniList(LinkList *L);
Status GetElem(LinkList L,int i,ElemType *e);
Status CreateList(LinkList *L,int n);  //尾插法
Status CreateList_H(LinkList *L,int n);//头插法
Status visit(ElemType c);
Status ListTraverse(LinkList L);
bool Empty(LinkList L);
Status DestoryList(LinkList *L);
Status ClearList(LinkList L);
int ListLength(LinkList L);
LinkList searchAD(LinkList L,ElemType e);
int searchNum(LinkList L,ElemType e);
Status ListInsert(LinkList L,int i,ElemType e);
Status deleElem(LinkList L,int i);

int main() {
    LinkList L,L_H;
    int n;
    int a=IniList(&L);     //a用于判断是否成功初始化

    printf("请输入尾插法创建链表的元素个数:");
    scanf("%d",&n);
    CreateList(&L,n);
    ListTraverse(L);
    printf("请输入头插法创建链表的元素个数:");
    scanf("%d",&n);
    CreateList_H(&L_H,n);
    ListTraverse(L_H);
    return 0;
}

补充

        为了更方便的对链表进行输出,新定义了两个子函数实现,代码如下:

//打印元素
Status visit(ElemType c) {
    printf("%d ",c);
    return OK;
}
//遍历链表
Status ListTraverse(LinkList L) {
    LinkList p=L->next;
    printf("链表中所有元素为: ");
    if(!p)
        return ERROR;//如果首元结点不存在返回0

    while(p) {
        visit(p->data);
        p=p->next;
    }
    printf("\n");
    return OK;
}


文章作者: KTpro
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源 KTpro !
  目录