bankaccount类需在构造和方法中强制约束状态:balance初始化为0.0,withdraw校验金额正且余额充足,异常信息含具体余额;用double或bigdecimal存金额,accountnumber设为final,增减操作返回boolean。

BankAccount 类怎么设计才不踩空指针和余额负数的坑
核心是把账户状态约束写进构造和方法里,而不是靠调用方自觉。比如 balance 初始化必须明确为 0.0,不能依赖默认值;withdraw(double amount) 必须先校验 amount > 0 和 balance >= amount,否则转账时出现负余额或透支会污染后续计算。
常见错误现象:测试时 new BankAccount("1001") 后直接 withdraw(100) 报 IllegalArgumentException 却没提示具体原因——建议在异常信息里带上 "Insufficient balance: " + balance。
- 不要用
float存金额,一律用double或更稳妥的BigDecimal(小项目用double可接受,但别做精度敏感运算) -
accountNumber建议设为final,避免被意外修改 - 所有修改余额的方法(
deposit、withdraw)返回boolean表示是否成功,比抛异常更适合业务流程控制
Bank 类如何管理多个账户而不引发并发问题
小型系统不一定要上 ConcurrentHashMap,但如果测试时模拟多线程存取(比如两个线程同时对同一账户 deposit),用 HashMap 就会出 ConcurrentModificationException 或余额错乱。最轻量解法是给关键方法加 synchronized,比如 synchronized Account getAccount(String number) 和 synchronized boolean transfer(...)。
使用场景:命令行交互中一般单线程,但如果你后续想加个简易 HTTP 接口(比如用 SparkJava),并发就立刻成为现实问题。
立即学习“Java免费学习笔记(深入)”;
- 账户存储结构选
Map<string bankaccount></string>,key 用accountNumber,查得快且自然去重 - 别在
Bank构造器里预置测试账户,放到独立的initSampleData()方法里,方便单元测试隔离 -
transfer(String from, String to, double amount)必须是原子操作:先查双账户、再校验余额、再扣减与增加——中间任何一步失败都要回滚(即不改变任一账户状态)
main 方法怎么组织才能快速验证逻辑又不写死一堆 new
别把所有测试逻辑塞进 main:它只负责启动和调度。真实做法是抽一个 BankConsoleApp 类,里面用 Scanner 解析用户输入(如 “deposit 1001 500”),再委托给 Bank 实例执行。这样 main 就只剩三行:Bank bank = new Bank();、BankConsoleApp app = new BankConsoleApp(bank);、app.run();。
容易踩的坑:用 Scanner.nextLine() 读数字前如果刚用过 nextInt(),会跳过下一行——统一用 nextLine() 再转 Double.parseDouble() 更稳。
- 支持的命令至少包括:
create <number></number>、deposit <number><amount></amount></number>、withdraw <number><amount></amount></number>、balance <number></number>、transfer <from><to><amount></amount></to></from> - 每条命令执行后打印结果,比如 “OK” 或 “ERROR: Insufficient balance” ——别只抛异常却不输出
- 输入解析用
String.split("\s+")即可,不用上正则,小项目够用
项目结构怎么分包才不会后期改到崩溃
就三个包足够:model(放 BankAccount)、service(放 Bank)、ui(放 BankConsoleApp 和 main)。别建 util 或 exception 包——小项目里自定义异常(如 InsufficientBalanceException)直接扔 model 下就行,NumberUtils 这类工具类压根不需要,Double.parseDouble() 已经够用。
路径示例:src/main/java/com/example/bank/model/BankAccount.java,src/main/java/com/example/bank/service/Bank.java。
- 类名全部首字母大写,文件名严格匹配,IDE 通常会帮你检查
- 不要为了“规范”加 Lombok 或 Spring Boot,
@Getter看似省事,但新人看不懂生成的代码,debug 时还多一层间接 - 如果后续要导出 jar,确保
MANIFEST.MF里Main-Class指向ui.BankConsoleApp,而不是写在main所在类里










