在不使用动态内存分配的情况下,在c中执行单链路列表的问题

问题描述 投票:-1回答:1

我正在用'c'编写单链路列表的程序,没有使用动态内存分配。但它正在进入无限循环。 整个程序。

#include <stdio.h>

struct SingleLinkedList
{
    int data;
    struct SingleLinkedList* next;
};

typedef struct SingleLinkedList sll;

void insertAtStart(sll** head, int data)
{
    sll newNode = { data, NULL };
    newNode.data = data;
    if (*head == NULL)
    {
        *head = &newNode;
        return;
    }
    newNode.next = *head;
    *head = &newNode;
}

void insertAtEnd(sll** head, int data)
{
    sll newNode = { data, NULL };
    newNode.data = data;
    if (*head == NULL)
    {
        *head = &newNode;
        return;
    }
    sll* node = *head;
    while (node -> next != NULL)
    {
        node = node -> next;
    }
    node -> next = &newNode;
}

void insertAfterNode(sll **head, int data)
{
    int nodeNum, count = 1;
    printf("\nEnter the node number to insert after: ");
    scanf("%d", &nodeNum);
    if (*head == NULL)
    {
        printf("\nThe List is empty!\n");
        return;
    }
    sll *prevNode = *head;
    while (count < nodeNum && prevNode -> next != NULL)
    {
        prevNode = prevNode -> next;
        count++;
    }
    if (count < nodeNum)
    {
        printf("\nThere are only %d nodes in the list\n", count);
        return;
    }

    sll newNode = { data, NULL };
    newNode.next = prevNode -> next;
    prevNode -> next = &newNode;
}

int choiceSelection()
{
    int choice;
    printf("\nSelect an Option:\n");
    printf("1. Insert At Beginning\n");
    printf("2. Insert At Last\n");
    printf("3. Insert After Certain Node\n");
    printf("4. Print all nodes\n");
    printf("5. Exit\n");
    scanf("%d", &choice);
    return choice;
}

int dataEntry()
{
    int data;
    printf("\nEnter the data: ");
    scanf("%d", &data);
    return data;
}

void print(sll* node)
{
    int count = 1;
    while(node != NULL)
    {
        printf("\n----------------%d----------------\n", count);
        printf("Data: %d", node -> data);
        printf("\tAddress: %p", node);
        printf("\tNext: %p\n", node -> next);
        node = node -> next;
        count++;
    }
}

int main()
{
    sll *head = NULL;
    enum option {
        InsertAtStart = 1,
        InsertAtEnd = 2,
        InsertAfterNode = 3,
        Print = 4,
        Exit = 5,
    } choice;

    while (choice != Exit)
    {
        choice = choiceSelection();
        switch (choice)
        {
            case InsertAtStart:
                insertAtStart(&head, dataEntry());
                break;
            case InsertAtEnd:
                insertAtEnd(&head, dataEntry());
                break;
            case InsertAfterNode:
                insertAfterNode(&head, dataEntry());
                break;
            case Print:
                print(head);
                break;
            case Exit:
                break;

            default:
                printf("\nIncorrect Choice..Please choose among 1, 2, 3, 4, 5\n");
                break;
        }
    }
    printf("\nExiting!");
    return 0;
}

输出是:

Select an Option:
1. Insert At Beginning
2. Insert At Last
3. Insert After Certain Node
4. Print all nodes
5. Exit
1

Enter the data: 2

Select an Option:
1. Insert At Beginning
2. Insert At Last
3. Insert After Certain Node
4. Print all nodes
5. Exit
4

----------------1----------------
Data: 0 Address: 0x7ffe2e94c0c0 Next: 0x7ffe2e94c0c0

----------------2----------------
Data: 0 Address: 0x7ffe2e94c0c0 Next: 0x7ffe2e94c0c0

----------------3----------------
Data: 0 Address: 0x7ffe2e94c0c0 Next: 0x7ffe2e94c0c0

----------------4----------------
Data: 0 Address: 0x7ffe2e94c0c0 Next: 0x7ffe2e94c0c0

----------------5----------------
Data: 0 Address: 0x7ffe2e94c0c0 Next: 0x7ffe2e94c0c0

----------------6----------------
Data: 0 Address: 0x7ffe2e94c0c0 Next: 0x7ffe2e94c0c0

----------------7----------------
Data: 0 Address: 0x7ffe2e94c0c0 Next: 0x7ffe2e94c0c0

----------------8----------------
Data: 0 Address: 0x7ffe2e94c0c0 Next: 0x7ffe2e94c0c0

----------------9----------------
Data: 0 Address: 0x7ffe2e94c0c0 Next: 0x7ffe2e94c0c0

----------------10----------------
Data: 0 Address: 0x7ffe2e94c0c0 Next: 0x7ffe2e94c0c0

----------------11----------------
Data: 0 Address: 0x7ffe2e94c0c0 Next: 0x7ffe2e94c0c0

----------------12----------------
Data: 0 Address: 0x7ffe2e94c0c0 Next: 0x7ffe2e94c0c0
--------^C

需要手动终止 谁能告诉我问题出在哪里?还是不使用动态内存分配就不行?

c data-structures memory-management linked-list structure
1个回答
2
投票

没有某种动态分配是不可能的。如果你不想使用 malloc你可以使用你自己的分配领域的实现,但你不能使用悬空指针,你的实现就是这样的。

但是你不能使用悬空指针,这正是你的实现所做的。

void insertAtStart(sll** head, int data)
{
    sll newNode = { data, NULL };

newNode 是一个带有 自动 存储持续时间,这意味着它只存在于它被声明的函数返回(或被退出)之前。

    newNode.data = data;
    if (*head == NULL)
    {
        *head = &newNode;

所以现在 *head 指向一个具有自动存储期限的对象。但看看接下来会发生什么。

        return;

只要这个 return 执行。insertAtStart 终止,其所有局部变量的寿命,包括 newNode 戛然而止。而当一个对象的寿命结束时,指向该对象的指针的可用性也就结束了。

    }
    newNode.next = *head;
    *head = &newNode;
}

我应该注意的是,虽然有一些规则反对你在这里做的事情,但实际上并没有任何东西试图去执行它们。事情只是以神秘的方式失败。

说对象的寿命结束,并不意味着存储该对象的内存不复存在。你的电脑并没有配备小纳米机器人,它可以构建和拆卸物理内存。它意味着内存不再包含该对象,并可能(将)被重新用于其他目的。

同样,虽然C标准明确指出,指向终止对象的指针不再可用("当指针所指向的对象......达到其生命期的终点时,指针的值就变得不确定了。"),但实际上并没有什么能阻止你尝试使用该指针,问题是它可能指向放置在同一内存中的不同对象。这就是这里发生的事情:结果是你的链接列表最终变成了一个循环的垃圾列表。

© www.soinside.com 2019 - 2024. All rights reserved.