【面试笔试算法】Program 3 : Complicated Expression(网易游戏笔试题)
描述
在lisp语言中,表达式都用前缀法表示,例如,1 + 2 在lisp中使用(+ 1 2)来表示,其中,表达式的括号是必需的。Bob 想实现一种类 lisp 的脚本语言,这个语言中的表达式是这样定义的:
1、每个表达式的形式都是 ( operator arg_1 arg_2 ... ),即由左括号,运算符,运算数,和右括号组成。
2、运算符包括三种,分别是'+', '-', '*'。
3、运算符如果可以接受一个运算数,就称为一元运算符,如果可以接受两个运算数,就称为二元运算符,如果可以接受三个及以上运算数,就称为多元运算符。
+ 可以是一元,二元,乃至多元运算符。
(+ 9) 表示正整数9;
(+ 1 2) 代表 1 + 2;
(+ 1 2 3 4) 代表 1, 2, 3, 4 的连加和。
- 可以是一元或者二元运算符。
(- 9) 表示整数-9;
(- 5 4) 代表 5 - 4。
* 可以是二元运算符或者多元运算符。
(* 2 3) 代表 2 * 3;
(* 1 2 3 4) 代表 1 * 2 * 3 * 4。
4、运算数可以是一个正整数,或者0,还可以是另外一个表达式,例如
(+ (* 2 3) 1) 也是一个合法的表达式,代表了 2 * 3 + 1
5、运算符与运算数之间,运算数与运算数相互之间都以空格进行分隔。在不产生歧义的情况下,空格也可以省略。例如,
(+ 2 3) 和 ( +2 3) 都被认为是合法的输入,且有相同的意义,代表 2 + 3。(+ 23)也是一个合法的输入,但其代表的是正整数23。
输入
输入由多行组成,第一行包含一个正整数T(T <= 5000)。代表共包含T组测试数据。
接下来会有T行输入。每一行包含一个表达式。这个表达式可能是一个合法的表达式,也可能是不合法的表达式。
每行数据所包含的字符数,不超过1000000。
输入数据中的数字全部是十进制正整数或者0,而且正整数的值不会大于65535。
输入中的表达式只包含三种不同类型的错误,除此之外的情况,都不必考虑。
第一类错误,括号不匹配,例如
(+ (* 2 3) 1)) 多了一个右括号;
+ 2 3 缺少括号等。
第二类错误,运算符缺失,例如
(2 3) 没有运算符。
第三类错误,运算符与数字不匹配,例如
(- 1 2 3),由于 - 只能接受最多两个算数,所以这也是一个不合法的表达式。
输出
对于合法的表达式,请输出表达式的值,对于不合法的表达式,请输出“invalid expression”。
可以假设最后的结果不超过32位整数的表示范围,但不保证在某些情况下,计算的中间值全部不超过32位整数的表示范围。
5 (+ 1 (* 2 3))) (2 3) (- 3 2 1) (+ (+ 1 2) (* 2 3) (- 2 1)) (- 2)
invalid expression
invalid expression
10
-2
#include "stdafx.h"
#include <string>
#include <stdio.h>
#include <stack>
#include <iostream>
using namespace std;
int pow( int n)
{
int result = 1 ;
if (n == 0 )
{
return 1 ;
}
else {
for ( int i = 0 ; i < n; ++i )
{
result *= 10 ;
}
}
return result ;
}
int main()
{
int n;
string str;
stack<char > temp;
cin >> n ;
cin.ignore (0x7fffffff, '\n');
while (n--)
{
stack <int> num;
bool hasnum = false;
getline (cin, str);
bool isexit = false;
if ( str[0 ] != '(' )
{
cout << "invalid expression" << endl;
}
else
{
for ( int i = 0 ; i < str.length (); ++ i)
{
if ( str[i ] == '(' )
{
if ( str[i + 1 ] != '+' && str[i + 1 ] != '-' && str[i + 1 ] != '*' ) {
cout << "invalid expression" << endl;
isexit = true ;
break;
}
else {
temp .push( str[i ]);
}
}
else if (str[i ] == ')' ) {
int multi = 0;
int count = 0;
int cnt = 0 ;
int subcount = 0;
bool isfirst = false;
int first = 0;
bool isback = false;
if ( temp.empty ())
{
cout << "invalid expression" << endl;
isexit = true ;
break;
}
while ( temp.size () > 0 && temp.top () != '(' )
{
if ( temp.top () >= '0' && temp.top () <= '9' )
{
hasnum = true ;
multi += ( temp.top () - '0' ) * pow(count );
count ++;
temp .pop();
}
else if (temp.top () == '#' )
{
if ( hasnum == true)
{
num .push( multi);
hasnum = false ;
multi = 0 ;
count = 0 ;
if ( isfirst == false)
{
first = num. top();
isfirst = true ;
}
}
temp .pop();
}
else if (temp.top () == '+' )
{
int result = 0;
if ( hasnum == true)
{
num .push( multi);
hasnum = false ;
multi = 0 ;
count = 0 ;
}
while ( num.top () != first && num .size() > 1)
{
result += num. top();
num .pop();
}
result += num. top();
num .pop();
num .push( result);
temp .pop();
isback = true ;
}
else if (temp.top () == '-' )
{
int result = 0;
if ( hasnum == true)
{
num .push( multi);
hasnum = false ;
multi = 0 ;
count = 0 ;
}
int a[ 2] = { 0 };
while ( num.top () != first && num .size() > 1)
{
subcount ++;
a [0] = num.top ();
num .pop();
}
if ( subcount == 1)
{
a [1] = num.top ();
num .pop();
if ( a[1] == first)
{
result = a[ 0] - a[ 1];
}
else
result = a[ 1] - a[ 0];
}
else if (subcount >= 2) {
cout << "invalid expression" << endl;
isexit = true ;
break;
}
else {
result = 0 - num.top ();
num .pop();
}
subcount = 0 ;
num .push( result);
temp .pop();
isback = true ;
}
else if (temp.top () == '*' )
{
int result = 1;
if ( hasnum == true)
{
num .push( multi);
hasnum = false ;
multi = 0 ;
count = 0 ;
}
if ( num.size ())
{
while ( num.top () != first&& num .size() > 1)
{
result *= num. top();
num .pop();
}
result *= num. top();
num .pop();
}
else
{
cout << "invalid expression" << endl;
isexit = true ;
break;
}
subcount = 0 ;
num .push( result);
temp .pop();
isback = true ;
}
}
temp .pop();
}
else {
if ( str[i ] == ' ' )
{
temp .push( '#');
}
else
temp .push( str[i ]);
}
}
}
if (! isexit)
{
if (! temp.empty ())
{
cout << "invalid expression" << endl;
break;
}
if ( num.size () == 1 )
{
cout << num. top() << endl;
}
}
}
return 0;
}
【面试笔试算法】Program 3 : Complicated Expression(网易游戏笔试题)的更多相关文章
- 【面试笔试算法】Program 2:Amusing Digits(网易游戏笔试题)
时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 网易成立于1997年6月,是中国领先的互联网技术公司.其相继推出了门户网站.在线游戏.电子邮箱.在线教育.电子商务等多种服 ...
- 【面试笔试算法】Program 5 : 推箱子 (网易游戏笔试题)
时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 推箱子是一款经典游戏.如图所示,灰色格子代表不能通过区域,蓝色方格是箱子,黑色圆形代表玩家,含有圆点的格子代表目标点. 规 ...
- 【面试笔试算法】Program 4 : Best Compression Algorithms(网易游戏笔试题)
时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 易信是由网易和电信联合开发的一款即时通讯软件.除了语音聊天,免费电话等新功能以外,传统的文字信息聊天功能也得以保留,因此每 ...
- 【面试笔试算法】Program 6: 字符消除(hiho题库)
时间限制:1000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi最近在玩一个字符消除游戏.给定一个只包含大写字母"ABC"的字符串s,消除过程是如下进行的: 1) ...
- 【面试笔试算法】Problem 1 : DP滑雪问题--网易互联网算法实习生2017笔试题
Description Michael喜欢滑雪百这并不奇怪,因为滑雪的确很刺激.可是 为了获得速度,滑的区域必须向下倾斜,而且当你滑到坡底,你不得不再次走上坡或者等待升降机来载你.Michael想知道 ...
- 【面试笔试算法】Problem 9: 腾讯2016年研发实习笔试题:最长回文子串
(一)题目 问题:求给定字符串s的回文(palindrome)子串中,长度最大的回文子串的长度. 回文(palindrome)是指从左往右读和从右往左读字符串,看到的字符串都是一样的.比如" ...
- 【面试笔试算法】牛客网一站通Offer编程题2016.4.19
牛客网一站通offer (一)字符串变形 1. 题目: 对于一个给定的字符串,我们需要在线性(也就是O(n))的时间里对它做一些变形.首先这个字符串中包含着一些空格,就像"Hello Wor ...
- 【面试笔试算法】Problem 8: 然而沼跃鱼早就看穿了一切(hiho题库)
时间限制:1000ms 单点时限:1000ms 内存限制:256MB 描述 fjxmlhx每天都在被沼跃鱼刷屏,因此他急切的找到了你希望你写一个程序屏蔽所有句子中的沼跃鱼("marshtom ...
- 【面试笔试算法】Problem 7: 补提交卡(hiho题库)
时间限制:2000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho给自己定了一个宏伟的目标:连续100天每天坚持在hihoCoder上提交一个程序.100天过去了,小Ho查看自己的提交 ...
随机推荐
- ROS机器人程序设计-学习小结-
ROS官网 |易科 |虞坤林 |古月居 |ROSClub 学习ROS相关书籍推荐:http://blog.csdn.net/zhangrelay/article/details/52244746 RO ...
- 使用Intent传递对象
Intent 的用法相信你已经比较熟悉了,我们可以借助它来启动活动.发送广播.启动服务等.在进行上述操作的时候,我们还可以在Intent 中添加一些附加数据,以达到传值的效果,比如在FirstActi ...
- FFmpeg的HEVC解码器源代码简单分析:环路滤波(Loop Filter)
===================================================== HEVC源代码分析文章列表: [解码 -libavcodec HEVC 解码器] FFmpe ...
- 3-sum问题
给定一个整数数组,判断能否从中找出3个数a.b.c,使得他们的和为0,如果能,请找出所有满足和为0个3个数对. #define SIZE 10 void judgeAndPut(int* arr, i ...
- EBS总账(GL)模块常用表
select * from gl_sets_of_books 总帐 select * from gl_code_combinations gcc wheregcc.summary_flag='Y ...
- linux配置java环境变量 转过几个,这个最详细和靠谱
一. 解压安装jdk 在shell终端下进入jdk-6u14-linux-i586.bin文件所在目录,之后会在当前目录下生成一个jdk1.6.0_14目录二. 需要配置的环境变量 1. PATH环境 ...
- java中小数的处理:高精度运算用bigDecimal类,精度保留方法,即舍入方式的指定
一. 计算机的小数计算一定范围内精确,超过范围只能取近似值: 计算机存储的浮点数受存储bit位数影响,只能保证一定范围内精准,超过bit范围的只能取近似值. java中各类型的精度范围参见:http: ...
- (一一八)利用block实现链式编程
有些时候需要不断地调用方法,如果使用传统方案,需要拿到对象多次调用,例如有一个Ball对象,实现了up.down.left.right四个方法,分别控制球的运动方向,如果要实现球向右再向下,需要这么写 ...
- 02Framelayout:帧布局
Framelayout:帧布局 >概念:每个组件都是一帧 当前子组件会覆盖前一个组件 >属性: android:layout_gravity=&quo ...
- Java与C之间的socket通信
最近正在开发一个基于指纹的音乐检索应用,算法部分已经完成,所以尝试做一个Android App.Android与服务器通信通常采用HTTP通信方式和Socket通信方式.由于对web服务器编程了解较少 ...