コードをテストドライブするには、システムの入出力関数のラッパーを作成する必要があります。これは、依存性注入を使用して行うことができ、新しい整数を要求できるクラスを提供します。
public static class IntegerAsker {
private final Scanner scanner;
private final PrintStream out;
public IntegerAsker(InputStream in, PrintStream out) {
scanner = new Scanner(in);
this.out = out;
}
public int ask(String message) {
out.println(message);
return scanner.nextInt();
}
}
次に、モックフレームワーク(私はMockitoを使用)を使用して、関数のテストを作成できます。
@Test
public void getsIntegerWhenWithinBoundsOfOneToTen() throws Exception {
IntegerAsker asker = mock(IntegerAsker.class);
when(asker.ask(anyString())).thenReturn(3);
assertEquals(getBoundIntegerFromUser(asker), 3);
}
@Test
public void asksForNewIntegerWhenOutsideBoundsOfOneToTen() throws Exception {
IntegerAsker asker = mock(IntegerAsker.class);
when(asker.ask("Give a number between 1 and 10")).thenReturn(99);
when(asker.ask("Wrong number, try again.")).thenReturn(3);
getBoundIntegerFromUser(asker);
verify(asker).ask("Wrong number, try again.");
}
次に、テストに合格する関数を記述します。整数の要求/取得の重複を削除でき、実際のシステムコールがカプセル化されるため、関数ははるかにクリーンになります。
public static void main(String[] args) {
getBoundIntegerFromUser(new IntegerAsker(System.in, System.out));
}
public static int getBoundIntegerFromUser(IntegerAsker asker) {
int input = asker.ask("Give a number between 1 and 10");
while (input < 1 || input > 10)
input = asker.ask("Wrong number, try again.");
return input;
}
これはあなたの小さな例にとってはやり過ぎのように思えるかもしれませんが、このように開発しているより大きなアプリケーションを構築している場合は、かなり早く成果を上げることができます。