所实现的循环单链表的结构例如以下图所看到的:

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">

循环单链表的实现,和上一篇文章单链表的实现大致同样点击打开链接,略有差别:

1:循环推断的条件不再是s == NULL或者s->next == NULL,而是他们是否等于头指针。2: 断开链表时的处理,尾节点的next不是NULL,而是指向头结点

详细细节參考上一篇文章

头文件:SCList.h

#ifndef SCLIST_H
#define SCLIST_H #include<iostream>
#include<cassert>
using namespace std; typedef enum{ FALSE, TRUE }Status; template<class Type>
class List; template<class Type>
class ListNode
{
friend class List<Type>;
private:
Type data;
ListNode *next;
public:
ListNode() :data(Type()), next(NULL){}
ListNode(Type d, ListNode<Type> *n = NULL) :data(d), next(n){}
void SetData(Type d){ data = d; }
Type GetData()const{ return data; }
~ListNode(){}
}; template<class Type>
class List
{
private:
ListNode<Type> *first;
ListNode<Type> *last;
size_t size;
public:
List()
{
ListNode<Type> *s = new ListNode<Type>;
assert(s != NULL);
first = last = s;
last->next = first;
}
~List()
{
destory();
}
Status push_back(const Type &x)
{
ListNode<Type> *s = new ListNode<Type>(x);
if (s == NULL)
return FALSE;
last->next = s;
last = s;
last->next = first;
size++;
return TRUE;
}
void show_list()
{
ListNode<Type> *s = first->next;
while (s != first)
{
cout << s->data << "->";
s = s->next;
}
cout << "Nul." << endl;
}
Status push_front(const Type &x)
{
ListNode<Type> *s = new ListNode<Type>(x);
if (s == NULL)
return FALSE;
s->next = first->next;
first->next = s;
if (size == 0)
{
last = s;
last->next = first;//---------------->能够省略,think
}
size++;
return TRUE;
}
Status pop_back()
{
if (size == 0)//
{
cout << "循环单链表已空,无法尾删" << endl;
return FALSE;
}
ListNode<Type> *s = first;
while (s->next != last)
s = s->next;
delete last;
last = s;
last->next = first;
size--;
return TRUE;
}
Status pop_front()
{
if (size == 0)//
{
cout << "循环单链表已空。无法头删" << endl;
return FALSE;
}
ListNode<Type> *s = first->next;
first->next = s->next;
delete s;
if (size == 1)
{
last = first;
last->next = first;//能够省略-------------->
}
size--;
return TRUE;
}
Status insert_val(const Type &x)
{
ListNode<Type> *s = first;
while (s->next != first && s->next->data < x)
s = s->next;
if (s->next == first)
push_back(x);
else
{
ListNode<Type> *p = new ListNode<Type>(x);
assert(p != NULL);
p->next = s->next;
s->next = p;
size++;
}
return TRUE;
}
ListNode<Type>* find(const Type &x)
{
if (size == 0)
return NULL;
ListNode<Type> *s = first->next;
while (s != first)
{
if (s->data == x)
return s;
s = s->next;
}
return NULL;
}
Status delete_val(const Type &x)
{
ListNode<Type> *s = find(x);
if (s == NULL)
{
cout << "该元素不存在,无法删除" << endl;
return FALSE;
}
if (s == last)
{
pop_back();
}
else
{
ListNode<Type> *p = s->next;
s->data = p->data;
s->next = p->next;
if (p == last)//------------------->>注意
{
last = s;
}
delete p;
size--;
}
return TRUE;
}
//从第一个节点处断开,一分为二,头结点和第一个节点成为一个单独的循环单链表
//将剩余的节点依次按值插入该链表
void sort()
{
if (size == 0 || size == 1)
return;
ListNode<Type> *s = first->next;
ListNode<Type> *p = s->next;
last = s;
last->next = first;
while (p != first)
{
s = p;
p = p->next;
//insert_val(s->data);
//delete s;
ListNode<Type> *q = first;
while (q->next != NULL && q->next->data < s->data)
{
q = q->next;
}
if (q->next == NULL)
{
last->next = s;
last = s;
last->next = first;
}
else
{
s->next = q->next;
q->next = s;
}
}
}
//从第一个节点处断开,一分为二,头结点和第一个节点成为一个单独的循环单链表
//将剩余的节点依次进行头插
void reserve()
{
if (size == 0 || size == 1)
return;
ListNode<Type> *s = first->next;
ListNode<Type> *p = s->next;
last = s;
last->next = first;
while (p != first)
{
s = p;
p = p->next;
s->next = first->next;
first->next = s;
}
}
size_t lenth()
{
return size;
}
void clear()
{
if (size == 0)
return;
ListNode<Type> *s = first->next;
while (s != first)
{
if (size == 1)
{
last = first;
last->next = first;
}
else
{
first->next = s->next;
}
delete s;
size--;
s = first->next;
}
}
ListNode<Type>* next(ListNode<Type> *s)
{
if (s == last)//最后一个节点没有后继
return NULL;
else
return s->next;
}
ListNode<Type>* prio(ListNode<Type> *s)
{
if (s == first->next)//第一个节点没有前驱
return NULL;
ListNode<Type> *p = first;
while (p->next != s)
{
p = p->next;
}
return p;
}
void destory()
{
clear();
delete first;
first = last = NULL;
}
};
#endif

