数据结构 笔记:双向链表的实现

单链表的另一个缺陷

-单向性

·只能从头结点开始高效访问链表中的数据元素

-缺陷

·如果需要逆向访问单链表中的数据元素将极其低效

双向的线性表

-在“单链表”的结点中增加一个指针pre,用于指向当前结点的前驱结点

双向链表的继承层次结构

DualLinkList的定义

template<typename T>
class DualLinkList : public List<T>
{
protected:
    struct Node :public Object
    {
        T value;
        Node* next;
        Node* pre;
    };

    mutable
//    struct{
//        char reserved[sizeof(T)];
//        Node* next;
//    }
    Node
    m_header;

    Node* position(int i) const
    {
        Node* ret = reinterpret_cast<Node*>(&m_header);

        for(int p=0;p<i;p++)
        {
            ret = ret->next;
        }
        return ret;
    }

    virtual Node* create()
    {
        return new Node();
    }

    virtual void destroy(Node* pn)
    {
        delete pn;
    }

    int m_step;
    Node* m_current;
    int m_length;
public:
    DualLinkList()
    {
        //m_header.reserved("nul");
        m_header.next = NULL;
        m_header.pre = NULL;
        m_step = 1;
        m_current = NULL;
        m_length = 0;
    }
    virtual bool insert(const T& e)
    {
        return insert(m_length,e);
    }

    virtual bool insert(int i,const T& e)
    {
        bool ret = ((0 <= i) && (i <= m_length));

        if(ret)
        {
            Node* node = create();
            if( node != NULL)
            {
                Node *current = position(i);
                Node* next = current->next;
                //Node* pre = current->pre;
                //qDebug() << current;

                node->value = e;

                node->next = current->next;
                current->next = node;
                //qDebug() << "i = " << i <<" , e = " << e << "  " << m_length;
                if(current != reinterpret_cast<Node*>(&m_header))
                {
                    //qDebug() << 1;
                    node->pre = current;
                    next->pre = NULL;
                }
                else
                {
                    //qDebug() << 2;
                    node->pre = NULL;
                }


                if(next != NULL)
                {
                    //qDebug() << 4;
                    next->pre = node;
                }
                //qDebug() << 5;
                m_length++;
            }
            else
            {
                //抛出异常
                //qDebug() << "错错了";
            }
        }
        return ret;
    }
    virtual bool remove(int i)
    {
        bool ret = ((0 <= i) && (i < m_length));

        if(ret)
        {
            Node* current = position(i);

            Node* toDel = current->next;
            Node* next = toDel->next;
            if(m_current == toDel)
            {
                m_current = next;
            }
            current->next = next;

            if(next != NULL)
            {
                next->pre = toDel->pre;
            }

            m_length--;
            destroy(toDel);
        }
        return ret;
    }

    virtual bool set(int i,const T& e)
    {
        bool ret = ((0 <=i ) && (i < m_length));

        if(ret)
        {
            Node* current = position(i);

            current->next->value = e;
        }
        return ret;
    }

    virtual T get(int i) const
    {
        T ret;

        if(get(i,ret))
        {
            return ret;
        }
        else
        {
            //抛出异常
            //qDebug() << "错了";
        }

    }
    virtual bool get(int i, T& e) const
    {
        bool ret = ((0 <=i ) && (i < m_length));

        if(ret)
        {
            Node* current = position(i);
            e = current->next->value;
            //qDebug() << e;
        }
        return ret;
    }
    virtual int length() const
    {
        return m_length;
    }

    int find (const T& e) const
    {
        int ret = -1;
        int i = 0;
        Node* node = m_header.next;

        while(node)
        {
            if(node->value == e)
            {
                ret = i;
                break;
            }
            else
            {
                node = node->next;
                i++;
            }
        }
        return ret;
    }
    virtual void clear()
    {
        while(m_length > 0)
        {
            remove(0);
        }

        m_length = 0;
    }

    virtual bool move(int i,int step = 1)
    {
        bool ret = (0 <= i)&& (i< m_length) && (step >0);

        if(ret)
        {
            m_current = position(i)->next;
            m_step = step;
        }
        return ret;
    }

    virtual bool end()
    {
        return (m_current == NULL);
    }

    virtual T current()
    {
        if(!end())
        {
            return m_current->value;
        }
        else
        {
            //抛出异常
        }
    }

    virtual bool next()
    {
        int i  = 0;
        while((i < m_step) && !end())
        {
            m_current = m_current->next;
            i++;
        }
        return (i == m_step);
    }

    virtual bool pre()
    {
        int i  = 0;
        while((i < m_step) && !end())
        {
            m_current = m_current->pre;
            i++;
        }
        return (i == m_step);
    }

    ~DualLinkList()
    {
        clear();
    }
};

总结:

-双向链表是为了弥补单链表的缺陷而重新设计的

-再概念上,双向链表不是单链表,没有继承关系

-双向链表中的游标能够直接访问当前结点的前驱和后继

-双向链表是线性表概念的最终实现(更贴近理论上的线性表)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值