Как мне определить оператор против int в C, используя scanf? - PullRequest
0 голосов
/ 28 января 2010

Как мне прочитать следующие входные данные в моем калькуляторе RPN, чтобы найти оператора независимо от того, в каком порядке?

2
2 +
4

На данный момент мой scanf видит только первый символ в строке, и я могу только сделать это:

2
2 * * +1010 +
4

Я также пытаюсь добавить опцию для целочисленного или плавающего режима. (например, когда вводится «i», работайте с плавающей запятой и наоборот.)

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

#define MAX 100

int *p;
int *tos;
int *bos;

void push(int i);
int pop(void);

int main (void)
{
int a, b;
//float c, d;
char s[80];
//char op;  //declare string of 80 chars

p = (int *) malloc(MAX*sizeof(int)); //get stack memory
if (!p){
 printf("Allocation Failure\n");
 exit(1);
 }
 tos = p;
 bos = p + MAX-1;

 printf("\nRPN Calculator\n");
 printf("Enter 'i' for integer mode\n");
 printf("Enter 'f' for floating point mode\n");
 printf("Enter 'q' to quit\n");

 do {
 printf("> ");
 // gets(s);
 // scanf("%s", s);  //read integer
 scanf("%s", s);
 // switch (*s) {


 switch(*s) {
   case 'i':
       printf("(Integer Mode)\n");
   break;
   case 'f':
       printf("(Floating Point Mode)\n");
   break;
   case '+':
       a = pop();
       b = pop();
       printf("%d\n", a+b);
       push(a+b);
   break;
   case '-':
       a = pop();
       b = pop();
       printf("%d\n", b-a);
       push(b-a);
       break; 
  case '*':
       a = pop();
       b = pop();
       printf("%d\n", a*b);
       push(a*b);
  break;
  case '/':
       a = pop();
       b = pop();
       if(a == 0){
         printf("Cannot divide by zero\n");
  break;
  }
       printf("%d\n", b/a);
       push(b/a);
  break;
  case '.':
       a = pop();
       push(a);
       printf("Current value on top of stack: %d\n", a);
  break; 
  default:
       push(atoi(s));
  }

 } while (*s != 'q');

 return 0;
}


 // Put an element on the stack

 void push (int i)
 {
 if (p > bos){
   printf("Stack Full\n");
 return;
 }
  *p = i;
  p++;
 }

// Get the element from the top of the stack

int pop (void)
{
 p--;
 if(p < 0) {
    printf("Stack Underflow\n");
 return 0;
 }
 return *p;
 }

Ответы [ 3 ]

1 голос
/ 28 января 2010

Ваш scanf читает всю строку.Это следующий переключатель, который судит по первому символу и пропускает + в 2+.

Чтобы улучшить его, вы можете использовать функцию strtol.Он проанализирует целое число из строки и вернет вам место, где закончилось целое число - если это еще не конец строки, там может быть оператор.

Аналогичная функция для чисел с плавающей запятойis strtod.


Вот пример кода strtol, применимый к вашему примеру:

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


int main()
{
    char* input = "25+";
    char* endptr;

    int val = strtol(input, &endptr, 10);

    if (*endptr == '\0')
    {
        printf("Got only the integer: %d\n", val);
    }
    else
    {
        printf("Got an integer %d\n", val);
        printf("Leftover: %s\n", endptr);
    }


    return 0;
}
0 голосов
/ 28 января 2010

Я действительно не понял ваш код.

Если ожидается, что пользователь будет вводить один символ каждый раз, я имею в виду один символ + ввод, вам следует использовать простой символ вместо символа []. И если вы притворяетесь, что используете строку, вы должны ее получить и разобрать, сказал Пцико. Вы могли бы сделать что-то подобное. Проблема заключалась бы в обработке чисел с несколькими цифрами, но, подумав немного, вы можете решить эту проблему. Я написал попытку, но я уверен, что это не сработает.

printf ("\ nRPN Calculator \ n");
printf ("Введите 'i' для целочисленного режима \ n");
printf ("Введите 'f' для режима с плавающей запятой \ n");
printf ("Введите 'q', чтобы выйти \ n");
scanf ("% c", s);

переключатель (* s) {

case 'i':  
    printf("(Integer Mode)\n");  
break;  
case 'f':  
    printf("(Floating Point Mode)\n");  
break;  
case 'q':  
    printf("Bye Bye\n");  
    return;  
break;  

} printf («Введите выражение по одному символу каждый раз \ n»);

до {

scanf("%c", s);   
switch(s) {   
    case '+':   
        a = pop();      
        b = pop();     
        printf("%d\n", a+b);      
        push(a+b);
    break;
    case '-':
        a = pop();
        b = pop();
        printf("%d\n", b-a);
        push(b-a);
    break; 
    case '*':
        a = pop();
        b = pop();
        printf("%d\n", a*b);
        push(a*b);
    break;
    case '/':
        a = pop();
        b = pop();
        if(a == 0){
            printf("Cannot divide by zero\n");
            break;
        }
        printf("%d\n", b/a);
        push(b/a);
    break;
    case '.':
        a = pop();
        push(a);
        printf("Current value on top of stack: %d\n", a);
    break; 
    default:
        a = pop()*10+atoi(s);
        push(a);
}  

} while (s! = 'Q');

Другая проблема в вашем коде связана с функцией pop. Что вы хотите сделать с этим тестом:

if (p <0) {<br> printf ("Переполнение стека \ n");
возврат 0;
}

Вы ожидаете, что указатель достигнет адреса 0?

В любом случае, надеюсь, это не твоя домашняя работа.

0 голосов
/ 28 января 2010

Я не уверен, что полностью понял ваш вопрос, но вы можете перебрать строку так:

for(i = 0; i < strlen(s); i++)
{
   // Here comes your switch section like this
   switch(s[i]) {
    .....
   }

}

Не забудьте также включить string.h.

...