Как определить оператора vs.int в C с помощью scanf?
-
23-09-2019 - |
Вопрос
Как мне прочитать следующие данные в моем калькуляторе 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?
В любом случае, я надеюсь, что это не ваше домашнее задание.