文章类型: VC&C++
关键词: 单链表,基本操作,插入,删除,交,并,相邻元素,交换
内容摘要: 单链表的基本操作---插入,删除,交,并,相邻元素的交换等

单链表的基本操作---插入,删除,交,并,相邻元素的交换等

2017/10/25 14:36:24    来源:apple    阅读:

这个链表是带有表头的单链表。实现链表的一些规范操作,初始化,插入,删除等。包括两个头文件list.h,fatal.h,库函数list.c,测试函数testlist.c。头文件放的都是函数声明,库函数list.c放的的函数的定义。

头文件list.h

typedef int ElementType;
#ifndef _List_H//如果没有编译过
struct Node;
typedef struct Node *PtrToNode;
typedef PtrToNode List;
typedef PtrToNode Position;
#include<stdbool.h>
List MakeEmpty(List L);
void DeleteList(List L);
bool IsEmpty(List L);
bool IsLast(Position P, List L);
Position Find(ElementType X, List L);
void Delete(ElementType X, List L);
Position FindPrevious(ElementType X, List L);
void Insert(ElementType X, List L, Position P);
Position Header(List L);
Position First(List L);
Position Advance(Position P);
ElementType Retrieve(Position P);
void PrintList(const List L);
void PrintLots(List L, List P);
void SwapWithNext(Position BeforeP, List L);
List IntersectList(List L, List P);
List UnionList(Position L, Position P);
void ReverseList(List L);
#endif // !_List_H

头文件fatal.h

#include<stdio.h>
#include<stdlib.h>
#define Error(Str) FatalError(Str)
#define FatalError(Str) fprintf(stderr,"%s\n",Str),exit(1);

库函数list.c

//引用头文件
#include "list.h"
#include<stdlib.h>
#include "fatal.h"

//结构体定义
struct Node
{
    ElementType Element;
    Position Next;
};

//初始化链表
List MakeEmpty(List L)
{
    if (L != NULL)
        DeleteList(L);//如果链表非空,则删除链表
    L = malloc(sizeof(struct Node));
    if (L == NULL)
        FatalError("Out of memory!");
    L->Next = NULL;
    return L;
}

//删除链表
void DeleteList(List L)
{
    Position P, Temp;
    P = L->Next;
    L->Next = NULL;
    while (P != NULL)
    {
        Temp = P->Next;
        free(P);
        P = Temp;
    }
}

//判断链表是否为空
bool IsEmpty(List L)
{
    return L->Next==NULL;
}
 
//判断当前指针P是否指向链表最后一个元素
bool IsLast(Position P, List L)
{
    return P->Next==NULL;
}

//return Position of X in L;NULL if not found
Position Find(ElementType X, List L)
{
    Position P;
    P = L->Next;
    while (P != NULL && P->Element != X)
        P = P->Next;    
    return P;
}

//删除链表中的元素X,若返回NULL,说明在链表中没找到元素X
void Delete(ElementType X, List L)
{
    Position P, TempCell;
    P = FindPrevious(X, L);
    if (!IsLast(P, L))//当P不是尾针,说明找到了
    {
        TempCell = P->Next;
        P->Next = TempCell->Next;
        free(TempCell);
        TempCell = NULL;
    }

}


//如果返回的P指向最后一个元素,说明没有找到,1==IsLast(P,L)
Position FindPrevious(ElementType X, List L)
{
    Position P;
    P = L;
    while (P->Next != NULL&&P->Next->Element != X)
        P = P->Next;
    return P;
}

//插入元素X到位置P后面
void Insert(ElementType X, List L, Position P)
{
    Position  TmpCell;
    TmpCell = malloc(sizeof(struct Node));
    if (TmpCell == NULL)
        FatalError("Out of Space!!!");
    TmpCell->Element = X;
    TmpCell->Next = P->Next;
    P->Next = TmpCell;
}

//获取链表头
Position Header(List L)
{
    return L;
}

//获取链表第一个元素的位置
Position First(List L)
{
    return L->Next;
}

//获取位置P的下一个位置
Position Advance(Position P)
{
    return P->Next;
}

//提取位置P处结构里面的值
ElementType Retrieve(Position P)
{
    return P->Element;
}

//打印链表
void PrintList(const List L)
{
    Position P=Header(L);
    if (IsEmpty(L))
        printf("Empty list\n");
    else
    {
        do
        {
            P = Advance(P);
            printf("%d ", Retrieve(P));
        } while (!IsLast(P, L));
        printf("\n");
    }
}


//打印链表L中那些由P所指定的位置上的元素。例如P=1,3,4,6,将L
//中的第1,第3,第4,第6个元素打印出来
void PrintLots(List L, List P)
{
    int count = 1;
    Position Lpos, Ppos;
    Lpos = First(L);
    Ppos = First(P);
    while (Lpos != NULL&&Ppos != NULL)
    {
        if ( Ppos->Element == count++)
        {
            printf("%d ", Ppos->Element);
            Ppos = Advance(Ppos);
        }
        Lpos = Advance(Lpos);
    }


}


//通过只调整指针来交换两个相邻的元素,BeforeP是要调换两个元素的前一个指针
void SwapWithNext(Position BeforeP, List L)
{
    Position P, AfterP;
    if (BeforeP != NULL)
    {
        P = Advance(BeforeP);
        if (P != NULL)
        {
            AfterP = Advance(P);
            if (AfterP != NULL)
            {
                P->Next = AfterP->Next;
                BeforeP->Next = AfterP;
                AfterP->Next = P;
            }
        }
    }
}

