0

RPN 計算機で次の入力を読み取って、順序に関係なく演算子を見つけるにはどうすればよいですか?

2
2+
4

今のところ、私の scanf は文字列の最初の文字しか認識せず、これしかできません:

2
2
+
4

また、整数対浮動小数点モードのオプションを追加しようとしています。(ex. '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;
 }
4

3 に答える 3

1

あなた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;
}
于 2010-01-28T10:39:08.090 に答える
0

私は本当にあなたのコードを理解していませんでした。

ユーザーが毎回1文字を入力することを期待している場合、つまり1文字+入力を意味する場合、char[]の代わりに単純なcharを使用する必要があります。そして、文字列を使用するふりをする場合は、それを受け取って解析する必要があります pzico は言いました。あなたはそのようなことをすることができます。問題は複数桁の数字の扱いですが、少し考えればこの問題は解決できます。私は試みを書きましたが、うまくいかないと確信しています。

printf("\nRPN 電卓\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("式を 1 文字ずつ入力してください\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) {
printf("スタック アンダーフロー\n");
0 を返します。
}

ポインタがアドレス 0 に到達することを期待していますか?

とにかく、これがあなたの宿題でないことを願っています。

于 2010-01-28T12:06:48.060 に答える
0

あなたの質問を完全に理解したかどうかはわかりませんが、次のように文字列を反復処理できます。

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

}

string.h も含めることを忘れないでください。

于 2010-01-28T10:41:58.630 に答える