ThoughtWorks代码挑战——FizzBuzzWhizz
很久没发表过文章了,今天看到一篇文章 最难面试的IT公司之ThoughtWorks代码挑战——FizzBuzzWhizz游戏(C#解法)
看到LZ的2B青年代码,实在是惨不忍睹,故写篇文章来探讨下这类问题的一般思考。
原题:
FizzBuzzWhizz
你是一名体育老师,在某次课距离下课还有五分钟时,你决定搞一个游戏。此时有100名学生在上课。游戏的规则是:
1. 你首先说出三个不同的特殊数,要求必须是个位数,比如3、5、7。
2. 让所有学生拍成一队,然后按顺序报数。
3. 学生报数时,如果所报数字是第一个特殊数(3)的倍数,那么不能说该数字,而要说Fizz;如果所报数字是第二个特殊数(5)的倍数,那么要说Buzz;如果所报数字是第三个特殊数(7)的倍数,那么要说Whizz。
4. 学生报数时,如果所报数字同时是两个特殊数的倍数情况下,也要特殊处理,比如第一个特殊数和第二个特殊数的倍数,那么不能说该数字,而是要说FizzBuzz, 以此类推。如果同时是三个特殊数的倍数,那么要说FizzBuzzWhizz。
5. 学生报数时,如果所报数字包含了第一个特殊数,那么也不能说该数字,而是要说相应的单词,比如本例中第一个特殊数是3,那么要报13的同学应该说Fizz。如果数字中包含了第一个特殊数,那么忽略规则3和规则4,比如要报35的同学只报Fizz,不报BuzzWhizz。
一道看似简单的题目,其实并没有那么简单,如果你直接写的话,那估计就是:
if, if , else if , if , for, [0] [1] [2]….
我们来理解下这道题:
1. 你首先说出三个不同的特殊数,要求必须是个位数,比如3、5、7。
不同的三个数,必须是个位数,这些都是验证条件,你注意到了吗?
2. 让所有学生拍成一队,然后按顺序报数。
生成顺序的数字。
3. 学生报数时,如果所报数字是第一个特殊数(3)的倍数,那么不能说该数字,而要说Fizz;如果所报数字是第二个特殊数(5)的倍数,那么要说Buzz;如果所报数字是第三个特殊数(7)的倍数,那么要说Whizz。
规则:如果是某个特殊数的倍数,输出对应的值,否则输出数字。
4. 学生报数时,如果所报数字同时是两个特殊数的倍数情况下,也要特殊处理,比如第一个特殊数和第二个特殊数的倍数,那么不能说该数字,而是要说FizzBuzz, 以此类推。如果同时是三个特殊数的倍数,那么要说FizzBuzzWhizz。
规则:如果是多个特殊数的倍数,输出所有的对应值。
5. 学生报数时,如果所报数字包含了第一个特殊数,那么也不能说该数字,而是要说相应的单词,比如本例中第一个特殊数是3,那么要报13的同学应该说Fizz。如果数字中包含了第一个特殊数,那么忽略规则3和规则4,比如要报35的同学只报Fizz,不报BuzzWhizz。
规则:如果包含第一个特殊数字,则只输出第一个特殊数字所对应的值。
OK,思考下我们该怎样做?。。
我们来抽象的理解下题目: “给你输入一堆数字,然后你根据一定的规则进行parse,然后输出parse 的结果。”
所以这道题目想考察的是你如何定义这些规则,如何应用这些规则,该如何parse呢?
让我们看下规则Rule:
Rule,有优先级,然后可以对输入进行Parse,然后Parse又需要一个对应的字典。
所以Rule 像这样:
abstract class Rule
{
public abstract int Priority { get; } public Dictionary<int, string> SpecialDictionary { get; set; } public Rule(Dictionary<int, string> specialDictionary)
{
this.SpecialDictionary = specialDictionary;
} public bool ParseNum(int num, ref string result)
{
if ((SpecialDictionary != null) && (SpecialDictionary.Count > ))
{
return ParseNumCore(num, ref result);
}
else
{
return false;
}
} protected abstract bool ParseNumCore(int num, ref string result);
}
接着Rule3: 如果是某个特殊数的倍数,输出对应的值,否则输出数字,输出数字我放到最外层去处理了,当然如果需要也可以写个Rule2.
class Rule3 : Rule
{
public Rule3(Dictionary<int, string> specialDictionary)
: base(specialDictionary)
{
} public override int Priority
{
get { return ; }
} protected override bool ParseNumCore(int num, ref string result)
{
foreach (var special in SpecialDictionary)
{
if (num % special.Key == )
{
result = special.Value;
return true;
}
} return false;
}
}
Rule4:如果是多个特殊数的倍数,输出所有的对应值。
class Rule4 : Rule
{
public Rule4(Dictionary<int, string> specialDictionary)
: base(specialDictionary)
{
} public override int Priority
{
get { return ; }
} protected override bool ParseNumCore(int num, ref string result)
{
List<string> matches = new List<string>(); foreach (var special in SpecialDictionary)
{
if (num % special.Key == )
{
matches.Add(special.Value);
}
} if (matches.Count > )
{
result = string.Join("", matches);
return true;
}
else
{
return false;
}
}
}
Rule5:如果包含第一个特殊数字,则只输出第一个特殊数字所对应的值。
class Rule5 : Rule
{
public Rule5(Dictionary<int, string> specialDictionary)
: base(specialDictionary)
{
} public override int Priority
{
get { return ; }
} protected override bool ParseNumCore(int num, ref string result)
{
if (SpecialDictionary.Count > )
{
var firstSpecial = SpecialDictionary.First();
if (num.ToString().Contains(firstSpecial.Key.ToString()))
{
result = firstSpecial.Value;
return true;
}
} return false;
}
}
接下来:最重要的就是Parse 逻辑了,想一想应该怎样调用这些Rule呢(visitor ?):
foreach (var student in studentNums)
{
string parseResult = student.ToString();
foreach (Rule rule in rules)
{
if (rule.ParseNum(student, ref parseResult))
{
break;
}
} Console.WriteLine(parseResult);
}
下面是完整的代码:
private static void FizzBuzz()
{
bool isValidInput = false;
do
{
Console.WriteLine("please input three numbers which is units digit, use ',' division ");
string[] inputNums = Console.ReadLine().Split(','); if (ValidSpecialInput(inputNums))
{
isValidInput = true; // create special dictionary to parse the students nums.
Dictionary<int, string> special = new Dictionary<int, string>();
special.Add(Int32.Parse(inputNums[]), "Fizz");
special.Add(Int32.Parse(inputNums[]), "Buzz");
special.Add(Int32.Parse(inputNums[]), "Whizz"); // get students nums.
int studentsCount = ;
var studentNums = Enumerable.Range(, studentsCount); // create rules to parse.
var rules = new List<Rule>()
{
new Rule5(special),
new Rule4(special),
new Rule3(special),
}.OrderByDescending(r => r.Priority); // parse logic.
foreach (var student in studentNums)
{
string parseResult = student.ToString();
foreach (Rule rule in rules)
{
if (rule.ParseNum(student, ref parseResult))
{
break;
}
} Console.WriteLine(parseResult);
} Console.ReadLine();
}
else
{
Console.WriteLine("the input is not valid.");
}
}
while (isValidInput == false);
} private static bool ValidSpecialInput(string[] specialInputs)
{
bool result = false;
if (specialInputs.Length == )
{
return specialInputs.All(input =>
{
int num = ;
return Int32.TryParse(input, out num) && (num > ) && (num < );
});
} return result;
}
一些后续思考:
1:如果输入的不是三个,而是4个,5个 special, 应该怎么改?
2:如果学生数量不是100个,是1000个?
3:如果有限考虑Rule3,然后是Rule4,Rule5,应该怎么改?
4:如果还有另一个限制条件:比如如果数字是素数,把对应的值按反序输出,如何处理?
5:如果输入不是数字,而是字符串,应该如何处理?
完整源码下载:http://files.cnblogs.com/LoveJenny/FizzBuzz.7z
ThoughtWorks代码挑战——FizzBuzzWhizz的更多相关文章
- ThoughtWorks代码挑战——FizzBuzzWhizz游戏 通用高速版(C/C++ & C#)
最早看到这个题目是从@ 程序媛想事儿(Alexia) 的 最难面试的IT公司之ThoughtWorks代码挑战——FizzBuzzWhizz游戏 开始的,然后这几天陆陆续续有N个小伙伴发表了自己的文章 ...
- 最难面试的IT公司之ThoughtWorks代码挑战——FizzBuzzWhizz游戏
最近互联网招聘平台拉勾网在五一期间推出个“最难面试的IT公司”代码挑战活动,评选出了5个最难面试的IT公司,即:ThoughtWorks.Google.Unisys.Rackspace.Cypress ...
- 最艰难的采访IT公司ThoughtWorks代码挑战——FizzBuzzWhizz游戏
最近的互联网招聘平台拉勾网在五月推出了"最艰难的采访IT公司"码挑战活动,评选出了5个最难面试的IT公司,即:ThoughtWorks.Google.Unisys.Rackspac ...
- 最难面试的IT公司之ThoughtWorks代码挑战——FizzBuzzWhizz游戏(C#解法)
原题 看到那么多人看到前面这么糟粕的代码各种不忍直视后,楼主还是把最终实现放在页首吧. Console.WriteLine("说出三个不同的特殊数,用','隔开 &q ...
- 原生js代码挑战之动态添加双色球
var ballArr = []; //存放已有的红球,用来排除重复和排序window.onload = function(){ var btn = document.createElement(&q ...
- 关于今天很热的--FizzBuzzWhizz
今天早上到现在看到了3篇关于FizzBuzzWhizz的问题,第一篇是@程序媛想事儿(Alexia)[最难面试的IT公司之ThoughtWorks代码挑战--FizzBuzzWhizz游戏]其实题目不 ...
- 2014年5月份第1周51Aspx源码发布详情
郑州某高校学生考评系统源码 2014-5-5 [VS2008]功能介绍: 1.用户角色有部主任.教师.学生等. 2.可添加班级考评项目.学生考评项目. 3.可指定学生对班级.学生某考评项 ...
- 编译器是如何实现32位整型的常量整数除法优化的?[C/C++]
引子 在我之前的一篇文章[ ThoughtWorks代码挑战——FizzBuzzWhizz游戏 通用高速版(C/C++ & C#) ]里曾经提到过编译器在处理除数为常数的除法时,是有优化的,今 ...
- ThoughtWorks 面试备忘录
ThoughtWorks 面试备忘录 前言 前段时间 ThoughtWorks 在网上和拉勾网合作搞了一次网络招聘,名为抛弃简历!让代码说话!,可谓赚足了眼球,很多程序猿纷纷摩拳擦掌.踊跃提交代码,在 ...
随机推荐
- ubuntu svn
http://blog.csdn.net/neutrojan/article/details/37659747
- mac个人设置
修改spotlight快捷键 mac默认的command+space和我windows下的习惯冲突,修改为ctrl+space 删除输入法切换的快捷键 因为我不需要切换不同语言的快捷键.中英文切换直接 ...
- 不注册COM在Richedit中使OLE支持复制粘贴
正常情况下在Richedit中使用OLE,如果需要OLE支持复制粘贴,那么这个OLE对象必须是已经注册的COM对象. 注册COM很简单,关键问题在于注册时需要管理员权限,这样一来,如果希望APP做成绿 ...
- C#: using JsonReader avoid Deserialize Json to dynamic
namespace Test { using Microshaoft; using Test.Models; using Newtonsoft.Json; using System; using Sy ...
- HDU 5945 / BestCoder Round #89 1002 Fxx and game 单调队列优化DP
Fxx and game 问题描述 青年理论计算机科学家Fxx给的学生设计了一款数字游戏. 一开始你将会得到一个数\:XX,每次游戏将给定两个参数\:k,tk,t, 任意时刻你可以对你的数执行下面 ...
- 1.Android 视图及View绘制分析笔记之setContentView
自从1983年第一台图形用户界面的个人电脑问世以来,几乎所有的PC操作系统都支持可视化操作,Android也不例外.对于所有Android Developer来说,我们接触最多的控件就是View.通常 ...
- 从零开始---控制台用c写俄罗斯方块游戏(2)
上回说到下移的问题,这篇就说一下刷新的问题 我们控制台输出一般都是一行一行的输出,所以,在输出屏幕的时候,我们一个画面闪到另一个画面的效果 我刚开始弄的是用system("CLS" ...
- javascript系列:NaN类型
NaN,即非数值是一个特殊的数值,这个数值用于表示一个本来要返回数值的操作数未返回数值的情况下. ECMAScript中任何数值除以0会返回NaN,因此不影响其他代码运行. NaN特点: ...
- 运行jar应用程序引用其他jar包的四种方法
转载地址:http://www.iteye.com/topic/332580 大家都知道一个java应用项目可以打包成一个jar,当然你必须指定一个拥有main函数的main class作为你这个ja ...
- dotnetcore 单元测试
dotnetcore的单元测试目前支持的比较好的是xunit,首先通过nuget添加组件dotnet-test-xunit 和 xunit.如果有依赖注入可在构造方法中,相当于Nunit中的[Setu ...