c++ - 段错误和指针问题

标签 c++ linux

在类里面研究这个程序几天了,我不断遇到段错误。我可以注释掉我认为导致该问题的代码部分,并且我不再遇到错误,但随后我的打印功能不起作用。我明天会去拿一本 C++ 书来帮忙。

//linkedList.cpp
//Declaration of main and menu functions
//Programmer: Ronnie Warden
//Date: 2.15.15

#include<iostream>
#include "link.h"
using namespace std;

int main()
{
    Node *list = new Node;
    int delNode;
    int findNode;
    int choice = 1;
    list->next = NULL;

    while (choice != 5)
    {
        choice = menu();

        if (choice == 1)
            *list = insertNode(list);

        if (choice == 2)
        {   
            cout << "Enter the ID you wish to delete: ";
            cin >> delNode;
            *list = deleteNode(list, delNode);
        }

        if (choice == 3)
            printNode(list);

        if (choice == 4)
        {
            cout << "Enter the ID you are searching for: ";
            cin >> findNode;
            *list = searchNode(list, findNode);
        }

    if (choice < 1 || choice > 5)
        cout << "Invalid choice! Please try again." << endl;

}

return 0;
}

int menu()
{
    int choice = 1;

         cout << "1. Insert Node" << endl;
         cout << "2. Delete Node" << endl;
         cout << "3. Print List" << endl;
         cout << "4. Search List" << endl;
         cout << "5. Quit" << endl;
         cin >> choice;

return choice;
}

以下部分是引发段错误的部分。

//linkFun.cpp
//Declaration of functions used to manage the linked list
//Programmer: Ronnie Warden
//Date: 2.10.15

#include<iostream>
#include "link.h"
using namespace std;


/*************************************************************************************************************
Function: createNode
Parameters: No parameters
Return Type: Pointer to new node
Task: Create a node with "new" operator. Get a student data from keyboard and assign to members of the node.
*************************************************************************************************************/

Node createNode()
{
Node *newNode = new Node;

    cout << "Last Name?" << endl;
    cin >> newNode->lastName;
    cout << "First Name?" << endl;
    cin >> newNode->firstName;
    cout << "ID?" << endl;
    cin >> newNode->idNumber;

return *newNode;
}

/**************************************************************************************************************
Function: insertNode
Parameters: Pointer to the linked list
Return Type: Pointer to the linked list
Task: insert a new node to the linked list sorted by student's ID number. If ID is already in list notify user 
***************************************************************************************************************/

Node insertNode(Node *list)
{
    Node *newNode = new Node;
    Node *tmp = new Node;
    *newNode = createNode();
    int id = newNode->idNumber;

    if (list == NULL)           //Insert in empty list
        list->next = newNode;
    else
    {
         *tmp = searchNode(list, id);

         if (tmp->idNumber == newNode->idNumber)
             {
                cout << "ID number already in list! Please try again with a different ID number." << endl;
                insertNode(list);
        }

    if (list != NULL)
    {
        Node *tmp = list;

         if (tmp->idNumber > newNode->idNumber)     //Insert as first
         {
             newNode->next = tmp;
            list = newNode;
        }
    while (tmp->idNumber < newNode->idNumber)   //Search for insertion point
    {   
                if (tmp->next == NULL)        //Insert at end
                            tmp->next == newNode;

        tmp = tmp->next;

        if (tmp->idNumber < newNode->idNumber && tmp->next->idNumber > newNode->idNumber && tmp->next != NULL)  //Insert in-between
        {
            newNode->next = tmp->next->next;
            tmp->next = newNode;
        }

        }
    }
}
return *list;
}

 /***************************************************************************************************************
Function: searchNode
Parameters: Pointer to the linked list, student ID number
Return Type: Pointer to the node with matched ID number
Task: Search the linked list by student Id number. Notify user if ID is not found
***************************************************************************************************************/

