CTDChain<T,sort,equal,pointer> * nodeTemp = NULL;
------------------------------------------------------------------------------
using namespace std;
//双向不循环链表
//sort 指示是否有序
//equal 指示当出现相同节点时的处理方法
//pointer 指示是否是一个指针类型,在析构时必不可少
template <class T,
const int sort = 0,
const CtEqualMethod equal = CtEM_ALLOW,
const int pointer = 1>
--------------------------------------------------------------------------------------
class CTDChain
{
public:
CTDChain * m_pPrev;
CTDChain * m_pNext;
T m_tItem;
protected:
// 向链表头搜索一个节点, 失败时返回空指针,direct指示搜索方向
virtual CTDChain * SearchPrev(T &item)
{
if( TypeComp(m_tItem,item, CtCS_EQ) ) return this;
else if( m_pPrev )
return m_pPrev->CTDChain<T,sort,equal,pointer>:: SearchPrev(item);
else return NULL;
}
// 向链表尾部搜索一个节点, 失败时返回空指针,direct指示搜索方向
virtual CTDChain * SearchNext(T &item)
{
if( TypeComp(m_tItem,item, CtCS_EQ) ) return this;
else if( m_pNext )
return m_pNext->CTDChain<T,sort,equal,pointer>:: SearchNext(item);
else return NULL;
}
//从本节点开始向后查找最小的节点,由排序函调用
virtual CTDChain * SmallestNext()
{
CTDChain<T,sort,equal,pointer> * nodeNext = m_pNext;
CTDChain<T,sort,equal,pointer> * nodeSmall = this;
while( nodeNext )
{
if( nodeSmall->m_tItem > nodeNext->m_tItem ) nodeSmall = nodeNext;
nodeNext = nodeNext->m_pNext;
}
return nodeSmall;
}
public:
CTDChain(): m_pPrev(0), m_pNext(0), m_tItem(0){};
CTDChain(T &item): m_pPrev(0), m_pNext(0), m_tItem(0)
{
CTDChain<T,sort,equal,pointer>:: SetNodeItem(item);
}
//考贝构造函数
CTDChain(CTDChain & node)
{
m_pPrev = node.m_pPrev;
m_pNext = node.m_nNext;
if( pointer )
{
if( !m_tItem ) m_tItem = new T;
*m_tItem = *(node.nodeItem);
}
else m_tItem = node.nodeItem;
}
//析构时自动删除在它后面的节点
~CTDChain()
{
if( pointer && m_tItem ) delete m_tItem;
if( m_pPrev ) m_pPrev->m_pNext = NULL;
if( m_pNext )
{
m_pNext->m_pPrev = NULL;
delete m_pNext;
m_pNext = NULL;
}
}
// 找到链表头节点
CTDChain * Head()
{
if( !m_pPrev ) return this;
else return m_pPrev->CTDChain<T,sort,equal,pointer>:: Head();
}
// 找到链表尾节点
CTDChain * Tail()
{
if( !m_pNext ) return this;
else return m_pNext->CTDChain<T,sort,equal,pointer>:: Tail();
}
//统计链表中节点个数
int NodeNum()
{
int nNum = 0;
CTDChain<T,sort,equal,pointer> * nodeTemp = Head();
while( nodeTemp )
{
nNum++;
nodeTemp = nodeTemp->m_pNext;
}
return nNum;
}
//从链表中移走一个节点
//注意:Pop头节点时,要有指针记录新的头节点
virtual void Pop()
{
if( m_pPrev ) m_pPrev->m_pNext = m_pNext;
if( m_pNext ) m_pNext->m_pPrev = m_pPrev;
m_pPrev = m_pNext = NULL;
}
//设置一个节点的值,替换掉原有的值
virtual int SetNodeItem(const T &item)
{
if( pointer && m_tItem ) delete m_tItem;
m_tItem = item;
return CtRC_REPLACE;
}
int TypeComp(const T &a, const T&b, CtCompartStyle style = CtCS_GT)
{
if( pointer )
{
switch( style )
{
case CtCS_GT: return ((*a) > (*b)); break;
case CtCS_GE: return ((*a) >= (*b)); break;
case CtCS_EQ: return ((*a) == (*b)); break;
case CtCS_NE: return ((*a) != (*b)); break;
case CtCS_LT: return ((*a) < (*b)); break;
case CtCS_LE: return ((*a) <= (*b)); break;
default: return 0;
}
}
else
{
switch( style )
{
case CtCS_GT: return (a > b); break;
case CtCS_GE: return (a >= b); break;
case CtCS_EQ: return (a == b); break;
case CtCS_NE: return (a != b); break;
case CtCS_LT: return (a < b); break;
case CtCS_LE: return (a <= b); break;
default: return 0;
}
}
return 1;
}
// 搜索一个节点, 失败时返回空指针
virtual CTDChain * Search(T &item)
{
CTDChain<T,sort,equal,pointer> * nodeTemp = NULL;
if( TypeComp(m_tItem,item,CtCS_EQ) ) return this;
if( sort )
{//有序
if( TypeComp(item,m_tItem) ) nodeTemp = CTDChain<T,sort,equal,pointer>:: SearchNext(item);
else nodeTemp = CTDChain<T,sort,equal,pointer>:: SearchPrev(item);
}
else
{//无序
nodeTemp = CTDChain<T,sort,equal,pointer>:: SearchPrev(item);
if( !nodeTemp ) nodeTemp = CTDChain<T,sort,equal,pointer>:: SearchNext(item);
}
return nodeTemp;
}
//插入一个节点在本节点之前,node必须是一个单节点
virtual int PushPrev(CTDChain * node)
{
node->m_pNext = this;
node->m_pPrev = m_pPrev;
if( m_pPrev ) m_pPrev->m_pNext = node;
m_pPrev = node;
return CtRC_SUCCESS;
}
//插入一个节点在本节点之后,node必须是一个单节点
virtual int PushNext(CTDChain * node)
{
node->m_pNext = m_pNext;
node->m_pPrev = this;
if( m_pNext ) m_pNext->m_pPrev = node;
m_pNext = node;
return CtRC_SUCCESS;
}
//插入一个单节点
virtual int Push(CTDChain * &node)
{
CTDChain<T,sort,equal,pointer> * nodeTemp;
if( !node ) return CtRC_FAIL;
//判断相同节点
switch( equal )
{
case CtEM_REJECT: //不允许出现相同节点
case CtEM_REPLACE: //替换相同节点
nodeTemp = CTDChain<T,sort,equal,pointer>:: Search(node->m_tItem);
if( NULL == nodeTemp ) break;
if( equal == CtEM_REJECT ) return CtRC_REJECT;
return nodeTemp->CTDChain<T,sort,equal,pointer>:: SetNodeItem(node->m_tItem);
break;
case CtEM_ALLOW: //允许出现相同节点
default:
break;
}
//查找插入点,这里不用考虑相等情况
nodeTemp = this;
if( sort )
{//有序
if( TypeComp(node->m_tItem,nodeTemp->m_tItem) )
{//向表尾查找
while( TypeComp(node->m_tItem,nodeTemp->m_tItem) )
{
if( !nodeTemp->m_pNext ) break;
nodeTemp = nodeTemp->m_pNext;
}
}
else
{//向表头查找
while( TypeComp(nodeTemp->m_tItem,node->m_tItem) )
{
if( !nodeTemp->m_pPrev )
return nodeTemp->CTDChain<T,sort,equal,pointer>:: PushPrev(node);
nodeTemp = nodeTemp->m_pPrev;
}
}
}
else
{//无序
nodeTemp = CTDChain<T,sort,equal,pointer>:: Tail();
}
//在上面还有一个插入点,是插入在链表的头部
return nodeTemp->CTDChain<T,sort,equal,pointer>:: PushNext(node);
}
virtual int Push(T &item)
{
CTDChain<T,sort,equal,pointer> * nodeTemp;
int nRet;
if( NULL == (nodeTemp = new CTDChain<T,sort,equal,pointer>(item)) )
{
RAISE_ERROR("警告:申请内存失败");
return CtRC_FAIL;
}
nRet = CTDChain<T,sort,equal,pointer>:: Push(nodeTemp);
switch( nRet )
{
case CtRC_FAIL: break;
case CtRC_SUCCESS: break;
case CtRC_REPLACE:
if( pointer ) nodeTemp->m_tItem = NULL;
case CtRC_REJECT:
delete nodeTemp;
break;
default: break;
}
return nRet;
}
//插入另外一个链表的所有节点,返回值是插入节点的个数
//插入完成后,原链表被删除
virtual int Merge(CTDChain * &chain)
{
CTDChain<T,sort,equal,pointer> * nodeTemp;
int nRet, nNum = 0;
if( !chain ) return 0;
//先截断原链表
if( chain->m_pPrev )
{
chain->m_pPrev->m_pNext = NULL;
chain->m_pPrev = NULL;
}
//如果允许相同节点且无序
if( equal == CtEM_ALLOW && !sort )
{
nNum = chain->NodeNum();
nodeTemp = CTDChain<T,sort,equal,pointer>:: Tail();
nodeTemp->m_pNext = chain;
chain->m_pPrev = nodeTemp;
}
else
{
nodeTemp = chain;
while( nodeTemp )
{
chain = chain->m_pNext;
nodeTemp->Pop();
nRet = CTDChain<T,sort,equal,pointer>:: Push(nodeTemp);
switch( nRet )
{
case CtRC_SUCCESS: nNum++; break;
case CtRC_REPLACE:
if( pointer ) nodeTemp->m_tItem = NULL;
case CtRC_REJECT:
case CtRC_FAIL:
delete nodeTemp;
break;
default: break;
}
nodeTemp = chain;
}
}
chain = NULL;
return nNum;
}
//排序函数, 它使用T的操作符 '>'
virtual void Sort()
{
CTDChain<T,sort,equal,pointer> * nodeSmall;
CTDChain<T,sort,equal,pointer> * nodeTemp = CTDChain<T,sort,equal,pointer>:: Head();
while( nodeTemp )
{
nodeSmall = nodeTemp->CTDChain<T,sort,equal,pointer>:: SmallestNext();
if( nodeSmall != nodeTemp )
{//最小节点不是当前节点
nodeSmall->CTDChain<T,sort,equal,pointer>:: Pop();
nodeTemp->CTDChain<T,sort,equal,pointer>:: PushPrev(nodeSmall);
}
else nodeTemp = nodeTemp->m_pNext;
}
}
- Redis list实现原理 - 双向循环链表
双向链表 双向表示每个节点知道自己的直接前驱和直接后继,每个节点需要三个域 查找方向可以是从左往右也可以是从右往左,但是要实现从右往左还需要终端节点的地址,所以通常会设计成双向的循环链表; 双向的循环 ...
- Dancing Links 模板
struct dl{ // x: line, y: column struct node{ int c, left, right, up, down; }; vector<node> a; ...
- 通过例子进阶学习C++(七)CMake项目通过模板库实现约瑟夫环
本文是通过例子学习C++的第七篇,通过这个例子可以快速入门c++相关的语法. 1.问题描述 回顾一下约瑟夫环问题:n 个人围坐在一个圆桌周围,现在从第 s 个人开始报数,数到第 m 个人,让他出局:然 ...
- 侯捷STL学习(一)
开始跟着<STL源码剖析>的作者侯捷真人视频,学习STL,了解STL背后的真实故事! 视频链接:侯捷STL 还有很大其他视频需要的留言 第一节:STL版本和重要资源 STL和标准库的区别 ...
- 侯捷STL课程及源码剖析学习1
1.C++标准库和STL C++标准库以header files形式呈现: C++标准库的header files不带后缀名(.h),例如#include <vector> 新式C hea ...
- 侯捷STL学习(一)--顺序容器测试
开始跟着<STL源码剖析>的作者侯捷真人视频,学习STL,了解STL背后的真实故事! 视频链接:侯捷STL 还有很大其他视频需要的留言 第一节:STL版本和重要资源 STL和标准库的区别 ...
- 详解DLX及其应用
什么是DLX? 让我们看看百度百科上的解释:在 计算机科学 中, Dancing Links ,舞蹈链, 也叫 DLX, 是由 Donald Knuth 提出的数据结构,目的是快速实现他的 X算法.X ...
- Josephus环类问题,java实现
写出一个双向的循环链表,弄一个计数器,我定义的是到三的时候,自动删除当前节点,很简单. package Com; import java.util.Scanner; /* * 约瑟夫环问题,有n个人组 ...
- Redis入门指南(第2版) Redis设计思路学习与总结
https://www.qcloud.com/community/article/222 宋增宽,腾讯工程师,16年毕业加入腾讯,从事海量服务后台设计与研发工作,现在负责QQ群后台等项目,喜欢研究技术 ...
随机推荐
- Android中asset文件夹和raw文件夹区别
res/raw和assets的相同点: 1.两者目录下的文件在打包后会原封不动的保存在apk包中,不会被编译成二进制. res/raw和assets的不同点: 1.res/raw中的文件会被映射到R. ...
- [CareerCup] 6.3 Water Jug 水罐问题
6.3 You have a five-quart jug, a three-quart jug, and an unlimited supply of water (but no measuring ...
- java中String类型变量的赋值问题
第一节 String类型的方法参数 运行下面这段代码,其结果是什么? package com.test; public class Example { String str = new String( ...
- 微信小程序全面实战,架构设计 && 躲坑攻略(小程序入门捷径教程)
最近集中开发了两款微信小程序,分别是好奇心日历(每天一条辞典+一个小投票)和好奇心日报(轻量版),直接上图: Paste_Image.png 本文将结合具体的实战经验,主要介绍微信小程序的基础知识.开 ...
- GnuDIP制作动态域名服务器(DDNS Server)_转载http://blog.sina.com.cn/s/blog_4d4c23530100rlfj.html
这个阶段在做DDNS,虽然有dyndns和tzo两个免费的国外的DDNS服务器(支持免费用户注册使用),但是公司需求中要有GnuDIP这种服务.于是只能自己制作DDNS服务器,颇费功夫,于是想把这段记 ...
- Boostrap(2)
网页布局 1.网格布局 网格布局就是把网页分为许多小格子,看起来像table,然后在每个小格子中放我们的内容.当然,我们也可以指定一片区域使用网格系统.网格布局主要是应用在移动设备上的,使用方法如下: ...
- Javascript基础系列之(二)变量
javascript 中变量通过var关键字(variable)来声明的. var school = "beijingyizhong" 也可以通过var 关键字给变量多个值. va ...
- 每天一个linux命令(16):whereis 命令
whereis命令只能用于程序名的搜索,而且只搜索二进制文件(参数-b).man说明文件(参数-m)和源代码文件(参数-s).如果省略参数,则返回所有信息. 和 find相比,whereis查找的速度 ...
- 调研Android平台开发环境的发展演变
Android是Google推出的开源手机操作系统,主要以开发应用为主,要进行Android开发首先得搭建好开发平台.最近在搭建Android的开发环境,发现往往一个小问题都能花费你大半天时间,从刚开 ...
- 我眼中的Android IDE
我作为一个Android小白,首先跟Android打交道的就是它的IDE(Integrated Development Environment,集成开发环境)了. 记得刚开始时是从图书馆借了本Andr ...