尝试使用 do while 循环捕获

时间:2021-05-25 16:05:28

标签: java try-catch java.util.scanner do-while

使用我的代码,我试图告诉用户不要输入字符串,直到用户输入一个整数,但在运行程序时它是无限的。

public static void main(String[] args) {

  int age = 1;
  Utilisateur utilisateur = new Utilisateur();
  Scanner u = new Scanner(System.in);

  System.out.println("Enter your Name: ");
  utilisateur.setNom(u.nextLine());
  System.out.println("Enter your Surname: ");
  utilisateur.setPrenom(u.nextLine());
  System.out.println("Enter your Matricule: ");
  utilisateur.setMatricule(u.nextLine());
  System.out.println("Enter your Sexe: ");
  utilisateur.setSexe(u.nextLine());

  do {
    try {
      System.out.println("Enter your Age: ");
      utilisateur.setAge(u.nextInt());
      System.out.println(utilisateur.detail());
      age = 2;
    } catch (Exception e) {
      System.out.println("Enter a valid age ");

    }
  }
  while (age == 1);
}
}

2 个答案:

答案 0 :(得分:0)

您应该在 catch 块中添加 u.nextLine(); 以跳过在扫描仪中输入的无效值。

答案 1 :(得分:0)

好的,让我们先清理一下代码。整个“年龄”变量有点奇怪。它似乎包含有关您是否已阅读该年龄的一些状态。但这有点布尔值,不是吗?因此,让我们考虑到这一点重做代码。我将首先将 do-while 更改为简单的 while,但我们可以稍后将其更改回来。此外,如果您更喜欢法语,最好将“u”重命名为“keyboard”或“clavier”。

    public static void main(String[] args) {
        Utilisateur utilisateur = new Utilisateur();
        Scanner clavier = new Scanner(System.in);

        System.out.println("Enter your Name: ");
        utilisateur.setNom(clavier.nextLine());
        System.out.println("Enter your Surname: ");
        utilisateur.setPrenom(clavier.nextLine());
        System.out.println("Enter your Matricule: ");
        utilisateur.setMatricule(clavier.nextLine());
        System.out.println("Enter your Sexe: ");
        utilisateur.setSexe(clavier.nextLine());

        boolean hasEnteredAge = false;
        while(!hasEnteredAge) {
                System.out.println("Enter your Age: ");
                String ageInput = clavier.nextLine().trim(); // remove leading and trailing whitespace. " 21 " becomes "21".
            try {
                int age = Integer.parseInt(ageInput);
                utilisateur.setAge(age);
                System.out.println(utilisateur);
                hasEnteredAge = true;
            } catch (Exception e) {
                System.out.println("Enter a valid age.");
            }
        }
    }
}

请注意,我将变量移动到循环的开头,这是我们需要了解这一事实的地方,以及我们如何将其初始化为 false。我们现在必须在之后将其设置为 true。

但我认为这里还有一些事情要做。我们有一堆印刷品,然后是输入。当然,这可以转化为一种方法,使这看起来更好一些?但在我们这样做之前,我们应该再看看循环。我们可以通过多种方式进行循环。我们可以

        do {
            System.out.println("Enter your Age: ");
            String ageInput = clavier.nextLine().trim(); // remove leading and trailing whitespace. " 21 " becomes "21".
            try {
                int age = Integer.parseInt(ageInput);
                utilisateur.setAge(age);
                System.out.println(utilisateur);
                break; // this means that we should exit the loop
            } catch (Exception e) {
                System.out.println("Enter a valid age.");
            }
        }while(true); // So if we ever get here, we're not done.

在这里,我们依靠休息让我们摆脱循环。这有效,但我个人不喜欢它。然而,这不是一个错误的事情,所以我将把它留在里面。你也可以像旧的 do-while 循环一样使用它:

boolean hasEnteredAge = false;
        do {
                System.out.println("Enter your Age: ");
                String ageInput = clavier.nextLine().trim(); // remove leading and trailing whitespace. " 21 " becomes "21".
            try {
                int age = Integer.parseInt(ageInput);
                utilisateur.setAge(age);
                System.out.println(utilisateur);
                hasEnteredAge = true;
            } catch (Exception e) {
                System.out.println("Enter a valid age.");
            }
        } while (!hasEnteredAge);

不管你选择哪个,都很好。 现在让我来解决打印线的问题并阅读:

如果您添加一个接受提示并返回字符串的方法“prompt”,您可以像这样轻松地简化它:

public class EnterNameHere {
    private static Scanner clavier = new Scanner(System.in);

    public static String prompt(String prompt) {
        System.out.println(prompt);
        return clavier.nextLine().trim();
    }
    // ... The rest is as before.
}

现在,阅读部分变得非常简单:

    public static void main(String[] args) {
        Utilisateur utilisateur = new Utilisateur();

        utilisateur.setNom(prompt("Enter your Name: "));
        utilisateur.setPrenom(prompt("Enter your surname: "));
        utilisateur.setMatricule(prompt("Enter your matricule: "));
        utilisateur.setSexe(prompt("Enter your sex: "));

一个重要的问题出现了:如果我们要对字符串输入这样做,为什么不对整数(int)输入也这样做呢? 我建议:

public static int promptInt(String prompt) {
        String value = prompt(prompt);
        try {
            return Integer.parseInt(value);
        } catch(NumberFormatException ignored) {
            System.out.println("Invalid number: '" + value + "'");
            return promptInt(prompt); // We try again!
        }
    }

请注意,如果您愿意,如果调用方法 promptInt 不起作用,我们会打印一条错误消息,然后再试一次。在它全部崩溃之前这只会工作几百次,但这应该足够了。 (如果您不希望发生这种情况,您当然可以采用之前的 while 循环方法。)方法或函数多次调用自身直到工作完成的技巧称为“递归”,它同样强大就像循环一样。刚接触编程的人可能会感到困惑,但我认为这个例子很简单。如果不是,您可以简单地替换提到的整个循环。当然,有一种方法叫prompt,另一种叫promptInt。为了避免混淆,我们将 prompt-method 重命名为 promptString,整个程序就变成了:

public class YourNameHere {
    private static final Scanner clavier = new Scanner(System.in);

    public static String promptString(String prompt) {
        System.out.print(prompt);
        return clavier.nextLine().trim();
    }

    public static int promptInt(String prompt) {
        String value = promptString(prompt);
        try {
            return Integer.parseInt(value);
        } catch(NumberFormatException ignored) {
            System.out.println("Invalid number: '" + value + "'");
            return promptInt(prompt); // We try again!
        }
    }

    public static void main(String[] args) {
        Utilisateur utilisateur = new Utilisateur();

        utilisateur.setNom(promptString("Enter your Name: "));
        utilisateur.setPrenom(promptString("Enter your surname: "));
        utilisateur.setMatricule(promptString("Enter your matricule: "));
        utilisateur.setSexe(promptString("Enter your sex: "));
        utilisateur.setAge(promptInt("Enter your age: "));

        System.out.println("You have created an utilisateur: " + utilisateur);
    }
}

当然还要加上Utilisateur的定义。 我认为这是一种更简单的方法,通过创建为您完成无聊工作的方法,您可以阅读 main 方法中的代码并立即了解发生了什么。需要了解的可以上去看看帮助提示方法。