我正在制作text adventure in C#,有人建议我使用dispatch table代替切换语句。
这是switch语句代码:
#region Public Methods
public static void Do(string aString)
{
if(aString == "")
return;
string verb = "";
string noun = "";
if (aString.IndexOf(" ") > 0)
{
string[] temp = aString.Split(new char[] {' '}, 2);
verb = temp[0].ToLower();
noun = temp[1].ToLower();
}
else
{
verb = aString.ToLower();
}
switch(Program.GameState)
{
case Program.GameStates.Playing:
if (IsValidInput(Commands, verb, true))
{
switch(verb) //this is the switch statement
{
case "help":
case "?":
WriteCommands();
break;
case "exit":
case "quit":
Program.GameState = Program.GameStates.Quit;
break;
case "move":
case "go":
MoveTo(noun);
break;
case "examine":
Examine(noun);
break;
case "take":
case "pickup":
Pickup(noun);
break;
case "drop":
case "place":
Place(noun);
break;
case "use":
Use(noun);
break;
case "items":
case "inventory":
case "inv":
DisplayInventory();
break;
case "attack":
//attack command
break;
}
}
break;
case Program.GameStates.Battle:
if(IsValidInput(BattleCommands, verb, true))
{
switch(verb) //this is the other switch statement
{
case "attack":
//attack command
break;
case "flee":
case "escape":
//flee command
break;
case "use":
//use command
break;
case "items":
case "inventory":
case "inv":
//items command
break;
}
}
break;
}
}
#endregion
如何重构此操作以使用调度表?
答案 0 :(得分:12)
最简单的方法是使用委托字典。
例如:
Dictionary<string, Action> dispatch = new Dictionary<string, Action>();
dispatch["help"] = new Action(() => Console.WriteLine("Hello"));
dispatch["dosomething"] = new Action(() =>
{
// Do something else
Console.WriteLine("Do Something");
});
// Call the 'help' command
dispatch["help"]();
对于多个不同的参数,最简单的方法是使用基本委托并使用动态调用。
Dictionary<string, Delegate> dispatch = new Dictionary<string, Delegate>();
dispatch["help"] = new Action(() => Console.WriteLine("Hello"));
dispatch["dosomething"] = new Action<string>(s => Console.WriteLine(s));
dispatch["help"].DynamicInvoke();
dispatch["dosomething"].DynamicInvoke("World");
如果使用.NET 4,您还可以在运行时使用动态类型来解决,以减少动态调用的混乱。
Dictionary<string, dynamic> dispatch = new Dictionary<string, dynamic>();
dispatch["help"] = new Action(() => Console.WriteLine("Hello"));
dispatch["dosomething"] = new Action<string>(s => Console.WriteLine(s));
dispatch["help"]();
dispatch["dosomething"]("World");
答案 1 :(得分:1)
也许他指的是'Double Dispatch'或Visitor Pattern。
您可以拆分代码以改进更多“调度程序”样式,如下所示:
public interface IGameState{
void Help();
void Question();
void Attack();
}
public interface ICommand{
bool IsValidFor(PlayingState state);
bool IsValidFor(BattleState state);
void Execute(IGameState state);
}
public class PlayingState : IGameState {
public void Help(){ // Do Nothing }
public void Question() { WriteCommands(); }
private void WriteCommands(){ }
}
public class Battle : IGameState{
public void Help(){ // Do Nothing }
public void Question() { WriteCommands(); }
public void Attack() { Roll(7); }
private void Roll(int numDice){ }
}
public class CommandBuilder{
public ICommand Parse(string verb){
switch(verb){
case "help":
return new HelpCommand();
case "?":
return new QuestionCommand();
case "attack":
return new AttackCommand();
default:
return new UnknownCommand();
}
}
}
public class QuestionCommand(){
bool IsValidFor(PlayingState state){
return true;
}
bool IsValidFor(BattleState state){
return false;
}
void Execute(IGameState state){
state.Question();
}
}
public static void Do(string aString){
var command = CommandBuilder.Parse(aString);
if(command.IsValidFor(Program.GameStates))
command.Execute(Program.Gamestates);
}
答案 2 :(得分:1)
收回@Jon Ericson的介绍:
调度表是一种数据结构,它将索引(或键,读取@pst的注释)值与操作相关联。对于switch类型的语句来说,它是一个相当优雅的替代品。
关于实施部分,请查看this question,尤其是this answer,其中恕我直言似乎非常正确,但仍然易于理解。