循环链表

✍ dations ◷ 2024-12-23 14:39:22 #数据结构

循环链表是一种链式存储结构,它的最后一个结点指向头结点,形成一个环。因此,从循环链表中的任何一个结点出发都能找到任何其他结点。循环链表的操作和单链表的操作基本一致,差别仅仅在于算法中的循环条件有所不同。

存储结构和单链表相同。

// 设立尾指针的单循环链表的12个基本操作void InitList(LinkList *L) { // 操作结果:构造一个空的线性表L    *L = (LinkList)malloc(sizeof(struct LNode)); // 产生头结点,并使L指向此头结点    if (!*L) // 存储分配失败        exit(OVERFLOW);    (*L)->next = *L; // 指针域指向头结点}void DestroyList(LinkList *L) { // 操作结果:销毁线性表L    LinkList q, p = (*L)->next; // p指向头结点    while (p != *L) { // 没到表尾        q = p->next;        free(p);        p = q;    }    free(*L);    *L = NULL;}void ClearList(LinkList *L) /* 改变L */ { // 初始条件:线性表L已存在。操作结果:将L重置为空表    LinkList p, q;    *L = (*L)->next; // L指向头结点    p = (*L)->next; // p指向第一个结点    while (p != *L) { // 没到表尾        q = p->next;        free(p);        p = q;    }    (*L)->next = *L; // 头结点指针域指向自身}Status ListEmpty(LinkList L) { // 初始条件:线性表L已存在。操作结果:若L为空表,则返回TRUE,否则返回FALSE    if (L->next == L) // 空        return TRUE;    else        return FALSE;}int ListLength(LinkList L) { // 初始条件:L已存在。操作结果:返回L中数据元素个数    int i = 0;    LinkList p = L->next; // p指向头结点    while (p != L) { // 没到表尾        i++;        p = p->next;    }    return i;}Status GetElem(LinkList L, int i, ElemType *e) { // 当第i个元素存在时,其值赋给e并返回OK,否则返回ERROR    int j = 1; // 初始化,j为计数器    LinkList p = L->next->next; // p指向第一个结点    if (i <= 0 || i > ListLength(L)) // 第i个元素不存在        return ERROR;    while (j < i) { // 顺指针向后查找,直到p指向第i个元素        p = p->next;        j++;    }    *e = p->data; // 取第i个元素    return OK;}int LocateElem(LinkList L, ElemType e, Status(*compare)(ElemType, ElemType)) { // 初始条件:线性表L已存在,compare()是数据元素判定函数    // 操作结果:返回L中第1个与e满足关系compare()的数据元素的位序。    //           若这样的数据元素不存在,则返回值为0    int i = 0;    LinkList p = L->next->next; // p指向第一个结点    while (p != L->next) {        i++;        if (compare(p->data, e)) // 满足关系            return i;        p = p->next;    }    return 0;}Status PriorElem(LinkList L, ElemType cur_e, ElemType *pre_e) { // 初始条件:线性表L已存在    // 操作结果:若cur_e是L的数据元素,且不是第一个,则用pre_e返回它的前驱,    //           否则操作失败,pre_e无定义    LinkList q, p = L->next->next; // p指向第一个结点    q = p->next;    while (q != L->next) { // p没到表尾        if (q->data == cur_e) {            *pre_e = p->data;            return TRUE;        }        p = q;        q = q->next;    }    return FALSE; // 操作失败}Status NextElem(LinkList L, ElemType cur_e, ElemType *next_e) { // 初始条件:线性表L已存在    // 操作结果:若cur_e是L的数据元素,且不是最后一个,则用next_e返回它的后继,    //           否则操作失败,next_e无定义    LinkList p = L->next->next; // p指向第一个结点    while (p != L) { // p没到表尾        if (p->data == cur_e) {            *next_e = p->next->data;            return TRUE;        }        p = p->next;    }    return FALSE; // 操作失败}Status ListInsert(LinkList *L, int i, ElemType e) /* 改变L */ { // 在L的第i个位置之前插入元素e    LinkList p = (*L)->next, s; // p指向头结点    int j = 0;    if (i <= 0 || i > ListLength(*L) + 1) // 无法在第i个元素之前插入        return ERROR;    while (j < i - 1) { // 寻找第i-1个结点        p = p->next;        j++;    }    s = (LinkList)malloc(sizeof(struct LNode)); // 生成新结点    s->data = e; // 插入L中    s->next = p->next;    p->next = s;    if (p == *L) // 改变尾结点        *L = s;    return OK;}Status ListDelete(LinkList *L, int i, ElemType *e) /* 改变L */ { // 删除L的第i个元素,并由e返回其值    LinkList p = (*L)->next, q; // p指向头结点    int j = 0;    if (i <= 0 || i > ListLength(*L)) // 第i个元素不存在        return ERROR;    while (j < i - 1) { // 寻找第i-1个结点        p = p->next;        j++;    }    q = p->next; // q指向待删除结点    p->next = q->next;    *e = q->data;    if (*L == q) // 删除的是表尾元素        *L = p;    free(q); // 释放待删除结点    return OK;}void ListTraverse(LinkList L, void(*vi)(ElemType)) { // 初始条件:L已存在。操作结果:依次对L的每个数据元素调用函数vi()    LinkList p = L->next->next; // p指向首元结点    while (p != L->next) { // p不指向头结点        vi(p->data);        p = p->next;    }    printf("\n");}

