我已经完成了用Java编写一种命令行的任务。
这是为了测试编译器的各个部分。你看到一个可以输入类似“read_source”,“parse”,“build_ast”,“ast2cfg”,“print_cfg”等命令的promt。
Java中是否有任何库帮助我构建解释器(或repl?)。 我知道一个python模块可以满足我的需求:cmd
我自己写了类似的东西:
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.Arrays;
import java.util.HashMap;
interface Command {
public abstract void action(String[] parameters);
}
public class Repl {
private String prompt = "\n$ ";
private HashMap<String, Command> commands;
private Command nullCommand = null;
private Command defaultCommand = null;
public Repl() {
commands = new HashMap<String, Command>();
}
public void setPrompt(String prompt) {
this.prompt = prompt;
}
public void setDefaultCommand(Command defaultCommand) {
this.defaultCommand = defaultCommand;
}
public void setNullCommand(Command nullCommand) {
this.nullCommand = nullCommand;
}
public void addCommand(String name, Command command) {
commands.put(name, command);
}
public void runRepl() {
if (nullCommand == null) {
System.err.println("There is no 'nullCommand' specified");
return;
}
if (defaultCommand == null) {
System.err.println("There is no 'defaultCommand' specified");
return;
}
try {
BufferedReader reader = new BufferedReader(new InputStreamReader(
System.in));
String commandName;
String[] parameters;
while (true) {
System.out.print(prompt);
commandName = reader.readLine();
if (commandName == null) {
nullCommand.action(null);
} else {
parameters = commandName.trim().split("\\s+");
Command com = commands.get(parameters[0]);
if (com != null) {
com.action(parameters);
} else {
defaultCommand.action(parameters);
}
}
}
} catch (Exception e) {
e.printStackTrace();
System.err.println("Internal error within compiler: stopping compilation");
System.exit(1);
}
}
/**
* Exapmpe:
*/
public static void main(String[] args) {
Repl repl = new Repl();
repl.addCommand("printparams", new Command() {
@Override
public void action(String[] parameters) {
System.out.println(Arrays.toString(parameters));
}
});
repl.addCommand("add", new Command() {
@Override
public void action(String[] parameters) {
if (parameters.length == 3) { // {"add", "2", "3"}
try {
int x = Integer.parseInt(parameters[1]);
int y = Integer.parseInt(parameters[2]);
System.out.println("Ergebnis: " + (x + y));
} catch (NumberFormatException e) {
System.out.println("Arguments have to be integers");
}
} else {
System.out.println("There have to be two arguments");
}
}
});
Command helpcommand = new Command() {
@Override
public void action(String[] parameters) {
System.out.println("There is: 'help', 'printparams [PARAMS...]', 'exit' and 'add INTEGER INTEGER'");
}
};
repl.addCommand("help", helpcommand);
repl.setDefaultCommand(helpcommand);
Command exitcommand = new Command() {
@Override
public void action(String[] parameters) {
System.out.println("Bye!");
System.exit(0);
}
};
repl.addCommand("exit", exitcommand);
repl.setNullCommand(exitcommand);
repl.runRepl();
}
}
它可以工作,但是很好,例如,如果使用模式如“用法:添加INTEGER INTEGER [INTEGER ...]”用于添加两个以上的数字将根据参数的方式生成解析,以便添加新命令并确保它们始终保持一致的工作量较少。
此外,我问自己是否过度工程。你会建议只写这样的循环吗?:
while(true) {
String command = getUserInput();
if (command.equals("dothis")) {
dothis1();
someMember = dothis2();
} else if (command.equals("dothat"))
dothat();
} else {
printHelp();
}
}
答案 0 :(得分:0)
使用switch case而不是if else梯形图来创建更优雅的代码,并避免拼写错误。
例如:
while(true){
Scanner scan = new Scanner(System.in);
String s = scan.nextLine()
switch(s){
case "Dothis" :
break;
case "Dothat" :
break;
}
答案 1 :(得分:0)
您首先回答 看起来过于复杂。我使用了很多像你这样的寄存器代码,但通常这是一个类用另一个类注册自己的时候。这个类会注册自己的想法我只是出于性能原因,如果你有一个大数量的命令而且你担心if {} else if {}
链需要很长时间。
我认为简单的解决方案(尽管不是性感的)在这里是最好的。
while(true) {
String command = getUserInput();
if (command.equals("dothis")) {
dothis1();
} else if (command.equals("dothat"))
dothat();
}
...
}
我始终选择KISS方法,注意性能影响。复杂的代码不仅会让其他人试图维护您的代码,而且当您在3个月内回复它时,您也会感到头疼。 : - )