JavaScanner用法与构造器最佳实践
时间:2025-07-12 13:18:26 454浏览 收藏
大家好,今天本人给大家带来文章《Java Scanner规范用法与构造器最佳实践》,文中内容主要涉及到,如果你对文章方面的知识点感兴趣,那就请各位朋友继续看下去吧~希望能真正帮到你们,谢谢!
问题剖析:Scanner与重复输入
在Java编程中,将Scanner对象的实例化以及用户输入操作(如myObj.nextInt())直接放置在类成员变量的初始化表达式中,或者放置在实例初始化块(即不带名称的 {} 代码块)中,是一种常见的误用。当一个类被实例化时,其成员变量的初始化表达式和实例初始化块会在构造器执行之前被执行。这意味着,每当通过new关键字创建一个新的对象实例时,这些代码都会被重新执行一遍。
考虑以下初始代码示例:
import java.util.Scanner; public class test { Scanner myObj = new Scanner(System.in); // 成员变量初始化 {System.out.print("Enter Carat Value: ");} // 实例初始化块 int c = myObj.nextInt(); // 成员变量初始化 // ... 其他成员变量和计算 public static void main(String[] args) { test a = new test(); // 第一次实例化,执行上述代码 test b = new test(); // 第二次实例化,再次执行 test c = new test(); // 第三次实例化,再次执行 // ... 输出 } }
在上述代码中,Scanner myObj = new Scanner(System.in);、{System.out.print("Enter Carat Value: ");} 和 int c = myObj.nextInt(); 都属于实例初始化的一部分。因此,当main方法中连续创建a、b、c三个test对象时,这些初始化逻辑会被执行三次。这导致程序会提示用户输入“Enter Carat Value:”三次,并且每次都需要用户输入一个整数,这显然不是预期的行为。
不推荐的做法及其弊端
将I/O操作(如Scanner的读取和System.out.print的输出)直接放在类成员变量的初始化或实例初始化块中,通常被认为是不良的编程实践,主要有以下几点弊端:
- 行为不可预测性: 对象的创建过程变得复杂且带有副作用。每次创建新对象都会触发I/O,这在大型应用中可能导致难以调试的问题。
- 资源管理困难: Scanner是一个需要被关闭的资源。如果每次实例化都创建一个新的Scanner,并且没有在适当的时机关闭它们,可能导致资源泄露。
- 职责混淆: 类的成员变量初始化应该专注于为对象提供初始状态,而不是执行复杂的业务逻辑或I/O操作。将输入/输出逻辑混入其中,违反了单一职责原则。
- 测试困难: 带有I/O副作用的类难以进行单元测试,因为每次测试都需要模拟用户输入。
尽管可以通过将Scanner声明为static并在main方法中只创建一个对象来部分解决重复输入的问题,但这只是一个权宜之计,并没有从根本上解决将I/O逻辑与对象初始化混淆的问题,也未能充分利用面向对象的特性。
推荐方案:利用构造器进行初始化与资源管理
在Java中,最佳实践是将对象的初始化逻辑、用户输入以及相关的计算封装在类的构造器中。构造器是专门用于创建对象时执行初始化操作的方法。通过这种方式,我们可以确保初始化逻辑只在对象被创建时执行一次,并且能够更好地管理资源。
以下是采用构造器模式的优化方案:
import java.util.Scanner; public class Test { // 类名建议首字母大写 // 1. 声明类属性,不在此处进行复杂初始化 int c; double g, cg, gm, mg; // 2. 定义构造器,接收Scanner实例作为参数 public Test(Scanner myObj) { System.out.print("Enter Carat Value: "); // 在构造器中进行属性赋值和计算 this.c = myObj.nextInt(); // 使用this明确是当前对象的c this.g = 0.20; this.cg = this.c * this.g; this.gm = this.cg * 1000; this.mg = this.gm * 0.00220462 / 1000; // 也可以选择在构造器中直接输出结果,或者将输出逻辑放在其他方法中 System.out.println("Carats in Grams: " + this.cg); System.out.println("Grams in Milligrams: " + this.gm); System.out.println("Milligrams in Pounds: " + this.mg); } public static void main(String[] args) { // 3. 在main方法中创建并管理Scanner实例 Scanner myObj = new Scanner(System.in); // 4. 通过调用构造器创建对象,此时会执行构造器中的初始化逻辑 // 只需要创建一个Test对象,即可完成一次输入和计算 new Test(myObj); // 或者 Test result = new Test(myObj); 如果需要保存对象引用 // 5. 使用完毕后关闭Scanner,释放系统资源 myObj.close(); } }
代码解析与最佳实践
- 属性声明与初始化分离: 在类级别只声明成员变量(int c; double g, cg, gm, mg;),而将它们的具体赋值和计算逻辑放在构造器中。这使得类的结构更清晰。
- 构造器的作用: public Test(Scanner myObj) 是一个构造器。当main方法中执行new Test(myObj)时,这个构造器会被调用。构造器内部的代码会执行一次,包括提示用户输入、读取值以及进行所有相关的计算和输出。
- Scanner的传递与管理: Scanner实例在main方法中创建,并作为参数传递给Test类的构造器。这种方式确保了Scanner的生命周期由main方法控制,并且可以确保在程序结束时通过myObj.close()正确关闭,避免资源泄露。
- 单一职责: Test类负责封装与“克拉值转换”相关的属性和计算逻辑。main方法则负责程序的入口点、Scanner资源的创建和关闭,以及Test对象的实例化。职责分离使得代码更易于理解和维护。
- 避免在方法外部进行I/O操作: 始终将System.out.print、myObj.nextInt()等I/O操作封装在方法(如构造器、其他业务方法或main方法)内部,而不是直接作为类成员变量的初始化表达式或实例初始化块的一部分。
总结
正确管理Scanner实例和合理组织对象初始化逻辑是Java编程中的重要一环。通过将用户输入和对象属性的赋值逻辑封装在构造器中,并在main方法中统一管理Scanner的创建与关闭,可以有效避免重复输入、资源泄露等问题,同时提升代码的可读性、可维护性和专业性。遵循这些最佳实践,有助于编写出健壮且符合面向对象原则的Java应用程序。
本篇关于《JavaScanner用法与构造器最佳实践》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于文章的相关知识,请关注golang学习网公众号!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
270 收藏
-
311 收藏
-
407 收藏
-
224 收藏
-
423 收藏
-
331 收藏
-
191 收藏
-
390 收藏
-
342 收藏
-
122 收藏
-
192 收藏
-
105 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 511次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 498次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习