測试文件:main.cpp

#include"SCList.h"

int main()
{
List<int> mylist;
int item;
int n;
int select = 1;
//ListNode<int> *p;
while (select)
{
cout << "*************************************** *" << endl;
cout << "*[1] push_back [2] push_front *" << endl;
cout << "*[3] show_list [4] pop_back *" << endl;
cout << "*[5] pop_front [6] insert_val *" << endl;
cout << "*[7] lenth [8] find *" << endl;
cout << "*[9] merge [10] delete_val*" << endl;
cout << "*[11] sort [12] reserve *" << endl;
cout << "*[13] next [14] clear *" << endl;
cout << "*[15] prio [0] quit_system*" << endl;
cout << "请选择:>";
cin >> select;
switch (select)
{
case 1:
cout << "请输入要插入的元素(-1结束):>";
while (cin >> item, item != -1)
{
mylist.push_back(item);
}
break;
case 2:
cout << "请输入要插入的元素(-1结束):>";
while (cin >> item, item != -1)
{
mylist.push_front(item);
}
break;
case 3:
mylist.show_list();
break;
case 4:
mylist.pop_back();
break;
case 5:
mylist.pop_front();
break;
case 6:
cout << "请输入要插入的元素:";
cin >> item;
mylist.insert_val(item);
break;
case 7:
cout << "长度为:" << mylist.lenth() << endl;
break;
case 8:
cout << "请输入要查找的元素:";
cin >> item;
if (mylist.find(item))
cout << "it's found" << endl;
else
cout << "it's not exist" << endl;
break;
case 9:
cout << "请输入要删除的位置:";
cin >> n;
//mylist.delete_pos(n,item);
break;
case 10:
cout << "请输入要删除的元素:";
cin >> item;
mylist.delete_val(item);
break;
case 11:
mylist.sort();
break;
case 12:
mylist.reserve();
break;
case 13:
cout << "请输入要查找后继的元素:";
cin >> item;
//p = mylist.next(item);
//if (p != NULL)
// cout << p->GetData() << endl;
break;
case 14:
mylist.clear();
break;
default:
break;
}
}
system("pause");
return 0;
}