双向循环链表

循环链表的应用问题

Josephu问题:据说著名犹太历史学家 Josephus有过以下的故事:在罗马人占领乔塔帕特后,39 个犹太人与Josephus及他的朋友躲到一个洞中,39个犹太人决定宁愿死也不要被敌人找到,于是决定了一个自杀方式,41个人排成一个圆圈,由第1个人开始报数,每报数到第3人该人就必须自杀,然后再由下一个重新报数,直到所有人都自杀身亡为止。然而Josephus 和他的朋友并不想遵从,Josephus要他的朋友先假装遵从,他将朋友与自己安排在第16个与第31个位置,于是逃过了这场死亡游戏。如何用循环链表来求解Josephu问题?

相关

  • 白铅矿白铅矿是一种矿物名。晶体常呈假六方双堆状、板状及棒状,集合体为粒状或致密块状。产于铅锌硫化物矿床的氧化带,通常是有方铅矿氧化成铅矾,再由铅矾受碳酸溶液作用而成的表生矿
  • 层孔虫层孔虫(Stromatoporoids)是一类古老的海生无脊椎动物,生存于寒武纪至白垩纪,在志留纪及泥盆纪最为繁盛,常形成礁体,石炭纪至三叠纪无化石记录,中生代仅有少数代表,现已完全灭绝。于1
  • 宋师襄宋师襄(?-1643年),字一衷,陕西耀州城内人,明朝政治人物。万历四十四年(1616年)丙辰科进士。初任南乐县知县。天启年间官监察御史。屡次上疏,被革职。崇祯帝继位,复起为太仆寺少卿,以太常
  • 大元外国语高等学校大元外国语高等学校(대원외국어고등학교/Daewon Foreign Language High School)是一所位于韩国首尔特别市广津区的外国语高等学校。该校由CJ集团总裁李完熙于1984年创立,与大一
  • 妖怪传 猫目小僧‘妖怪伝 猫目小僧’(ようかいでん ねこめこぞう)是1976年4月1日至同年9月30日在东京12チャンネル(现・テレビ东京)放送的日本电视动画。放送时间毎周木曜19:00 - 19:30(JST)。全2
  • 阿勒娜阿勒娜(印尼语:Alena,1981年9月9日-),本名吴慧丽,印尼语名卡罗琳·古纳万(印尼语:Caroline Gunawan),是印度尼西亚女歌手,出身于东爪哇玛琅,印尼华侨,能操英、印尼、爪哇、华语,推出两张专
  • 哈维·贝内特哈维·贝内特(英文:Harve Bennett,1930年8月17日-2015年2月25日),生名哈维·贝内特·菲施曼(英文:Harve Bennett Fischman),于1930年8月17日出生于伊利诺伊州芝加哥,是美国的一位电视、
  • 柯蓝柯蓝(1972年7月25日-),原名锺好好,湖南省岳阳市平江县人,中国大陆女性节目主持人、演员、模特儿,是中华人民共和国开国上将钟期光的孙女。1994年加入卫视音乐台,成为第一代亚洲VJ。1
  • 内湖保护区守护联盟台北市内湖保护区守护联盟于2011年8月成立,是以内湖区大湖里、秀湖里的居民为主的民间社团,以推展内湖保护区守护运动、维护内湖保护区应有之数量、功能与正常使用为宗旨。司
  • 伊格 (城镇)坐标:45°58′9.09″N 14°32′21.72″E / 45.9691917°N 14.5393667°E / 45.9691917; 14.5393667伊格(斯洛文尼亚语:Ig),是斯洛文尼亚的城镇,位于该国中部,处于内卡尼奥拉地区,是