//求两个链表的交集
List IntersectList(List L1, List L2)
{
    List ResultList;
    Position L1Pos, L2Pos, ResultPos;
    ResultList = MakeEmpty(NULL);
    L1Pos = First(L1);
    L2Pos = First(L2);
    ResultPos = Header(ResultList);
    while (L1Pos!=NULL&&L2Pos!=NULL)
    {
        if (L1Pos->Element < L2Pos->Element)
            L1Pos = Advance(L1Pos);
        else if (L1Pos->Element > L2Pos->Element)
            L2Pos = Advance(L2Pos);
        else
        {
            Insert(L1Pos->Element, ResultList, ResultPos);
            ResultPos= Advance(ResultPos);
            L1Pos = Advance(L1Pos);
            L2Pos = Advance(L2Pos);
        }
    }
    return ResultList;
}

//求两个链表的并集
List UnionList(Position L1, Position L2)
{
    List ResultList;
    ElementType InsertElement;
    Position L1Pos, L2Pos, ResultPos;
    ResultList = MakeEmpty(NULL);
    L1Pos = First(L1);
    L2Pos = First(L2);
    ResultPos = Header(ResultList);
    while (L1Pos != NULL&&L2Pos != NULL)
    {
        if (L1Pos->Element < L2Pos->Element)
        {
            InsertElement = L1Pos->Element;
            L1Pos = Advance(L1Pos);
        }
        else if (L1Pos->Element > L2Pos->Element)
        {
            InsertElement = L2Pos->Element;
            L2Pos = Advance(L2Pos);
        }
        else
        {
            InsertElement = L1Pos->Element;
            L1Pos = Advance(L1Pos);
            L2Pos = Advance(L2Pos);
        }
        Insert(InsertElement, ResultList, ResultPos);
        ResultPos = Advance(ResultPos);
    }
    while (L1Pos != NULL)
    {
        Insert(L1Pos->Element, ResultList, ResultPos);
        ResultPos = Advance(ResultPos);
        L1Pos = Advance(L1Pos);
    }
    while (L2Pos != NULL)
    {
        Insert(L2Pos->Element, ResultList, ResultPos);
        ResultPos = Advance(ResultPos);
        L2Pos = Advance(L2Pos);
    }
    return ResultList;
}

//将链表以O(N)进行反转链表
void ReverseList(List L)
{
    if (IsEmpty(L)||L->Next->Next==NULL)//如果链表是空或者只有一个元素,则不排序
        return;
    Position CurrentPos, PreviousPos, NextPos;
    CurrentPos = L->Next;//当前指针
    PreviousPos = NULL;//上一个指针
    NextPos = L->Next->Next;//探测指针
    while (NextPos!=NULL)//探测指针指向链表尾部时,停止循环
    {
        CurrentPos->Next = PreviousPos;//当前指针指向指向上一个指针
        PreviousPos = CurrentPos;//将上一个指针后移
        CurrentPos = NextPos;//将当前指针后移
        NextPos = Advance(NextPos);//将探测指针后移,直到触到尾指针NULL
    }
    CurrentPos->Next = PreviousPos;//最后一个指针还没有链接前一个指针
    L->Next = CurrentPos;//将头结点链接原来尾节点
}

测试函数testlist.c

#include<stdlib.h>
#include "list.h"
main()
{
    List L,L1;
    Position P,P1;
    int i;
    L = MakeEmpty(NULL);
    P = Header(L);
    PrintList(L);

    L1 = MakeEmpty(NULL);
    P1 = Header(L1);
    PrintList(L1);


    for (i = 0; i < 50; i+=2)
    {
        Insert(i, L, P);
        //PrintList(L);
        P = Advance(P);
    }
    PrintList(L);
    printf("\n");
    for (i = 1; i < 100; i+=3)
    {
        Insert(i, L1, P1);
        //PrintList(L);
        P1 = Advance(P1);
    }
    PrintList(L1);
    printf("\n");
    ReverseList(L1);
    PrintList(IntersectList(L, L1));
    printf("\n");
    PrintList(UnionList(L, L1));
    //PrintLots(L, L1);

    //SwapWithNext(L, L);//换头两个元素

    //for (i = 0; i < 10; i += 2)
    //    Delete(i, L);
    //for (i = 0; i < 10; i++)
    //{
    //    if ((i % 2 == 0) == (Find(i, L) != NULL))
    //        printf("Find fails\n");
    //}
    //printf("Finished deletions\n");
    //PrintList(L);
    DeleteList(L);
    DeleteList(L1);
    return 0;
}
↑ 上一篇文章:VC中的MFC基于CSplitterWnd类的多窗口分割(附实例下载) 关键词:VC,MFC,CSplitterWnd,多窗口,分割,实.. 发布日期:2017/10/19 17:15:55
↓ 下一篇文章:双链表的基本操作---插入,删除,交,并,相邻元素的交换,不相邻元素的交换 关键词:双链表,双向链表,基本操作,插入,删除,交,并,相邻元素.. 发布日期:2017/10/25 14:41:48
相关文章:
双链表的基本操作---插入,删除,交,并,相邻元素的交换,不相邻元素的交换 关键词:双链表,双向链表,基本操作,插入,删除,交,并,相邻元素,交换,不相邻元素的交换 发布日期:2017-10-25 14:41
js操作html的table,包括添加行,添加列,删除行,删除列 关键词:js,html,table,添加行,添加列,删除行,删除列 发布日期:2015-07-28 17:07
C++ 使用delete删除指针 关键词:C++,使用delete删除指针 发布日期:2017-08-30 14:49
相关目录:.NETVC&C++软件开发
我要评论
正在加载评论信息......