【c++版数据结构】之循环单链表的实现(带头结点以及尾节点)的更多相关文章

  1. C实现头插法和尾插法来构建单链表(不带头结点)

    链表的构建事实上也就是不断插入节点的过程.而节点的插入能够分为头插法和尾插法. 头插法就是在头结点后插入该节点,始终把该节点作为第一个节点.尾插法就是在链表的最后一个节点处插入元素,作为最后一个节点. ...

  2. *单链表[递归&不带头结点]

    不带头结点的单链表,递归法比较简明!(必背!) 单链表的结构: typedef struct node{ int data; struct node *next; }*List,Node; 创建第一种 ...

  3. C语言实现单链表(不带头结点)节点的插入

    对单链表进行增删改查是最主要的操作.我在上一篇博客<C语言实现链表节点的删除>实现了删除单链表中的某个节点. 这里我们要来实如今某个位置插入节点.演示样例代码上传至https://gith ...

  4. 简单约瑟夫环的循环单链表实现(C++)

    刚刚接触C++以及数据结构,今天做了第一次尝试用C++和数据结构解决问题,问题是基于约瑟夫环问题的简单版. 先来看看约瑟夫环问题的介绍: 约瑟夫环是一个数学的应用问题:已知n个人(以编号1,2,3.. ...

  5. 数据结构——Java实现单链表

    一.分析 单链表是一种链式存取的数据结构,用一组地址任意的存储单元存放线性表中的数据元素.链表中的数据是以结点来表示的,每个结点由元素和指针构成.在Java中,我们可以将单链表定义成一个类,单链表的基 ...

  6. C代码实现非循环单链表

    C代码实现非循环单链表, 直接上代码. # include <stdio.h> # include <stdlib.h> # include <malloc.h> ...

  7. PTA 循环单链表区间删除 (15 分)

    本题要求实现带头结点的循环单链表的创建和单链表的区间删除.L是一个带头结点的循环单链表,函数ListCreate_CL用于创建一个循环单链表,函数ListDelete_CL用于删除取值大于min小于m ...

  8. C语言版本:循环单链表的实现

    SClist.h #ifndef __SCLIST_H__ #define __SCLIST_H__ #include<cstdio> #include<malloc.h> # ...

  9. c语言循环单链表

    /************************************************************************* > File Name: singleLin ...

随机推荐

  1. Python 33(2)进程理论

    一:什么是进程         进程指的是一个正在进行 / 运行的程序,进程是用来描述程序执行过程的虚拟概念 进程vs程序 程序:一堆代码 进程:程序的执行的过程 进程的概念起源于操作系统,进程是操作 ...

  2. Python 29 异常处理, 元类

    所学内容 异常处理(常用) AttributeError ··························  试图访问一个对象没有的树形,比如foo.x,但是foo没有属性xIOError ··· ...

  3. Android框架式编程之Android Architecture Components

    1. 当前Android开发面临的问题 Android开发不同于传统的桌面程序开发,桌面程序一般都有唯一的快捷方式入口,并且常作为单进程存在:而一个典型的Android应用通常由多个应用组件构成,包括 ...

  4. Zeppelin0.6.2之shiro安全配置 初探

    0.序 默认情况下,Zeppelin安装好并且配置完zeppelin-site.xml和zeppelin-env.sh后,我们进入的模式,从右上角就能看出来是anonymous模式,这种模式下会看见所 ...

  5. 通过学习Date和Calendar时写的日历

    package com.etc.util; import java.util.Calendar; import java.util.Scanner; public class Calendar2 { ...

  6. python练习--1、简易登录接口

    function:用于实现简单的用户登录接口. description:用户信息存储于字典中,增加用户需手动写入代码中.最多可以尝试登录9次,首次使用会先在程序目录下创建锁文件.若在用户栏输入quit ...

  7. Android 使用SQLite存储以及读取Drawable对象

    在进行Android开发过程中,我们经常会接触到Drawable对象,那么,若要使用数据库来进行存储及读取,该如何实现? 一.存储 //第一步,将Drawable对象转化为Bitmap对象 Bitma ...

  8. springboot + sharding-jdbc 学习

    官网地址:http://shardingsphere.io/document/current/cn/overview/ sharding-jdbc事务:https://blog.csdn.net/ya ...

  9. USB 接口探测分类

    USB 接口探测分类 SDP (Standand Downstream Port) 标准下行接口 标准USB都支持的接口 这种端口的D+和D-线上具有15kΩ下拉电阻.限流值如上讨论:挂起时为2.5m ...

  10. Spark on Yarn集群搭建

    软件环境: linux系统: CentOS6.7 Hadoop版本: 2.6.5 zookeeper版本: 3.4.8 主机配置: 一共m1, m2, m3这五部机, 每部主机的用户名都为centos ...