在构造函数中分配双向链表指针

时间:2013-04-23 21:25:32

标签: c++ list pointers constructor

我正在尝试为双链表创建一些节点并打印出来。所以我创建了我的dnode类:

template <typename T>
class dnode
{
    public:
        T nodeValue;
        dnode<T> *prev;
        dnode<T> *next;

        dnode() : prev(this), next(this) {}

        dnode(const T& item, dnode<T> *prevNode = NULL, dnode<T> *nextNode = NULL) :
            nodeValue(item), prev(prevNode), next(nextNode) {}

};

然后我有了writeList函数:

template <typename T>
void writeDLinkedList(dnode<T>* header, const string& seperator = " ")
{
    dnode<T> *p = header->next;

    while (p != header)
    {
        cout << p->nodeValue << seperator;
        p = p->next;
    }

    cout << endl << endl;
}

在main中,我创建了一个头指针和两个节点,使用constuctor分配循环列表中的上一个和下一个节点:

dnode<int> *header, *one, *two;

header = new dnode<int>(0, two, one);
one = new dnode<int> (10, header, two);
two = new dnode<int> (25, one, header);

writeDLinkedList(header);

当我调用writeDLinkedList时,我遇到了分段错误。我对此感到困惑,所以我最终尝试单独输出每个节点值以查看指针是否正常工作。事实证明他们不是。相反,我必须这样做才能使打印功能正常工作:

header = new dnode<int>;

one = new dnode<int> (10);
two = new dnode<int> (25);
header->next = one;
one->next = two;
two->next = header;

writeDLinkedList(header);

我想知道为什么我的构造函数不按预期方式工作。它是初始化列表吗?

3 个答案:

答案 0 :(得分:1)

你的构造函数正在运行。问题是您在给定值之前使用变量。

dnode<int> *header, *one, *two; 
// one and two have undefined values at this point

header = new dnode<int>(0, two, one);
// so undefined values get put into header->next and header->prev

在双向链表中,您有指针周期,节点A指向节点B,指向节点A.根据定义,指针周期不能仅在构造函数中创建。因为必须首先创建节点A或节点B 。首先创建的是无法创建指向另一个节点,因为其他节点尚不存在。

双重链表比你想象的要复杂一点。

答案 1 :(得分:0)

您没有正确管理节点。试试这个:

template <typename T>
class dnode
{
    public:
        T nodeValue;
        dnode<T> *prev;
        dnode<T> *next;

        dnode() : prev(NULL), next(NULL) {}

        dnode(const T& item, dnode<T> *prevNode = NULL) :
            nodeValue(item), prev(prevNode), next(NULL)
        {
            if (prev)
            {
                if (prev->next)
                    prev->next->prev = this;
                prev->next = this;
            }
        }

        ~dnode()
        {
            if (prev)
                prev->next = next;

            if (next)
                next->prev = prev;
        }
};

template <typename T>
void writeDLinkedList(dnode<T>* header, const string& seperator = " ")
{
    dnode<T> *p = header;

    while (p != NULL)
    {
        cout << p->nodeValue << seperator;
        p = p->next;
    }

    cout << endl << endl;
}

dnode<int> *header, *one, *two;

header = new dnode<int>(0);
one = new dnode<int> (10, header);
two = new dnode<int> (25, one);

writeDLinkedList(header);

答案 2 :(得分:0)

我认为你正在努力创建双重循环列表。您可以先创建单个节点,然后再连接它们。在连接两个节点后的双循环列表中,您必须将最后一个节点连接到第一个节点以关闭圆圈。 你可能想尝试类似的东西。我已经为int做了它,但它可以很容易地放入模板中。

#include<iostream>

class dnode
{

public:

    dnode(int a):prev(this), next(this), nodeValue(a)
     {}
    void connectNode(dnode *newNode)
     {
        if(newNode != NULL)
        {
            dnode*head = this->next;
            this->next = newNode;
            newNode->prev = this;
            newNode->next = head;
            head->prev = newNode;
        }
     }

    void writeDNode()
     {
     dnode *p = this->next;
        while(p != this)
         {
            std::cout<<"Element "<<p->nodeValue<<std::endl;
            p = p->next;
         }
     }
private:
    int nodeValue;
    dnode *prev;
    dnode *next;

 };


int main(int argc, char*argv[])
 {
    dnode* header = new dnode(-1);
    dnode *one = new dnode(23);
    dnode *two = new dnode(45);
    dnode *three = new dnode(67);
    header->connectNode(one);
    one->connectNode(two);
    two->connectNode(three);
    header->writeDNode();
    std::getchar();



 }