Java中Scanner与对象实例化:避免重复输入和提升代码结构

Java中Scanner与对象实例化:避免重复输入和提升代码结构

本文旨在探讨在Java中将Scanner对象和输入逻辑放置在类字段初始化器中时,因多次创建对象而导致重复输入的问题。文章将详细阐述其原因,并提供最佳实践,包括使用构造方法进行对象初始化、合理管理Scanner的生命周期,以及区分实例初始化与静态初始化的重要性,从而帮助开发者编写更健壮、可维护的代码。

1. 问题背景:实例初始化与重复输入

在Java中,直接在类定义中声明并初始化非Static的字段(也称为实例初始化器或字段初始化器),或者使用实例初始化块({}代码块),这些代码会在每次创建该类的新实例时执行。当Scanner对象的创建和输入操作被放置在这些位置时,就会导致每次实例化对象时都重复进行输入提示。

考虑以下示例代码,它展示了这种常见的问题模式:

package galitkami;  import java.util.Scanner;  public class test {      Scanner myObj = new Scanner(System.in); // 实例字段初始化     {System.out.print("Enter Carat Value: ");} // 实例初始化块     int c = myObj.nextInt(); // 实例字段初始化,执行输入      double g = 0.20;     double cg = c*g;     double gm = cg*1000;     double mg = gm*0.00220462/1000;      public static void main(String[] args) {         test a = new test(); // 第一次实例化,执行一次输入         test b = new test(); // 第二次实例化,执行第二次输入         test c = new test(); // 第三次实例化,执行第三次输入          System.out.println("Carats in Grams: "+a.cg);         System.out.println("Grams in Milligrams: "+b.gm);         System.out.println("Milligrams in Pounds: "+c.mg);     } }

在上述代码中,Scanner myObj = new Scanner(System.in);、{System.out.print(“Enter Carat Value: “);} 和 int c = myObj.nextInt(); 都属于实例初始化的一部分。当main方法中执行new test()三次时,这些初始化逻辑会被重复执行三次,从而导致程序三次提示用户输入。这不仅造成了不必要的重复操作,也违背了程序设计的直观逻辑。

2. 最佳实践:使用构造方法与合理管理Scanner

为了解决上述问题并遵循良好的编程实践,我们应该将对象的初始化逻辑,特别是涉及用户输入和资源(如Scanner)管理的逻辑,放置在更合适的位置。

立即学习Java免费学习笔记(深入)”;

2.1 构造方法(constructor)的应用

构造方法是Java中用于初始化新创建对象状态的特殊方法。它是执行对象创建后立即运行的代码块,是设置对象初始属性值的理想场所。通过将输入逻辑放入构造方法,我们可以更精确地控制何时进行输入。

import java.util.Scanner;  public class Test { // 建议类名首字母大写      // 声明类属性,但不在此处初始化,等待构造方法或特定方法进行初始化     int c;     double g, cg, gm, mg;      // 类构造方法:用于初始化Test对象     // 接受一个Scanner对象作为参数,避免在构造方法内部重复创建Scanner     public Test(Scanner myObj) {         System.out.print("Enter Carat Value: ");         c = myObj.nextInt(); // 在构造方法中获取输入          // 根据输入值计算其他属性         g = 0.20;         cg = c * g;         gm = cg * 1000;         mg = gm * 0.00220462 / 1000;          // 可以在构造方法中直接打印结果,或者在main方法中访问对象属性后打印         System.out.println("Carats in Grams: " + cg);         System.out.println("Grams in Milligrams: " + gm);         System.out.println("Milligrams in Pounds: " + mg);     }      public static void main(String[] args) {         // 在main方法中创建并管理唯一的Scanner实例         Scanner myObj = new Scanner(System.in);          // 通过调用构造方法创建Test对象,此时构造方法中的逻辑(包括输入)会被执行         // 如果只需要一次输入并计算,只需创建一个Test对象         new Test(myObj); // 创建一个Test对象并执行其构造方法          // 如果需要创建多个Test对象,且每个对象都需要独立输入,则可以多次调用构造方法         // Test test1 = new Test(myObj);         // Test test2 = new Test(myObj); // 这将再次提示输入          // 重要:使用完毕后关闭Scanner,释放系统资源         myObj.close();     } }

代码解释:

  • 属性声明与初始化分离: 类字段c, g, cg, gm, mg 仅被声明,它们的赋值操作被移到了构造方法中。
  • 构造方法参数: Test(Scanner myObj) 构造方法接受一个Scanner实例作为参数。这意味着Scanner对象是在main方法中创建并管理的,确保了只有一个Scanner实例用于整个程序的输入。
  • 输入与计算逻辑: 用户输入和基于输入进行计算的逻辑都被封装在构造方法中。当main方法中调用new Test(myObj)时,这些逻辑只执行一次。
  • 资源管理: Scanner对象在main方法中创建,并在使用完毕后通过myObj.close()关闭,这是良好的资源管理习惯,避免资源泄露。

2.2 静态Scanner的考量

在某些情况下,如果确实需要Scanner在整个应用程序生命周期内只被初始化一次,并且其输入不与特定对象实例绑定,可以将其声明为static。

import java.util.Scanner;  public class TestStaticScanner {      static Scanner myObj = new Scanner(System.in); // 静态字段初始化,只执行一次     // 静态初始化块,只在类加载时执行一次     static {         System.out.print("Enter Carat Value (Static): ");     }     final int c = myObj.nextInt(); // 实例字段初始化,但myObj已在静态初始化时准备好      double g = 0.20;     double cg = c * g;     double gm = cg * 1000;     double mg = gm * 0.00220462 / 1000;      public static void main(String[] args) {         // 即使创建多个TestStaticScanner对象,输入也只发生一次         TestStaticScanner a = new TestStaticScanner();         TestStaticScanner b = new TestStaticScanner(); // 不会再次提示输入         TestStaticScanner c = new TestStaticScanner(); // 不会再次提示输入          System.out.println("Carats in Grams: " + a.cg);         System.out.println("Grams in Milligrams: " + a.gm);         System.out.println("Milligrams in Pounds: " + a.mg);          myObj.close(); // 关闭静态Scanner     } }

注意:

  • static的特性: static成员属于类本身,而不是类的某个实例。static字段和static初始化块只在类加载到jvm时执行一次。因此,即使创建多个TestStaticScanner对象,myObj和nextInt()也只被调用一次。
  • 局限性: 这种方法虽然解决了重复输入的问题,但它意味着所有TestStaticScanner实例都将共享同一个c值(因为c在第一个实例被创建时就已经确定),这可能不符合每个对象需要独立输入的场景。如果每个对象需要不同的输入,则构造方法是更优的选择。
  • final关键字: 在final int c = myObj.nextInt();中,final确保c一旦被赋值就不能更改。

3. 总结与最佳实践建议

为了编写健壮、可维护的Java代码,请遵循以下建议:

  1. 避免在实例初始化器中执行I/O操作: 避免将Scanner的创建、输入提示或数据读取等操作直接放在非static的字段初始化或实例初始化块中。这些代码会在每次创建对象时执行,容易导致意外行为。
  2. 使用构造方法进行对象初始化: 将对象状态的初始化逻辑(包括获取用户输入来设置初始值)封装在构造方法中。这样可以清晰地控制对象创建时的行为。
  3. 统一管理Scanner实例: 最好在程序的入口点(如main方法)创建并管理一个Scanner实例。如果需要将Scanner传递给其他方法或构造方法,通过参数传递,而不是在每个需要输入的地方都创建新的Scanner。
  4. 关闭Scanner资源: 在Scanner使用完毕后,务必调用close()方法关闭它,释放底层系统资源(如输入流)。这通常在main方法的末尾或使用try-with-resources语句完成。
  5. 理解static与实例成员的区别 static成员属于类,只加载一次;实例成员属于对象,每次创建对象时都会初始化。根据需求选择合适的修饰符。

通过采纳这些实践,您可以有效地管理Java中的输入流,避免常见的陷阱,并构建结构更清晰、行为更可预测的应用程序。

© 版权声明
THE END
喜欢就支持一下吧
点赞9 分享