【问题标题】:issue with multifunction utility program in c.c中的多功能实用程序问题。
【发布时间】:2013-06-09 09:03:54
【问题描述】:

以下程序应该从命令行读取(使用 argv[])并执行三个函数之一:

right(根据边长判断一个三角形是否为直角三角形:通过输入 mu -r [sidea] [sideb] [sidec]

findtext:(在给定文件中查找字符串的序列:(ie,"hello") 并列出找到它的行号。

count:(计算制表符、单词和退格的数量)。

Findtext.c 正在正常运行:

例如,如果我输入 ./mu -f [string] [file.txt]

它成功地列出了文件和找到它的文件中的行号。

但是当我使用 -r(right) 选项运行相同的代码时,它会给我以下分段错误:

程序名称:./mu 3 分段错误(核心转储)

这段代码我哪里出错了?

#include <stdio.h>
#include <stdlib.h>
//#include "count.h"
//#include "right.h"
//#include "findtext.h"

#define STAND_ALONE 1
void right(int, char **);
void count(int, char **);
void findtext(int, char **);
#ifdef STAND_ALONE 
int main(int argc, char *argv[])
{

      printf("\n");
      printf("Can only use one option(-f, -c, -r) at once. sorry!\n");
      printf("\n");


         printf("Program name: %s\n", argv[0]);

    while ((argc > 1) && (argv[1][0] == '-'))
    {
        switch (argv[1][1])
        {
            case 'f': // findtext.c
                printf("%s\n",&argv[1][2]);
                findtext(argc, argv);
                                break;

            case 'r': // right.c
                printf("%s\n",&argv[1][2]);
                right(argc, argv);
                break;
                        case 'c': // count.c
                printf("%s\n",&argv[1][2]);
                count(argc,argv);
                break;

            default:
                printf("Wrong Argument: %s\n", argv[1]);

        }

        ++argv;
        --argc;
    }
    return (0);
}

#endif

void right(int argc, char *argv[]){
int a;
int b;
int c;
int largest;
int a2;
int b2;
int c2;

/*if(argc != 4){
printf("please enter 3 sides, only \n");
} */

a = atoi(argv[2]);
b = atoi(argv[3]);
c = atoi(argv[4]);
//printf("argv2:%s ",argv[2]);
if((a <= 0 )|| (b <= 0) || (c <= 0))
     printf("Only positive values allowed\n"); exit(0);

a2 = (a*a);
b2 = (b*b);
c2 = (c*c);


if((c > a) && (c > b))
     largest = c;
if((b > a) && (b > c))
     largest = b;
if((a > b) && (a > c))
     largest = a;



if(largest == a){
  printf("HEy hey hey!");
  if((b2 + c2) == a2){ printf("%s %s %s is a right triangle\n",argv[2],argv[3],argv[4]); } 
   else{printf("%s %s %s is not a right triangle\n",argv[3],argv[2],argv[4]);}
 }

if(largest == b){
   printf("HEy");
  if((a2 + c2) == b2){ printf("%s %s %s is a right triangle\n",argv[2],argv[3],argv[4]); } 
   else{printf("%s %s %s is not a right triangle\n",argv[2],argv[3],argv[4]);}

 }

if(largest == c){
  printf("yo");
   if((a2 + b2) == c2){ printf("%s %s %s is a right triangle\n",argv[2],argv[3],argv[4]); } 
   else{printf("%s %s %s is not a right triangle\n",argv[2],argv[3],argv[4]);}

 }




} /* end method right() */



void findtext(int argc, char *argv[]){
FILE *fin;
char buffer[100];
int counter;
char *ptr = buffer;
char *result;

//if(argc != 3) {printf("Usage: %s filename  argc:%d\n", argv[0], argc); exit(1);}
  fin = fopen(argv[3], "r");
  if(!fin) {printf("Unable to open %s\n", argv[2]); exit(1); }

 counter = 0; 
 while (fgets(buffer, 99, fin)){
   counter = counter + 1; 
   if(strstr(ptr,argv[2])){
      printf("%d. %s", counter, ptr);
      printf("\n");

     }

  }
  fclose (fin);


}


void count(int argc,  char **argv){
FILE *fin;

int lcounter = 0;
int count = 0;
char name[100];
char ch;

int word = 0;
int nchar = 0;



fin = fopen(argv[1],"r"); // open file

if(fin==0){

printf("Could not find specified file.\n");
exit(0);
}

while((ch = getc(fin)) != EOF)
{

  nchar++;
  if(ch == '\n')
    lcounter++;
  if(isspace(ch) || ch == '\t' || ch == '\n')
      word++;
}

printf("number of characters: %d\n",nchar);
printf("Lines: %d\n",lcounter);
printf("words: %d\n",word);
printf("\n");

fclose(fin);


}

【问题讨论】:

  • SO 不能替代调试器。
  • 请包含准确的错误消息文本。您是否获得了发生错误的行号?
  • 我认为 GDB 是一个非常有用/强大的调试器。
  • 它只是说分段错误它没有说它来自哪一行。
  • " 它没有说明它来自哪一行" -- 如果您使用正确的标志和工具,它会显示。

标签: c command-line-arguments


【解决方案1】:
if((a <= 0 )|| (b <= 0) || (c <= 0))
 printf("Only positive values allowed\n"); exit(0);

当心 if 块。您没有使用大括号。在 right 函数的这一点上,无论如何程序都会结束。

当然,这不是您要寻找的错误,但它提供了一个起点:错误必须在此点之前。

您可能需要调试这里发生的事情:

a = atoi(argv[2]);
b = atoi(argv[3]);
c = atoi(argv[4]);

【讨论】:

  • 多梅尼科,你的权利。我应该在那个 if 语句中添加大括号。但是,我希望程序在输入负值时退出。
  • 关于 a = atoi(argv[2]),我将其设置为从 char 转换为 int。我什至添加了测试打印语句以确保 a、b、c 具有值。
  • @user2467276 如果输入的边数少于三个,你想做什么? (显然,获得一个 segvio。)
【解决方案2】:
  1. 参见@Domenico De Felice 的回答。

  2. 添加#include &lt;string.h&gt;#include &lt;ctype.h&gt;

  3. 您正在访问 argv[1] 中的 chars,却不知道它的长度足以容纳这些字符。 while ((argc &gt; 1) &amp;&amp; (argv[1][0] == '-'))printf("%s\n",&amp;argv[1][2]); 应该只打印一个换行符,因为argv[1] 是“-r”。

  4. right() 中,您在不知道存在 5 个参数的情况下执行 c = atoi(argv[4]);

  5. 请查看您对有趣代码的需求++argv;--argc;。我真的不认为这些会做你想要的。

我认为您正在崩溃,因为您并不总是输入代码希望读取的参数(命令参数)的数量。在使用程序参数之前增加对它们的验证。

5.

【讨论】:

  • 那么我应该添加一个 if 语句来检查 argc 的值吗?比如 if(argc
  • 是的,您需要确保 argc 足够好。 (注意:我的第 3 点可能不正确。)告诉我有关 ++argv 的信息; --argc;看起来很麻烦。
  • 我假设用户输入 ./mu -r 3 4 5 例如 argv[0] 是 mu argv[1] 是 -r argv[2] 是 3 argv[3] 是 4 和 argv [4] 是 5。我还能如何访问三角形边长。
  • 请提供示例运行的记录,例如:Mark1a\Debug&gt;Mark1a -r 3 4 5Can only use one option(-f, -c, -r) at once. sorry!Program name: Mark1ayo3 4 5 is a right triangle 该代码适用于我。
  • 感谢 Domenic 和 Chux,我的分段错误消失了,编译器警告也消失了。但是,它不会打印侧面是否是正确的三角形。
【解决方案3】:

你正在崩溃,因为你试图增加 argv 这是一个数组变量,而不是一个指针。所以你不能做 argv++;

作为一个很好的编程原则,(虽然它是可选的)argc和argv不应该被触及;如果你想遍历它们,你应该使用其他变量来指向它们。

代码中还有许多其他的丑陋之处,即尝试访问指针和地址而不实际确认它们是否存在。

但是,从 argv 开始是您出现段错误的地方。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-01-24
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2014-04-29
    相关资源
    最近更新 更多