Вопрос

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

2
2+
4

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

2
2
+
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;
 }
Это было полезно?

Решение

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

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

Аналогичная функция для чисел с плавающей запятой: 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;
}

Другие советы

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

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

}

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

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

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

printf(" RPN Калькулятор ");
printf("Введите 'i' для целочисленного режима ");
printf("Введите 'f' для режима с плавающей запятой ");
printf("Для выхода введите 'q' ");
сканирование("%c", с);

переключатель(*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.Что вы хотите сделать с помощью этого теста:

если (р < 0) {
printf("Переполнение стека ");
вернуть 0;
}

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

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

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top