1

我对 C 很陌生,但我想构建一个程序,允许用户存储从首先存储到文本文件中的 traceroute/tracert 获得的 IP 地址。然后它允许他们打印下一个/上一个跃点。我使用了一个链表,但打印给了我一个分段错误。

我尝试浏览它,但我找不到任何错误,有人会指出我的错误并指导我吗?提前致谢!!

(抱歉压痕不好!!)

# include <stdio.h>
# include <string.h>
# include <stdlib.h>

int id = 0;
int list = 0;
int nodes = 0;

FILE *readFile(char *fileName)
{
  FILE *rawdata = fopen(fileName, "r");
  return rawdata;
}

void printMenu()
{
  printf(" ========== MENU =============\n\n");
  printf("1. Report ID of Previous Hops of a network node\n");
  printf("2. Identify the next hops of a network node\n");
  printf("3. Quit\n");
}

int getInput()
{
  int choice;
  printf("Please select your choice (1 to 3) =>: ");
  scanf("%d", &choice);
  return choice;
}

struct NodeInfo {
  int ID;
  int IP1;
  int IP2;
  int IP3;
  int IP4;
  struct NodeInfo *next;
  struct NodeInfo *prev;
};

typedef struct NodeInfo Node;
Node *Topology;

Node *createNode(int ip1, int ip2, int ip3, int ip4)
{
  Node *newNode = malloc(sizeof(Node));
  newNode->IP1 = ip1;
  newNode->IP2 = ip2;
  newNode->IP3 = ip3;
  newNode->IP4 = ip4;
  newNode->next = 0;            // NULL Pointer
  newNode->prev = 0;            // NULL Pointer
  return newNode;
}

void addToBack(Node * tempnode)
{
  Node *n = Topology;
  Node *tail = 0;
  while (n != NULL) {
    tail = n;
    n = n->next;
  }
  tail->next = tempnode;
  tempnode->prev = tail;
}


void printFile(FILE * newFile)
{

  char data[256], nth1[50], nth2[50], nth3[50], nth4[50], nth5[50],
      nth6[50], nth7[50], ip[50], ip2[15], ip2new[14];
  int linecount = -1, strlength;
  int ip1, ip2x, ip3, ip4;
  int ip11, ip21, ip31, ip41;

  if (newFile == NULL) {
    printf("There is an error with opening this file\n");
  } else {
    while (fgets(data, 256, newFile) != NULL) {

      if (linecount != 3) {
        linecount++;
        continue;
      } else {

        if (linecount == 3 && data[2] != '\0') {
          sscanf(data, "%s %s %s %s %s %s %s %s", nth1, nth2, nth3, nth4,
                 nth5, nth6, nth7, ip);
          sscanf(data, "%s %s %s %s %s %s %s %d.%d.%d.%d", nth1, nth2,
                 nth3, nth4, nth5, nth6, nth7, &ip1, &ip2x, &ip3, &ip4);

          if ((ip[0] <= 'z' && ip[0] >= 'a')
              || (ip[0] <= 'Z' && ip[0] >= 'A')) {
            sscanf(data, "%s %s %s %s %s %s %s %s %s",
                   nth1, nth2, nth3, nth4, nth5, nth6, nth7, ip, ip2);
            //Rescanning for anomaly results with additional hostname
            strncpy(ip2new, ip2 + 1, strlen(ip2) - 2);
            ip2new[strlen(ip2) - 2] = '\0';

            int i;
            char *temp;
            char *ipcmp[4];
            i = 0;
            temp = strtok(ip2new, ".");
            while (temp != NULL) {
              ipcmp[i++] = temp;
              temp = strtok(NULL, ".");
            }
            Node *tempnode = createNode(ip2new);
            if (Topology != 0) {
              addToBack(tempnode);
            } else {
              Topology = tempnode;
            }
          } else {
            printf("%s\n", ip);
            printf("%d.%d.%d.%d\n", ip1, ip2x, ip3, ip4);
            Node *tempnode2 = createNode(ip);
            if (Topology != 0) {
              addToBack(tempnode2);
            } else {
              Topology = tempnode2;
            }
            continue;
          }
        }
        if (linecount == 3 && data[2] == '\0') {
          linecount = -2;
          printf("\n");
        }
      }
    }
  }
}

void printNodes()
{
  Node *n = Topology;
  while (n != 0) {
    printf("The node is %d.%d.%d.%d\n", n->IP1, n->IP2, n->IP3, n->IP4);
    n = n->next;                // Jump to next node
  }
}

int main(int argc, char *argv[])
{

  int option, fail;
  FILE *filedata;
  char *file;
  file = argv[1];
  filedata = readFile(file);    //open file
  printFile(filedata);          //prints the ip addresses
  do {
    printMenu();
    option = getInput();
    switch (option) {
    case 1:
      printf("You have selected 1\n\n");
      fail = 0;
      printNodes();
      break;
    case 2:
      printf("You have selected 2\n\n");
      fail = 0;
      break;
    case 3:
      fail = 1;
      break;
    default:
      printf("Please enter a valid choice (1-3) \n");
      fail = 0;
      break;
    }
  } while (fail != 1);
  while (Topology != 0) {
    free(Topology);
    Topology = Topology->next;
  }
}
4

3 回答 3

1

您的创建节点方法有 4 个参数:

Node *createNode(int ip1, int ip2, int ip3, int ip4)

但是您只需传递一个参数即可调用此方法:

Node *tempnode = createNode(ip2new);
Node *tempnode2 = createNode(ip);

当您的方法只接受整数时,您也会传递数组。

这至少是代码中的两个错误来源。

于 2013-08-13T01:36:58.600 回答
1

我可以发现两个更正。首先是您createNode()使用错误数量的参数调用。您已定义createNode()接受 4 个整数参数,但在每个调用中,您都向其传递了一个字符串参数。编译器应该给你一个关于这个问题的诊断(我的编译器拒绝编译代码)。您永远不应该忽略编译器诊断(至少,在没有首先完全理解诊断的含义之前不要忽略)。

更改这些调用以使用您扫描的 4 个积分变量。

            Node *tempnode = createNode(ip1, ip2x, ip3, ip4);

            Node *tempnode2 = createNode(ip1, ip2x, ip3, ip4);

第二个错误是您if (linecount != 3) {只允许您在linecount等于 3 时处理数据,这可能不是您想要做的。您可能的意思是在linecount变为 3 之后处理所有数据行。

      if (linecount < 3) {
        linecount++;
        continue;
      } else {

我觉得您初始化linecount为很奇怪-1,但这仅意味着您在到达输入的第四行之前不会开始处理文件。

正如其他人所提到的,您无法从已释放的内存中读取。这样做会导致未定义的行为。在您的循环中释放 持有的内存Topology,在释放当前项目之前保存指向下一个项目的指针。

  while (Topology != 0) {
    void *cur = Topology;
    Topology = Topology->next;
    free(cur);
  }
于 2013-08-13T01:40:28.993 回答
0

这里有几个问题。

  1. 您混合使用NULL0。虽然两者都可以工作,但在 C 中使用 是惯用的NULL,并且最好在使用中保持一致。

  2. 您的createNode函数需要 4 个整数,但您的代码正在调用此函数并且只提供一个参数。

  3. 您的循环中有未定义的行为:

    while (Topology != 0) {
        free(Topology);
        Topology = Topology->next;
    }
    

    Topology一旦被释放,访问它是未定义的行为。在您的情况下,您正试图在其被释放next访问该成员。Topology

于 2013-08-13T01:46:15.400 回答