
本文详解如何在 public class 的 main 方法中成功实例化并执行另一个(非 public)顶层类中的交互式菜单逻辑,重点解决因缺少输入读取导致菜单“一闪而过”的常见新手问题。
在 Java 中,一个 .java 文件可包含一个 public 类(文件名必须与之相同)和若干默认访问权限(package-private)的顶层类——如示例中的 BankAccount。这类子类虽不可被其他包直接引用,但完全可在同一文件内的 main 类中自由实例化和调用。问题不在于“能否运行”,而在于菜单逻辑是否真正进入持续交互循环。
观察原代码的 menu() 方法,它仅打印了菜单标题(如 "a) Check Balance"),却未读取用户输入、也未实现后续分支逻辑或循环控制,导致程序输出欢迎信息后立即退出 JVM。修复的关键是:补全输入读取 + 添加主菜单循环 + 实现功能分发。
以下是修正后的完整可运行代码(已补全 menu() 并优化结构):
import java.util.Scanner;
public class BankApplication {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
System.out.println("Enter your 'Name' and 'CustomerId' to access your Bank account:");
String name = sc.nextLine();
String customerId = sc.nextLine();
BankAccount obj1 = new BankAccount(name, customerId);
obj1.menu(); // ✅ 正确调用:BankAccount 是同一文件中的合法顶层类
sc.close(); // 建议显式关闭 Scanner 避免资源泄漏
}
}
class BankAccount {
private double bal = 0.0; // 建议初始化,避免不确定值
private double prevTrans = 0.0;
private final String customerName; // 不可变字段更安全
private final String customerId;
BankAccount(String customerName, String customerId) {
this.customerName = customerName;
this.customerId = customerId;
}
void deposit(double amount) {
if (amount > 0) { // 改进:排除负数存款
bal += amount;
prevTrans = amount;
}
}
void withdraw(double amt) {
if (amt > 0 && bal >= amt) {
bal -= amt;
prevTrans = -amt;
} else if (bal < amt) {
System.out.println("Bank balance insufficient");
} else {
System.out.println("Invalid withdrawal amount");
}
}
void getPreviousTrans() {
if (prevTrans > 0) {
System.out.println("Deposited: " + prevTrans);
} else if (prevTrans < 0) {
System.out.println("Withdrawn: " + Math.abs(prevTrans));
} else {
System.out.println("No transaction occurred");
}
}
void menu() {
Scanner sc = new Scanner(System.in);
System.out.println("Welcome " + customerName);
System.out.println("Your ID: " + customerId);
System.out.println();
char option;
do {
System.out.println("=== BANK MENU ===");
System.out.println("a) Check Balance");
System.out.println("b) Deposit");
System.out.println("c) Withdraw");
System.out.println("d) Previous Transaction");
System.out.println("e) Exit");
System.out.print("Choose an option: ");
String input = sc.nextLine().trim();
if (input.length() == 0) continue;
option = input.charAt(0);
switch (option) {
case 'a':
System.out.println("Current Balance: " + bal);
break;
case 'b':
System.out.print("Enter deposit amount: ");
try {
double amount = Double.parseDouble(sc.nextLine());
deposit(amount);
} catch (NumberFormatException e) {
System.out.println("Invalid number format.");
}
break;
case 'c':
System.out.print("Enter withdrawal amount: ");
try {
double amount = Double.parseDouble(sc.nextLine());
withdraw(amount);
} catch (NumberFormatException e) {
System.out.println("Invalid number format.");
}
break;
case 'd':
getPreviousTrans();
break;
case 'e':
System.out.println("Thank you for banking with us!");
break;
default:
System.out.println("Invalid option! Please choose a–e.");
}
System.out.println(); // 空行分隔
} while (option != 'e');
sc.close();
}
}✅ 关键修复点总结:
立即学习“Java免费学习笔记(深入)”;
- 添加 do-while 循环:确保菜单持续运行,直到用户选择退出('e');
- 强制读取输入:使用 sc.nextLine() 获取用户键入,并通过 charAt(0) 提取首字符;
- 增强健壮性:加入 try-catch 处理非数字输入,避免 NumberFormatException 崩溃;
- 资源管理:在 menu() 和 main() 中分别关闭对应的 Scanner(注意:两个 Scanner 分别读取不同流,互不冲突);
- 代码规范:将 customerName/customerId 设为 final,bal/prevTrans 初始化为 0.0,提升可维护性。
⚠️ 注意事项:
- 同一文件中不可定义多个 public 类,但允许多个默认访问权限类(如 BankAccount),它们天然可被本文件内 public 类访问;
- 切勿在 menu() 中重复创建 new Scanner(System.in) 而不关闭——多次打开标准输入流虽不报错,但属不良实践;此处因作用域隔离(menu() 内 Scanner 在方法结束时关闭),是安全的;
- 若未来需将 BankAccount 拆分为独立文件,请将其声明为 public class BankAccount 并保存为 BankAccount.java,再通过 import 引入(当前单文件方案更契合初学者快速验证需求)。
掌握这一模式,你就能灵活组织多类协作的控制台应用——主类负责启动与注入依赖,业务类专注逻辑封装,这才是面向对象设计的起点。










