登录
首页 >  文章 >  java教程

Java主类调用子类菜单的正确方式

时间:2026-03-02 09:39:38 488浏览 收藏

本文深入剖析了Java中主类调用同一文件内非public子类菜单时“一闪而过”的根本原因——并非访问权限问题,而是菜单逻辑缺失输入读取、循环控制和功能分发;通过补全do-while交互循环、安全读取用户输入、结构化switch分支、异常处理及资源管理,手把手带你将一个静态打印的菜单升级为稳定可用的银行账户控制台应用,同时厘清单文件多类协作的底层规则与面向对象设计的最佳实践起点。

如何在 Java 主类中正确调用并运行子类的交互式菜单

本文详解如何在 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();
    }
}

关键修复点总结:

  • 添加 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 引入(当前单文件方案更契合初学者快速验证需求)。

掌握这一模式,你就能灵活组织多类协作的控制台应用——主类负责启动与注入依赖,业务类专注逻辑封装,这才是面向对象设计的起点。

以上就是《Java主类调用子类菜单的正确方式》的详细内容,更多关于的资料请关注golang学习网公众号!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>