問(wèn)題描述
本題總分:10 分
附件 prog.txt 中是一個(gè)用某種語(yǔ)言寫的程序冷冗。
其中 REPEAT k 表示一個(gè)次數(shù)為 k 的循環(huán)守屉。循環(huán)控制的范圍由縮進(jìn)表達(dá),從次行開(kāi)始連續(xù)的縮進(jìn)比該行多的(前面的空白更長(zhǎng)的)為循環(huán)包含的內(nèi)容蒿辙。
例如如下片段:
REPEAT 2:
?A = A + 4
?REPEAT 5:
??REPEAT 6:
???A = A + 5
??A = A + 7
?A = A + 8
A = A + 9
該片段中從 A = A + 4 所在的行到 A = A + 8 所在的行都在第一行的循環(huán)兩次中拇泛。
REPEAT 6: 所在的行到 A = A + 7 所在的行都在 REPEAT 5: 循環(huán)中滨巴。
A = A + 5 實(shí)際總共的循環(huán)次數(shù)是 2 × 5 × 6 = 60 次。
請(qǐng)問(wèn)該程序執(zhí)行完畢之后俺叭,A 的值是多少恭取?
解析
通過(guò)堆棧的方式,用兩個(gè)數(shù)組熄守,一個(gè)維護(hù)當(dāng)前行前空格數(shù)量用來(lái)判斷進(jìn)入新的循環(huán)蜈垮,一個(gè)維護(hù)當(dāng)前層次的添加的循環(huán)數(shù)量。逐行判斷裕照,通過(guò)冒號(hào)判定repeat語(yǔ)句或計(jì)算語(yǔ)句攒发,repeat則增加層次并且記錄當(dāng)前倍數(shù),計(jì)算語(yǔ)句則直接通過(guò)當(dāng)前倍數(shù)和當(dāng)前行數(shù)據(jù)對(duì)答案計(jì)算晋南;通過(guò)當(dāng)前行空格數(shù)量與前一行空格數(shù)量判斷是否減少層次惠猿,減少層次通過(guò)當(dāng)前倍數(shù)和當(dāng)前層次添加的循環(huán)數(shù)量計(jì)算減少層次后的倍數(shù),并且對(duì)層次減一负间。
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
typedef pair<int, int> PII;
const int MAXN=2020;
char s[MAXN];//讀取文件
int a[MAXN],b[MAXN];//分別維護(hù)當(dāng)前各層次前空格數(shù)量和當(dāng)前層次增加循環(huán)數(shù)偶妖。
int w;//維護(hù)當(dāng)前行倍數(shù)。
int main() {
freopen("prog.txt", "r", stdin);
int pos = 0, ans = 0, w = 1;
gets(s); // 讀走第一行的 A = 0
a[pos] = -1, b[pos] = 1; // 防止在椪#空的時(shí)候彈棧
while (gets(s)) {
int n = strlen(s), p = 0;
while (s[p] == ' ')
p++; // 統(tǒng)計(jì)縮進(jìn)
while (a[pos] >= p)
w /= b[pos--];// 彈掉棧里縮進(jìn)大于等于當(dāng)前行的
if (s[n - 1] == ':') { // 當(dāng)前行是循環(huán)餐屎,壓棧
int k = s[n - 2] - '0';
pos = pos + 1;
w *= k;
a[pos] = p, b[pos] = k;
} else {
int k = s[n - 1] - '0';
ans += k * w;
}
}
printf("%d\n", ans);
return 0;
}