UVA 1596 Bug Hunt (大模拟 栈)
题意:
输入并模拟执行一段程序,输出第一个bug所在的行。
每行程序有两种可能:
数组定义:
格式为arr[size]。
例如a[10]或者b[5],可用下标分别是0~9和0~4。定义之后所有元素均为未初始化状态。
赋值语句:
格式为arr[index]=value。
或者arr[index] = arr[arr[index]]。
例如a[0]=3或者a[a[0]]=a[1]。
赋值语句和数组定义可能会出现两种bug:
下标index越界;
使用未初始化的变量(index和value都可 能出现这种情况。
程序不超过1000行,每行不超过80个字符且所有常数均为小于2 31的非负整数。
分析:
可以分为3种情况考虑:

首先考虑开以下两个map

声明语句:
对于①: 那么"a["与最后的"]"都是肯定存在的, 所以声明语句的a是肯定合法的,只需要判断...能不能转化为一个值num即可,如果可以转换为一个值就是语句就是合法的
arr_maxi[a] = num;
赋值语句:
对于②:需要赋值部分,判断条件
1.需要去找一下a有没有定义,
2.判断...能不能转化一个值num,
3.num有没大于a的最大下标,
对于③:值部分,判断条件
1.b有没有定义
2.“...”能不能转化为一个值num
3.b[num]有没有初始化
如果②③都满足了, 那么这句赋值语句就是合法的,插入 arr_value[pair(数组名,下标)] = 值
现在问题就是,如何判断...能不能转化为一个值
我的方法是使用栈,首先明确, 每条语句只有:大小写字母,[,],数字,四种情况
那么可以开一个 字母栈 一个其他栈
然后遇到字母就将字母入字母栈 遇到其他就入其他栈(这里要注意将字符串的数字转化为int再入栈)。
入其他栈有一个类似括号匹配的过程,
如果碰到“]”,那么就一直出栈直到遇到“[’, 那么中间遇到的那个数字就是,数组名为字母栈顶,下标为这个数字的值, 如果这个值存在, 那么就将这个值入栈, 直到其他栈最后只剩下一个值,没有其他元素。
#include <bits/stdc++.h>
using namespace std;
typedef pair<char, int> PCI;
map<PCI, int> arr_value; // 数组名和下标, 值
map<char, int> arr_maxi;//数组名 , 最大下标
int F = ;
int line;
void error()
{
if(!F)
{
F = ;
cout << line << "\n";
}
}
int read(const string& s, int type)
{
stack<int> name;
stack<int> index;
int k = , v = ;
char a = ;
if(type == )//等于号左边
{
if(!arr_maxi.count(s[]))//未定义
{
return -;
}
}
else if(type == )//要赋的值
{
if(!arr_maxi.count(s[]))//未定义
{
return -;
}
}
for(int i = ; i <= s.size() - ; i++)//求出括号内的东西
{
if(isalpha(s[i]))
{
name.push(s[i]);
}
else
{
if(isdigit(s[i]))
{
int j = i;
int sum = ;
while(isdigit(s[j]))//将字符串转换为十进制数
{ sum *= ;
sum += s[j] - '';
j++;
}
i = j;
index.push(sum);
if(index.size() == ) //此时只有数字
{
v = sum;
break;
}
}
else
{
if(s[i] == '[')//左括号
{
index.push(s[i]);
}
else //右括号
{
k = index.top();
index.pop();//下标出栈
index.pop();//左括号出栈
a = name.top();
name.pop(); //名字出栈 if(arr_value.count(make_pair(a,k)))//找到a[k]
{
if(index.empty())//只剩下这个数
{
v = arr_value[make_pair(a,k)];
break;
}
else
{
index.push(arr_value[make_pair(a,k)]);
}
}
else//找不到a[k]的值 == 1.a没定义 2. a[k]没赋值
{
return -;
}
}
}
}
}
if(type == ) //等于号左边
{
if(v <= arr_maxi[s[]]) //小于最大下标
{
return v;
}
else
{
return -;
}
}
if(type == ) //要赋的值,
{
if(!arr_value.count(make_pair(s[],v)))
{
return -;
}
else
{
return arr_value[make_pair(s[],v)];
}
}
if(type == )//声明语句
{
return v;
}
}
int main()
{
#if LOCAL
freopen("1.txt","r",stdin);
// freopen("2.txt","w",stdout);
#endif // LOCAL ios::sync_with_stdio(false);
string s;
while(getline(cin,s) && s[] != '.')
{ arr_value.clear();
arr_maxi.clear();
line = ;
F = ;
do
{
if(s.find('=')!= string :: npos)//赋值语句
{
string a,b;
a = s.substr(,s.find('='));
b = s.substr(s.find('=')+);
int value ,flag = ;
if(isdigit(b[]))//如果第一个是数字,那么只有数字
{
flag = ;
int sum = ;
for(int i = ; i < b.size(); i++)
{
sum *= ;
sum += b[i] - '';
}
value = sum;
}
else//第一个是字母
{
value= read(b,);//先读取要赋的值
}
if(value < )//错误
{
error();
}
int index = read(a,);
if (index < )
{
error();
}
arr_value[make_pair(s[],index)] = value; }
else //声明语句
{
int v = read(s,);
if (v < )
{
error();
}
else
{
arr_maxi[s[]] = v - ;
}
}
line ++;
}while(getline(cin , s) && s[] != '.');
if(!F)//读到最后都没发现错误
{
cout <<"0\n";
}
}
}
UVA 1596 Bug Hunt (大模拟 栈)的更多相关文章
- UVa 1596 Bug Hunt (string::find && map && 模拟)
题意 : 给出几组由数组定义与赋值构成的编程语句, 有可能有两种BUG, 第一种为数组下标越界, 第二种为使用尚未定义的数组元素, 叫你找出最早出现BUG的一行并输出, 每组以' . '号分隔, 当有 ...
- 【UVA】1596 Bug Hunt(模拟)
题目 题目 分析 算是个模拟吧 代码 #include <bits/stdc++.h> using namespace std; map<int,int> a[ ...
- UVa 1596 Bug Hunt (STL栈)
题意:给定两种操作,一种是定义一个数组,另一种是赋值,让你找出哪一步时出错了,出错只有两种,一种是数组越界,另一种是访问未定义变量. 析:当初看到这个题时,感觉好麻烦啊,然后就放过去了,而现在要重新回 ...
- 【技巧性(+递归运用)】UVa 1596 - Bug Hunt
In this problem, we consider a simple programming language that has only declarations of onedimensio ...
- uva 1596 Bug Hunt
In this problem, we consider a simple programming language that has only declarations of one-dimensi ...
- KEILC51可重入函数及模拟栈浅析
MARK:文章中的红色部分是个人的理解. KEILC51可重入函数及模拟栈浅析 关键字:keilc51,模拟堆栈,可重入函数调用,参数传递,C?XBP,C?ADDXBP 摘要:本文较详细的介绍了kei ...
- NOIP2017 时间复杂度 大模拟
再写一道大模拟题. 由于是限时写的,相当于考场代码,乱的一批. 题目链接:P3952 时间复杂度 先记几个教训: 字符串形式的数字比较大小老老实实写函数,字典序都搞错几次了 栈空的时候不但pop()会 ...
- Java连载69-接受输入、用数组模拟栈
一.编写一个酒店管理系统 1.直接上代码 package com.bjpowernode.java_learning; public class D69_1_ { //编写一个程序模拟酒店的管理系 ...
- java模拟栈的操作
栈是一种有序列表,可以使用数组的结构来储存栈的数据内容 思路 1. 创建一个栈类StackArray 2. 定义一个top来模拟栈顶,初始化为-1 3. 入栈: 当有数据加入到栈的时候 top++ s ...
随机推荐
- 给Ambari集群里安装可视化分析利器工具Hue步骤(图文详解)
扩展博客 以下,是我在手动的CDH版本平台下,安装Hue. CDH版本大数据集群下搭建Hue(hadoop-2.6.0-cdh5.5.4.gz + hue-3.9.0-cdh5.5.4.tar.gz) ...
- failed to push some refs to 'https://gitee.com/ftl_663/java-shop.git'
1.git init 2.git add . 3.git commit -m "init" 4.git remote add origin https://gitee.com/ ...
- Hdu 3488 Tour (KM 有向环覆盖)
题目链接: Hdu 3488 Tour 题目描述: 有n个节点,m条有权单向路,要求用一个或者多个环覆盖所有的节点.每个节点只能出现在一个环中,每个环中至少有两个节点.问最小边权花费为多少? 解题思路 ...
- 【洛谷3822】[NOI2017] 整数(线段树压位)
题目: 洛谷 3822 分析: 直接按题意模拟,完了. 将每次加 / 减拆成不超过 \(32\) 个对单独一位的加 / 减. 考虑给一个二进制位(下称「当前位」)加 \(1\) 时,如果这一位本来就是 ...
- SIFT特征点检测与匹配
SIFT的步骤如下: (1) 尺度空间极值检测(Scale-space Extrema Detection) 也就是在多尺度高斯差分(Difference of Gauss)空间中检测极值点(3x3x ...
- 题解报告:Luogu P3368 【模板】树状数组 2(区间修改,单点查询)
题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数数加上x 2.求出某一个数的和 输入输出格式 输入格式: 第一行包含两个整数N.M,分别表示该数列数字的个数和操作的总个数. ...
- 给ambari集群里的kafka安装基于web的kafka管理工具Kafka-manager(图文详解)
不多说,直接上干货! 参考博客 基于Web的Kafka管理器工具之Kafka-manager的编译部署详细安装 (支持kafka0.8.0.9和0.10以后版本)(图文详解)(默认端口或任意自定义端口 ...
- 转】MongoDB 自动分片 auto sharding
原博文出自于: http://blog.fens.me/category/%E6%95%B0%E6%8D%AE%E5%BA%93/page/4/ 感谢! MongoDB 自动分片 auto shard ...
- Java文件上传(基础性)
/** * * 上传文件 * */ public class FileUploadServlet2 extends HttpServlet { protected void doGet(HttpSer ...
- SpringBoot_自定义配置属性
@ConfigurationProperties 在aplication.properties 中添加如下一段配置: mysql.jdbcName=com.mysql.jdbc.Driver mysq ...