Node searchNode(Node *list, int id)
{
    Node *missing = new Node;

    if (list->idNumber == id)   //Checks if first node matches id number
         return *list;
    else 
    {
        Node *tmp = new Node;   //creates temporary pointer to scroll through list
        while (tmp->idNumber != id)
       {    
            if (tmp->next == NULL)  //checks if number is missing returns sentinel if not found
            {
                missing->idNumber = 9999;   
                return *missing;
            }

             tmp = tmp->next;
         }
     return *tmp;

     }
 }

/***************************************************************************************************************
Function: deleteNode
Parameters: Pointer to the linked list, student ID number
Return Type: Pointer to the list
Task: Delete a node from the list. Notify user if no node matches
***************************************************************************************************************/

Node deleteNode(Node *list, int id)
{   
    Node *tmp = new Node;
    *tmp = searchNode(list, id);    
return *list;

}

/***************************************************************************************************************
Function: printNode
Parameters: Pointer to the linked list
Return Type: None
Task: Display the linked list
***************************************************************************************************************/

void printNode(Node *list)
{
    Node *tmp = new Node;
    tmp = list;

    while (tmp != NULL)
    {
        cout << tmp->lastName << endl;
        cout << tmp->firstName << endl;
        cout << tmp->idNumber << endl;
        tmp = tmp->next;
    }
}

如果我注释掉 insertNode 函数中的整个 else 语句,我将停止获取 seg 错误,但是当我调用打印时,它会打印 3 个空行,然后打印 0。任何帮助将不胜感激,如果出现,我没有使用类的原因是因为我不允许这样做,它必须以结构化数据类型完成。

#ifndef LINK_H
#define LINK_H

struct Node {
    char lastName[20];
    char firstName[20];
    int idNumber;
    Node *next;
};

int menu();
Node createNode();
Node insertNode(Node*);
Node searchNode(Node*, int);
Node deleteNode(Node*, int);
void printNode(Node*);

#endif

最佳答案

你的主要问题是你没有正确使用指针 - 你有点介于指针和对象之间。 (我不会说明这是 C++ 中的 C 风格程序)

例如:您的 createNode() 方法返回一个 Node 对象。它通过在堆上创建一个新的Node,然后返回该节点的内容来实现这一点。这意味着您实际上发生了内存泄漏。没有人记得指向您放入堆中的节点的指针,因此它泄漏了。

我的第一步是让 createNode() 返回一个 Node*

Node* createNode()
{
    Node *newNode = new Node;

    // All your cout/cin stuff

    return newNode;
}

我会遵循与其他方法相同的模式 - 使用指针而不是对象。以searchNode()为例。因为您使用的是对象,所以需要一个特殊的“Not Found”节点。如果使用指针,则可以针对“未找到的情况”返回 NULL(在 C++ 中最好为 0)。您也不需要new和临时变量。

Node* searchNode(Node* list, int id)
{
    while(list != 0) {
        if (list->id == id) {
            return list;
        }
        // This only impacts the local list. It doesn't
        // permanently change the value of the list
        // passed in.
        list = list->next;
    }
    return 0;
}

关于 searchNode() 的奖励积分:由于它是一个排序列表,因此一旦找到大于您要查找的 id 的 id,您就可以停止搜索。

既然你已经做到了这一点,你的插入方法就简单了一些:

  1. 从用户处获取新节点
  2. 检查它是否存在 - 如果存在,则删除您刚刚创建的节点并给出错误
  3. 如果没有,请搜索列表并找到插入新节点的位置。

关于c++ - 段错误和指针问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28620783/

相关文章:

linux - 为什么只有在按回车后才能在串行连接中接收到字符?

c++ - 从 Satellite DLL 加载 "special"MFC 控件

c++ - 包括标题和 Main.h

python - 使用 top 生成每个内核的 Linux CPU 信息报告

python - 使用xterm打开新控制台: How to while the current console is printing,也在新控制台上打印

linux - 不执行 "make modules_install"就执行 "make modules"有意义吗?

c++ - 传递 `boost::variant` 数据类型的优雅方法

c++ - 大型复杂的构造函数还是私有(private)方法?

c++ - 如何在 GDI+ C++ 中克隆 solidbrush

linux - 如何为 kubectl 命令提供输入?