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;
        }
    }

C++ 模板双向不循环链表!!的更多相关文章

  1. Redis list实现原理 - 双向循环链表

    双向链表 双向表示每个节点知道自己的直接前驱和直接后继,每个节点需要三个域 查找方向可以是从左往右也可以是从右往左,但是要实现从右往左还需要终端节点的地址,所以通常会设计成双向的循环链表; 双向的循环 ...

  2. Dancing Links 模板

    struct dl{ // x: line, y: column struct node{ int c, left, right, up, down; }; vector<node> a; ...

  3. 通过例子进阶学习C++(七)CMake项目通过模板库实现约瑟夫环

    本文是通过例子学习C++的第七篇,通过这个例子可以快速入门c++相关的语法. 1.问题描述 回顾一下约瑟夫环问题:n 个人围坐在一个圆桌周围,现在从第 s 个人开始报数,数到第 m 个人,让他出局:然 ...

  4. 侯捷STL学习(一)

    开始跟着<STL源码剖析>的作者侯捷真人视频,学习STL,了解STL背后的真实故事! 视频链接:侯捷STL 还有很大其他视频需要的留言 第一节:STL版本和重要资源 STL和标准库的区别 ...

  5. 侯捷STL课程及源码剖析学习1

    1.C++标准库和STL C++标准库以header files形式呈现: C++标准库的header files不带后缀名(.h),例如#include <vector> 新式C hea ...

  6. 侯捷STL学习(一)--顺序容器测试

    开始跟着<STL源码剖析>的作者侯捷真人视频,学习STL,了解STL背后的真实故事! 视频链接:侯捷STL 还有很大其他视频需要的留言 第一节:STL版本和重要资源 STL和标准库的区别 ...

  7. 详解DLX及其应用

    什么是DLX? 让我们看看百度百科上的解释:在 计算机科学 中, Dancing Links ,舞蹈链, 也叫 DLX, 是由 Donald Knuth 提出的数据结构,目的是快速实现他的 X算法.X ...

  8. Josephus环类问题,java实现

    写出一个双向的循环链表,弄一个计数器,我定义的是到三的时候,自动删除当前节点,很简单. package Com; import java.util.Scanner; /* * 约瑟夫环问题,有n个人组 ...

  9. Redis入门指南(第2版) Redis设计思路学习与总结

    https://www.qcloud.com/community/article/222 宋增宽,腾讯工程师,16年毕业加入腾讯,从事海量服务后台设计与研发工作,现在负责QQ群后台等项目,喜欢研究技术 ...

随机推荐

  1. 一个因为粗心的Bug

    /** * 数据绑定,分页显示 */ private void updataMenu(final EditText search) { if(listwz==null) { return; } pag ...

  2. [CareerCup] 9.9 Eight Queens 八皇后问题

    9.9 Write an algorithm to print all ways of arranging eight queens on an 8x8 chess board so that non ...

  3. Linux第六次学习笔记

    存储器层次结构 存储器系统是一个具有不同容量.成本和访问时间的存储设备的层次结构. CPU寄存器保存着最常用的数据. 主存储器(简称主存)暂时存放存储在容量较大的.慢速磁盘上的数据. 高速缓存存储器作 ...

  4. 20155301-滕树晨 第二次随笔作业--从现有技能获取的经验应用于JAVA中

    第二次随笔--从现有技能获取的经验应用于JAVA中 你有什么技能比大多人(超过90%以上)更好? 这个想了半天,有一个是我乒乓球还是比较擅长的,在学校里可能比百分之90的人要强,在外面肯定是不如了.再 ...

  5. HoloLens开发手记 - Unity之摄像头篇

    当你穿戴好HoloLens后,你就会处在全息应用世界的中心.当你的项目开启了"Virtual Reality Support"选项并选中了"Windows Hologra ...

  6. 用Wireshark抓包分析超过70秒的请求

    超过70秒的请求是通过分析IIS日志发现的: 10.159.63.104是SLB的内网IP. 通过Wireshark抓包分析请求是9:22:21收到的(tcp.stream eq 23080): 09 ...

  7. RVM 解决 Ruby 的版本问题

    RVM 是一个命令行工具,可以提供一个便捷的多版本 Ruby 环境的管理和切换. RVM 的官网是 https://rvm.io/. 如果你打算学习 Ruby / Rails, RVM 是必不可少的工 ...

  8. #Linux学习笔记# 自定义shell终端提示符

    我使用的Linux发行版是LinuxMint 17.2 Rafaela,默认情况下Terminal中的shell提示包括了用户名.主机名.当前目录(绝对路径)和提示符.这样会导致当进入一个比较深的目录 ...

  9. sublime2的一些基本常用的操作

    1.全局搜 ctrl shift f 如果你的快捷键有冲突的话,那么你在find的菜单中有find in file这个中找.

  10. android之网络操作(1)

    一.网络操作 在网络操作中JAVA已经为我提供了一套API用来进行网络操作,在android开发中我们仍可以这套API来做开发.下面通过一个简单的例子来了解一下android下的网络操作. 点击图中下 ...