aboutsummaryrefslogtreecommitdiff
path: root/ch7/7-05_postfix-calc-scanf.c
blob: fa20f27a58aab37e5c6fcbb65ad19ee46f4d7c07 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
#include <stdio.h>
#include <stdlib.h>
#include <ctype.h>
#include <math.h>
#include <string.h>

/* The C Programming Language: 2nd Edition
 *
 * Exercise 7-5: Rewrite the postfix calculator of Chapter 4 to use scanf
 * and/or sscanf to do the input and number conversion.
 *
 * Notes:
 * scanf wants allocated memory of some sort, meaning you're either going
 * to pass it a pointer (with malloc()'d memory) or you're passing a variable
 * with the unary & (address-of) operator. Failing to use scanf correctly
 * _will_ create runtime problems! gcc won't tell you what happens without
 * -Wall and -Wextra, but scanf segfaults by trying to access an uninitialized
 * pointer.
 *
 * Thus there are two ways to use scanf:
 *
 * int foo = 0;
 * scanf("%d", &foo);
 * // do stuff
 *
 * -- or --
 *
 * int *foo = malloc(sizeof(int));
 * scanf("%d", foo);
 * // do stuff
 * free(foo);
 *
 * For the sake of sanity in an easy exercise, I'll be using the former form
 * since a few values on the stack won't hurt anything.
 */

#define MAXOP    100
#define MAXVAL   100
#define BUFSIZE  100

void push(double);
double pop(void);

int sp = 0; // Next free stack position
double val[MAXVAL]; // Value stack
char buf[BUFSIZE]; // buffer
int bufp = 0; // next free position in buf
double vars[27];

/* Reverse Polish calculator:
 *
 * Binary operations (+-*\)
 *     operand operand operator
 *
 * Example: 6 minus 2 in Reverse Polish Notation is "6 2 -"
 */
int main() {
	char *c;
	double a = 0, op2;
	char s[MAXOP];
	char ltr = '\0';
	while (scanf("%s%c", s, &ltr) == 2) {
		if (sscanf(s, "%lf", &a) == 1) {
			push(a);
		} else if (sscanf(s, "%s", buf)) {
			for (c = buf; *c; c++) {
				switch (*c) {
					case '+':
						push(pop() + pop());
						break;
					case '*':
						push(pop() * pop());
						break;
					case '-':
						op2 = pop();
						push(pop() - op2);
						break;
					case '/':
						op2 = pop();
						if (op2 != 0.0) {
							push(pop() / op2);
						} else {
							printf("Error: Cannot divide by zero.\n");
						}
						break;
					default:
						printf("Error: Unknown command %s\n", s);
						break;
				}
			}
			if (ltr == '\n') {
				printf("%.8f\n", pop());
			}
		}
	}
	return 0;
}

void push(double f) {
	if (sp < MAXVAL) {
		val[sp++] = f;
	} else {
		printf("Error: Stack full. Cannot push %g\n", f);
	}
}

double pop(void) {
	if (sp > 0) {
		return val[--sp];
	} else {
		printf("Error: Stack empty.\n");
		return 0.0;
	}
}